diff --git a/.eslintignore b/.eslintignore
index c4f7298047..e453170087 100644
--- a/.eslintignore
+++ b/.eslintignore
@@ -1,4 +1,4 @@
src/component-index.js
test/end-to-end-tests/node_modules/
-test/end-to-end-tests/riot/
+test/end-to-end-tests/element/
test/end-to-end-tests/synapse/
diff --git a/.eslintignore.errorfiles b/.eslintignore.errorfiles
index db90d26ba7..1c0a3d1254 100644
--- a/.eslintignore.errorfiles
+++ b/.eslintignore.errorfiles
@@ -12,5 +12,5 @@ test/components/views/dialogs/InteractiveAuthDialog-test.js
test/mock-clock.js
src/component-index.js
test/end-to-end-tests/node_modules/
-test/end-to-end-tests/riot/
+test/end-to-end-tests/element/
test/end-to-end-tests/synapse/
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 5aac4e2974..151888a17e 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,86 @@
+Changes in [3.10.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.10.0) (2020-12-07)
+=====================================================================================================
+[Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.10.0-rc.1...v3.10.0)
+
+ * Upgrade to JS SDK 9.3.0
+
+Changes in [3.10.0-rc.1](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.10.0-rc.1) (2020-12-02)
+===============================================================================================================
+[Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.9.0...v3.10.0-rc.1)
+
+ * Upgrade to JS SDK 9.3.0-rc.1
+ * Translations update from Weblate
+ [\#5461](https://github.com/matrix-org/matrix-react-sdk/pull/5461)
+ * Fix VoIP call plinth on dark theme
+ [\#5460](https://github.com/matrix-org/matrix-react-sdk/pull/5460)
+ * Add sanity checking around widget pinning
+ [\#5459](https://github.com/matrix-org/matrix-react-sdk/pull/5459)
+ * Update i18n for Appearance User Settings
+ [\#5457](https://github.com/matrix-org/matrix-react-sdk/pull/5457)
+ * Only show 'answered elsewhere' if we tried to answer too
+ [\#5455](https://github.com/matrix-org/matrix-react-sdk/pull/5455)
+ * Fixed Avatar for 3PID invites
+ [\#5442](https://github.com/matrix-org/matrix-react-sdk/pull/5442)
+ * Slightly better error if we can't capture user media
+ [\#5449](https://github.com/matrix-org/matrix-react-sdk/pull/5449)
+ * Make it possible in-code to hide rooms from the room list
+ [\#5445](https://github.com/matrix-org/matrix-react-sdk/pull/5445)
+ * Fix the stickerpicker
+ [\#5447](https://github.com/matrix-org/matrix-react-sdk/pull/5447)
+ * Add live password validation to change password dialog
+ [\#5436](https://github.com/matrix-org/matrix-react-sdk/pull/5436)
+ * LaTeX rendering in element-web using KaTeX
+ [\#5244](https://github.com/matrix-org/matrix-react-sdk/pull/5244)
+ * Add lifecycle customisation point after logout
+ [\#5448](https://github.com/matrix-org/matrix-react-sdk/pull/5448)
+ * Simplify UserMenu for Guests as they can't use most of the options
+ [\#5421](https://github.com/matrix-org/matrix-react-sdk/pull/5421)
+ * Fix known issues with modal widgets
+ [\#5444](https://github.com/matrix-org/matrix-react-sdk/pull/5444)
+ * Fix existing widgets not having approved capabilities for their function
+ [\#5443](https://github.com/matrix-org/matrix-react-sdk/pull/5443)
+ * Use the WidgetDriver to run OIDC requests
+ [\#5440](https://github.com/matrix-org/matrix-react-sdk/pull/5440)
+ * Add a customisation point for widget permissions and fix amnesia issues
+ [\#5439](https://github.com/matrix-org/matrix-react-sdk/pull/5439)
+ * Fix Widget event notification text including spurious space
+ [\#5441](https://github.com/matrix-org/matrix-react-sdk/pull/5441)
+ * Move call listener out of MatrixChat
+ [\#5438](https://github.com/matrix-org/matrix-react-sdk/pull/5438)
+ * New Look in-Call View
+ [\#5432](https://github.com/matrix-org/matrix-react-sdk/pull/5432)
+ * Support arbitrary widgets sticking to the screen + sending stickers
+ [\#5435](https://github.com/matrix-org/matrix-react-sdk/pull/5435)
+ * Auth typescripting and validation tweaks
+ [\#5433](https://github.com/matrix-org/matrix-react-sdk/pull/5433)
+ * Add new widget API actions for changing rooms and sending/receiving events
+ [\#5385](https://github.com/matrix-org/matrix-react-sdk/pull/5385)
+ * Revert room header click behaviour to opening room settings
+ [\#5434](https://github.com/matrix-org/matrix-react-sdk/pull/5434)
+ * Add option to send/edit a message with Ctrl + Enter / Command + Enter
+ [\#5160](https://github.com/matrix-org/matrix-react-sdk/pull/5160)
+ * Add Analytics instrumentation to the Homepage
+ [\#5409](https://github.com/matrix-org/matrix-react-sdk/pull/5409)
+ * Fix encrypted video playback in Chrome-based browsers
+ [\#5430](https://github.com/matrix-org/matrix-react-sdk/pull/5430)
+ * Add border-radius for video
+ [\#5333](https://github.com/matrix-org/matrix-react-sdk/pull/5333)
+ * Push name to the end, near text, in IRC layout
+ [\#5166](https://github.com/matrix-org/matrix-react-sdk/pull/5166)
+ * Disable notifications for the room you have recently been active in
+ [\#5325](https://github.com/matrix-org/matrix-react-sdk/pull/5325)
+ * Search through the list of unfiltered rooms rather than the rooms in the
+ state which are already filtered by the search text
+ [\#5331](https://github.com/matrix-org/matrix-react-sdk/pull/5331)
+ * Lighten blockquote colour in dark mode
+ [\#5353](https://github.com/matrix-org/matrix-react-sdk/pull/5353)
+ * Specify community description img must be mxc urls
+ [\#5364](https://github.com/matrix-org/matrix-react-sdk/pull/5364)
+ * Add keyboard shortcut to close the current conversation
+ [\#5253](https://github.com/matrix-org/matrix-react-sdk/pull/5253)
+ * Redirect user home from auth screens if they are already logged in
+ [\#5423](https://github.com/matrix-org/matrix-react-sdk/pull/5423)
+
Changes in [3.9.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.9.0) (2020-11-23)
===================================================================================================
[Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.9.0-rc.1...v3.9.0)
diff --git a/package.json b/package.json
index f3b8104663..b328823b24 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "matrix-react-sdk",
- "version": "3.9.0",
+ "version": "3.10.0",
"description": "SDK for matrix.org using React",
"author": "matrix.org",
"repository": {
@@ -50,7 +50,7 @@
"lint:types": "tsc --noEmit --jsx react",
"lint:style": "stylelint 'res/css/**/*.scss'",
"test": "jest",
- "test:e2e": "./test/end-to-end-tests/run.sh --riot-url http://localhost:8080"
+ "test:e2e": "./test/end-to-end-tests/run.sh --app-url http://localhost:8080"
},
"dependencies": {
"@babel/runtime": "^7.10.5",
@@ -58,6 +58,7 @@
"blueimp-canvas-to-blob": "^3.27.0",
"browser-encrypt-attachment": "^0.3.0",
"browser-request": "^0.3.3",
+ "cheerio": "^1.0.0-rc.3",
"classnames": "^2.2.6",
"commonmark": "^0.29.1",
"counterpart": "^0.18.6",
@@ -77,7 +78,6 @@
"html-entities": "^1.3.1",
"is-ip": "^2.0.0",
"katex": "^0.12.0",
- "cheerio": "^1.0.0-rc.3",
"linkifyjs": "^2.1.9",
"lodash": "^4.17.19",
"matrix-js-sdk": "github:matrix-org/matrix-js-sdk#develop",
@@ -159,6 +159,7 @@
"lolex": "^5.1.2",
"matrix-mock-request": "^1.2.3",
"matrix-react-test-utils": "^0.2.2",
+ "olm": "https://packages.matrix.org/npm/olm/olm-3.2.1.tgz",
"react-test-renderer": "^16.13.1",
"rimraf": "^2.7.1",
"stylelint": "^9.10.1",
diff --git a/res/css/_common.scss b/res/css/_common.scss
index 7ab88d6f02..87336a1c03 100644
--- a/res/css/_common.scss
+++ b/res/css/_common.scss
@@ -170,7 +170,6 @@ input[type=text]:focus, input[type=password]:focus, textarea:focus {
border: 1px solid rgba($primary-fg-color, .1);
// these things should probably not be defined globally
margin: 9px;
- flex: 0 0 auto;
}
.mx_textinput {
diff --git a/res/css/_components.scss b/res/css/_components.scss
index c316493862..12a65bbbde 100644
--- a/res/css/_components.scss
+++ b/res/css/_components.scss
@@ -45,14 +45,13 @@
@import "./views/auth/_InteractiveAuthEntryComponents.scss";
@import "./views/auth/_LanguageSelector.scss";
@import "./views/auth/_PassphraseField.scss";
-@import "./views/auth/_ServerConfig.scss";
-@import "./views/auth/_ServerTypeSelector.scss";
@import "./views/auth/_Welcome.scss";
@import "./views/avatars/_BaseAvatar.scss";
@import "./views/avatars/_DecoratedRoomAvatar.scss";
@import "./views/avatars/_MemberStatusMessageAvatar.scss";
@import "./views/avatars/_PulsedAvatar.scss";
@import "./views/avatars/_WidgetAvatar.scss";
+@import "./views/context_menus/_CallContextMenu.scss";
@import "./views/context_menus/_IconizedContextMenu.scss";
@import "./views/context_menus/_MessageContextMenu.scss";
@import "./views/context_menus/_StatusMessageContextMenu.scss";
@@ -79,11 +78,13 @@
@import "./views/dialogs/_MessageEditHistoryDialog.scss";
@import "./views/dialogs/_ModalWidgetDialog.scss";
@import "./views/dialogs/_NewSessionReviewDialog.scss";
+@import "./views/dialogs/_RegistrationEmailPromptDialog.scss";
@import "./views/dialogs/_RoomSettingsDialog.scss";
@import "./views/dialogs/_RoomSettingsDialogBridges.scss";
@import "./views/dialogs/_RoomUpgradeDialog.scss";
@import "./views/dialogs/_RoomUpgradeWarningDialog.scss";
@import "./views/dialogs/_ServerOfflineDialog.scss";
+@import "./views/dialogs/_ServerPickerDialog.scss";
@import "./views/dialogs/_SetEmailDialog.scss";
@import "./views/dialogs/_SettingsDialog.scss";
@import "./views/dialogs/_ShareDialog.scss";
@@ -125,6 +126,8 @@
@import "./views/elements/_RichText.scss";
@import "./views/elements/_RoleButton.scss";
@import "./views/elements/_RoomAliasField.scss";
+@import "./views/elements/_SSOButtons.scss";
+@import "./views/elements/_ServerPicker.scss";
@import "./views/elements/_Slider.scss";
@import "./views/elements/_Spinner.scss";
@import "./views/elements/_StyledCheckbox.scss";
diff --git a/res/css/structures/auth/_Login.scss b/res/css/structures/auth/_Login.scss
index 02436833a2..9c98ca3a1c 100644
--- a/res/css/structures/auth/_Login.scss
+++ b/res/css/structures/auth/_Login.scss
@@ -18,7 +18,7 @@ limitations under the License.
.mx_Login_submit {
@mixin mx_DialogButton;
width: 100%;
- margin-top: 35px;
+ margin-top: 24px;
margin-bottom: 24px;
box-sizing: border-box;
text-align: center;
@@ -33,12 +33,6 @@ limitations under the License.
cursor: default;
}
-.mx_AuthBody a.mx_Login_sso_link:link,
-.mx_AuthBody a.mx_Login_sso_link:hover,
-.mx_AuthBody a.mx_Login_sso_link:visited {
- color: $button-primary-fg-color;
-}
-
.mx_Login_loader {
display: inline;
position: relative;
@@ -87,10 +81,13 @@ limitations under the License.
}
.mx_Login_underlinedServerName {
+ width: max-content;
border-bottom: 1px dashed $accent-color;
}
div.mx_AccessibleButton_kind_link.mx_Login_forgot {
+ display: block;
+ margin: 0 auto;
// style it as a link
font-size: inherit;
padding: 0;
diff --git a/res/css/views/auth/_AuthBody.scss b/res/css/views/auth/_AuthBody.scss
index 0ba0d10e06..8f0c758e7a 100644
--- a/res/css/views/auth/_AuthBody.scss
+++ b/res/css/views/auth/_AuthBody.scss
@@ -37,6 +37,10 @@ limitations under the License.
color: $authpage-primary-color;
}
+ h3.mx_AuthBody_centered {
+ text-align: center;
+ }
+
a:link,
a:hover,
a:visited {
@@ -96,12 +100,6 @@ limitations under the License.
}
}
-.mx_AuthBody_editServerDetails {
- padding-left: 1em;
- font-size: $font-12px;
- font-weight: normal;
-}
-
.mx_AuthBody_fieldRow {
display: flex;
margin-bottom: 10px;
@@ -146,6 +144,14 @@ limitations under the License.
display: block;
text-align: center;
width: 100%;
+
+ > a {
+ font-weight: $font-semi-bold;
+ }
+}
+
+.mx_SSOButtons + .mx_AuthBody_changeFlow {
+ margin-top: 24px;
}
.mx_AuthBody_spinner {
diff --git a/res/css/views/auth/_ServerTypeSelector.scss b/res/css/views/auth/_ServerTypeSelector.scss
deleted file mode 100644
index fbd3d2655d..0000000000
--- a/res/css/views/auth/_ServerTypeSelector.scss
+++ /dev/null
@@ -1,69 +0,0 @@
-/*
-Copyright 2019 New Vector Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-.mx_ServerTypeSelector {
- display: flex;
- margin-bottom: 28px;
-}
-
-.mx_ServerTypeSelector_type {
- margin: 0 5px;
-}
-
-.mx_ServerTypeSelector_type:first-child {
- margin-left: 0;
-}
-
-.mx_ServerTypeSelector_type:last-child {
- margin-right: 0;
-}
-
-.mx_ServerTypeSelector_label {
- text-align: center;
- font-weight: 600;
- color: $authpage-primary-color;
- margin: 8px 0;
-}
-
-.mx_ServerTypeSelector_type .mx_AccessibleButton {
- padding: 10px;
- border: 1px solid $input-border-color;
- border-radius: 4px;
-}
-
-.mx_ServerTypeSelector_type.mx_ServerTypeSelector_type_selected .mx_AccessibleButton {
- border-color: $input-valid-border-color;
-}
-
-.mx_ServerTypeSelector_logo {
- display: flex;
- justify-content: center;
- height: 18px;
- margin-bottom: 12px;
- font-weight: 600;
- color: $authpage-primary-color;
-}
-
-.mx_ServerTypeSelector_logo > div {
- display: flex;
- width: 70%;
- align-items: center;
- justify-content: space-evenly;
-}
-
-.mx_ServerTypeSelector_description {
- font-size: $font-10px;
-}
diff --git a/res/css/views/auth/_ServerConfig.scss b/res/css/views/context_menus/_CallContextMenu.scss
similarity index 56%
rename from res/css/views/auth/_ServerConfig.scss
rename to res/css/views/context_menus/_CallContextMenu.scss
index a7e0057ab3..55b73b0344 100644
--- a/res/css/views/auth/_ServerConfig.scss
+++ b/res/css/views/context_menus/_CallContextMenu.scss
@@ -1,6 +1,5 @@
/*
-Copyright 2015, 2016 OpenMarket Ltd
-Copyright 2019 The Matrix.org Foundation C.I.C.
+Copyright 2020 New Vector Ltd
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
@@ -15,21 +14,10 @@ See the License for the specific language governing permissions and
limitations under the License.
*/
-.mx_ServerConfig_help:link {
- opacity: 0.8;
-}
-
-.mx_ServerConfig_error {
- display: block;
- color: $warning-color;
-}
-
-.mx_ServerConfig_identityServer {
- transform: scaleY(0);
- transform-origin: top;
- transition: transform 0.25s;
-
- &.mx_ServerConfig_identityServer_shown {
- transform: scaleY(1);
- }
+.mx_CallContextMenu_item {
+ width: 205px;
+ height: 40px;
+ padding-left: 16px;
+ line-height: 40px;
+ vertical-align: center;
}
diff --git a/src/RoomListSorter.js b/res/css/views/dialogs/_RegistrationEmailPromptDialog.scss
similarity index 58%
rename from src/RoomListSorter.js
rename to res/css/views/dialogs/_RegistrationEmailPromptDialog.scss
index 0ff37a6af2..31fc6d7a04 100644
--- a/src/RoomListSorter.js
+++ b/res/css/views/dialogs/_RegistrationEmailPromptDialog.scss
@@ -1,5 +1,5 @@
/*
-Copyright 2015, 2016 OpenMarket Ltd
+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,18 +14,15 @@ See the License for the specific language governing permissions and
limitations under the License.
*/
-'use strict';
+.mx_RegistrationEmailPromptDialog {
+ width: 417px;
-function tsOfNewestEvent(room) {
- if (room.timeline.length) {
- return room.timeline[room.timeline.length - 1].getTs();
- } else {
- return Number.MAX_SAFE_INTEGER;
+ .mx_Dialog_content {
+ margin-bottom: 24px;
+ color: $tertiary-fg-color;
+ }
+
+ .mx_Dialog_primary {
+ width: 100%;
}
}
-
-export function mostRecentActivityFirst(roomList) {
- return roomList.sort(function(a, b) {
- return tsOfNewestEvent(b) - tsOfNewestEvent(a);
- });
-}
diff --git a/res/css/views/dialogs/_ServerPickerDialog.scss b/res/css/views/dialogs/_ServerPickerDialog.scss
new file mode 100644
index 0000000000..b01b49d7af
--- /dev/null
+++ b/res/css/views/dialogs/_ServerPickerDialog.scss
@@ -0,0 +1,78 @@
+/*
+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_ServerPickerDialog {
+ width: 468px;
+ box-sizing: border-box;
+
+ .mx_Dialog_content {
+ margin-bottom: 0;
+
+ > p {
+ color: $secondary-fg-color;
+ font-size: $font-14px;
+ margin: 16px 0;
+
+ &:first-of-type {
+ margin-bottom: 40px;
+ }
+
+ &:last-of-type {
+ margin: 0 24px 24px;
+ }
+ }
+
+ > h4 {
+ font-size: $font-15px;
+ font-weight: $font-semi-bold;
+ color: $secondary-fg-color;
+ margin-left: 8px;
+ }
+
+ > a {
+ color: $accent-color;
+ margin-left: 8px;
+ }
+ }
+
+ .mx_ServerPickerDialog_otherHomeserverRadio {
+ input[type="radio"] + div {
+ margin-top: auto;
+ margin-bottom: auto;
+ }
+ }
+
+ .mx_ServerPickerDialog_otherHomeserver {
+ border-top: none;
+ border-left: none;
+ border-right: none;
+ border-radius: unset;
+
+ > input {
+ padding-left: 0;
+ }
+
+ > label {
+ margin-left: 0;
+ }
+ }
+
+ .mx_AccessibleButton_kind_primary {
+ width: calc(100% - 64px);
+ margin: 0 8px;
+ padding: 15px 18px;
+ }
+}
diff --git a/res/css/views/elements/_SSOButtons.scss b/res/css/views/elements/_SSOButtons.scss
new file mode 100644
index 0000000000..f762468c7f
--- /dev/null
+++ b/res/css/views/elements/_SSOButtons.scss
@@ -0,0 +1,49 @@
+/*
+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_SSOButtons {
+ display: flex;
+ justify-content: center;
+
+ .mx_SSOButton {
+ position: relative;
+ width: 100%;
+ padding-left: 32px;
+ padding-right: 32px;
+
+ > img {
+ object-fit: contain;
+ position: absolute;
+ left: 8px;
+ top: 4px;
+ }
+ }
+
+ .mx_SSOButton_mini {
+ box-sizing: border-box;
+ width: 50px; // 48px + 1px border on all sides
+ height: 50px; // 48px + 1px border on all sides
+
+ > img {
+ left: 12px;
+ top: 12px;
+ }
+
+ & + .mx_SSOButton_mini {
+ margin-left: 24px;
+ }
+ }
+}
diff --git a/res/css/views/elements/_ServerPicker.scss b/res/css/views/elements/_ServerPicker.scss
new file mode 100644
index 0000000000..ae1e445a9f
--- /dev/null
+++ b/res/css/views/elements/_ServerPicker.scss
@@ -0,0 +1,88 @@
+/*
+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_ServerPicker {
+ margin-bottom: 14px;
+ border-bottom: 1px solid rgba(141, 151, 165, 0.2);
+ display: grid;
+ grid-template-columns: auto min-content;
+ grid-template-rows: auto auto auto;
+ font-size: $font-14px;
+ line-height: $font-20px;
+
+ > h3 {
+ font-weight: $font-semi-bold;
+ margin: 0 0 20px;
+ grid-column: 1;
+ grid-row: 1;
+ }
+
+ .mx_ServerPicker_help {
+ width: 20px;
+ height: 20px;
+ background-color: $icon-button-color;
+ border-radius: 10px;
+ grid-column: 2;
+ grid-row: 1;
+ margin-left: auto;
+ text-align: center;
+ color: #ffffff;
+ font-size: 16px;
+ position: relative;
+
+ &::before {
+ content: '';
+ width: 24px;
+ height: 24px;
+ position: absolute;
+ top: -2px;
+ left: -2px;
+ mask-position: center;
+ mask-size: contain;
+ mask-repeat: no-repeat;
+ mask-image: url('$(res)/img/element-icons/i.svg');
+ background: #ffffff;
+ }
+ }
+
+ .mx_ServerPicker_server {
+ color: $primary-fg-color;
+ grid-column: 1;
+ grid-row: 2;
+ margin-bottom: 16px;
+ }
+
+ .mx_ServerPicker_change {
+ padding: 0;
+ font-size: inherit;
+ grid-column: 2;
+ grid-row: 2;
+ }
+
+ .mx_ServerPicker_desc {
+ margin-top: -12px;
+ color: $tertiary-fg-color;
+ grid-column: 1 / 2;
+ grid-row: 3;
+ margin-bottom: 16px;
+ }
+}
+
+.mx_ServerPicker_helpDialog {
+ .mx_Dialog_content {
+ width: 456px;
+ }
+}
diff --git a/res/css/views/rooms/_MemberList.scss b/res/css/views/rooms/_MemberList.scss
index f00907aeef..1e3506e371 100644
--- a/res/css/views/rooms/_MemberList.scss
+++ b/res/css/views/rooms/_MemberList.scss
@@ -46,6 +46,11 @@ limitations under the License.
}
}
+.mx_GroupMemberList_query,
+.mx_GroupRoomList_query {
+ flex: 0 0 auto;
+}
+
.mx_MemberList_chevron {
position: absolute;
right: 35px;
@@ -59,10 +64,8 @@ limitations under the License.
flex: 1 1 0px;
}
-.mx_MemberList_query,
-.mx_GroupMemberList_query,
-.mx_GroupRoomList_query {
- flex: 1 1 0;
+.mx_MemberList_query {
+ height: 16px;
// stricter rule to override the one in _common.scss
&[type="text"] {
@@ -70,10 +73,6 @@ limitations under the License.
}
}
-.mx_MemberList_query {
- height: 16px;
-}
-
.mx_MemberList_wrapper {
padding: 10px;
}
@@ -113,10 +112,10 @@ limitations under the License.
}
}
-.mx_MemberList_inviteCommunity span {
- background-image: url('$(res)/img/icon-invite-people.svg');
+.mx_MemberList_inviteCommunity span::before {
+ mask-image: url('$(res)/img/icon-invite-people.svg');
}
-.mx_MemberList_addRoomToCommunity span {
- background-image: url('$(res)/img/icons-room-add.svg');
+.mx_MemberList_addRoomToCommunity span::before {
+ mask-image: url('$(res)/img/icons-room-add.svg');
}
diff --git a/res/css/views/rooms/_Stickers.scss b/res/css/views/rooms/_Stickers.scss
index 94f42efe83..da86797f42 100644
--- a/res/css/views/rooms/_Stickers.scss
+++ b/res/css/views/rooms/_Stickers.scss
@@ -22,7 +22,7 @@
iframe {
// Sticker picker depends on the fixed height previously used for all tiles
- height: 273px;
+ height: 283px; // height of the popout minus the AppTile menu bar
}
}
diff --git a/res/css/views/voip/_CallView.scss b/res/css/views/voip/_CallView.scss
index 2b87181b1e..dbe2c27e41 100644
--- a/res/css/views/voip/_CallView.scss
+++ b/res/css/views/voip/_CallView.scss
@@ -16,8 +16,8 @@ limitations under the License.
*/
.mx_CallView {
- border-radius: 10px;
- background-color: $input-lighter-bg-color;
+ border-radius: 8px;
+ background-color: $voipcall-plinth-color;
padding-left: 8px;
padding-right: 8px;
// XXX: CallContainer sets pointer-events: none - should probably be set back in a better place
@@ -26,6 +26,7 @@ limitations under the License.
.mx_CallView_large {
padding-bottom: 10px;
+ margin: 5px 5px 5px 18px;
.mx_CallView_voice {
height: 360px;
@@ -38,20 +39,176 @@ limitations under the License.
.mx_CallView_voice {
height: 180px;
}
+
+ .mx_CallView_callControls {
+ bottom: 0px;
+ }
+
+ .mx_CallView_callControls_button {
+ &::before {
+ width: 36px;
+ height: 36px;
+ }
+ }
+
+ .mx_CallView_voice_holdText {
+ padding-top: 10px;
+ padding-bottom: 25px;
+ }
}
.mx_CallView_voice {
position: relative;
display: flex;
+ flex-direction: column;
align-items: center;
justify-content: center;
background-color: $inverted-bg-color;
+ border-radius: 8px;
+}
+
+.mx_CallView_voice_avatarsContainer {
+ display: flex;
+ flex-direction: row;
+ align-items: center;
+ justify-content: center;
+ div {
+ margin-left: 12px;
+ margin-right: 12px;
+ }
+}
+
+.mx_CallView_voice_hold {
+ // This masks the avatar image so when it's blurred, the edge is still crisp
+ .mx_CallView_voice_avatarContainer {
+ border-radius: 2000px;
+ overflow: hidden;
+ position: relative;
+ &::after {
+ position: absolute;
+ content: '';
+ width: 100%;
+ height: 100%;
+ top: 50%;
+ left: 50%;
+ transform: translate(-50%, -50%);
+ background-color: rgba(0, 0, 0, 0.6);
+ background-image: url('$(res)/img/voip/paused.svg');
+ background-position: center;
+ background-size: 40px;
+ background-repeat: no-repeat;
+ }
+ .mx_CallView_pip &::after {
+ background-size: 30px;
+ }
+ }
+ .mx_BaseAvatar {
+ filter: blur(20px);
+ overflow: hidden;
+ }
+}
+
+.mx_CallView_voice_secondaryAvatarContainer {
+ border-radius: 2000px;
+ overflow: hidden;
+ position: relative;
+ &::after {
+ position: absolute;
+ content: '';
+ width: 100%;
+ height: 100%;
+ top: 50%;
+ left: 50%;
+ transform: translate(-50%, -50%);
+ background-color: rgba(0, 0, 0, 0.6);
+ background-image: url('$(res)/img/voip/paused.svg');
+ background-position: center;
+ background-size: 40px;
+ background-repeat: no-repeat;
+ }
+ .mx_CallView_pip &::after {
+ background-size: 24px;
+ }
+}
+
+.mx_CallView_voice_holdText {
+ height: 20px;
+ padding-top: 20px;
+ padding-bottom: 15px;
+ color: $accent-fg-color;
+ .mx_AccessibleButton_hasKind {
+ padding: 0px;
+ font-weight: bold;
+ }
}
.mx_CallView_video {
width: 100%;
position: relative;
z-index: 30;
+ border-radius: 8px;
+ overflow: hidden;
+}
+
+.mx_CallView_video_hold {
+ overflow: hidden;
+
+ // we keep these around in the DOM: it saved wiring them up again when the call
+ // is resumed and keeps the container the right size
+ .mx_VideoFeed {
+ visibility: hidden;
+ }
+}
+
+.mx_CallView_video_holdBackground {
+ position: absolute;
+ width: 100%;
+ height: 100%;
+ left: 0;
+ right: 0;
+ background-repeat: no-repeat;
+ background-size: cover;
+ background-position: center;
+ filter: blur(20px);
+ &::after {
+ content: '';
+ display: block;
+ position: absolute;
+ width: 100%;
+ height: 100%;
+ left: 0;
+ right: 0;
+ background-color: rgba(0, 0, 0, 0.6);
+ }
+}
+
+.mx_CallView_video_holdContent {
+ position: absolute;
+ top: 50%;
+ left: 50%;
+ transform: translate(-50%, -50%);
+ font-weight: bold;
+ color: $accent-fg-color;
+ text-align: center;
+
+ &::before {
+ display: block;
+ margin-left: auto;
+ margin-right: auto;
+ content: '';
+ width: 40px;
+ height: 40px;
+ background-image: url('$(res)/img/voip/paused.svg');
+ background-position: center;
+ background-size: cover;
+ }
+ .mx_CallView_pip &::before {
+ width: 30px;
+ height: 30px;
+ }
+ .mx_AccessibleButton_hasKind {
+ padding: 0px;
+ }
}
.mx_CallView_header {
@@ -60,17 +217,22 @@ limitations under the License.
flex-direction: row;
align-items: center;
justify-content: left;
-
- .mx_BaseAvatar {
- margin-right: 12px;
- }
}
.mx_CallView_header_callType {
+ font-size: 1.2rem;
font-weight: bold;
vertical-align: middle;
}
+.mx_CallView_header_secondaryCallInfo {
+ &::before {
+ content: '·';
+ margin-left: 6px;
+ margin-right: 6px;
+ }
+}
+
.mx_CallView_header_controls {
margin-left: auto;
}
@@ -105,16 +267,31 @@ limitations under the License.
}
}
+.mx_CallView_header_callInfo {
+ margin-left: 12px;
+ margin-right: 16px;
+}
+
.mx_CallView_header_roomName {
font-weight: bold;
font-size: 12px;
line-height: initial;
+ height: 15px;
+}
+
+.mx_CallView_secondaryCall_roomName {
+ margin-left: 4px;
}
.mx_CallView_header_callTypeSmall {
font-size: 12px;
color: $secondary-fg-color;
line-height: initial;
+ height: 15px;
+ overflow: hidden;
+ white-space: nowrap;
+ text-overflow: ellipsis;
+ max-width: 240px;
}
.mx_CallView_header_phoneIcon {
@@ -173,6 +350,12 @@ limitations under the License.
}
}
+// Makes the alignment correct
+.mx_CallView_callControls_nothing {
+ margin-right: auto;
+ cursor: initial;
+}
+
.mx_CallView_callControls_button_micOn {
&::before {
background-image: url('$(res)/img/voip/mic-on.svg');
@@ -203,6 +386,18 @@ limitations under the License.
}
}
+.mx_CallView_callControls_button_more {
+ margin-left: auto;
+ &::before {
+ background-image: url('$(res)/img/voip/more.svg');
+ }
+}
+
+.mx_CallView_callControls_button_more_hidden {
+ margin-left: auto;
+ cursor: initial;
+}
+
.mx_CallView_callControls_button_invisible {
visibility: hidden;
pointer-events: none;
diff --git a/res/img/element-icons/i.svg b/res/img/element-icons/i.svg
new file mode 100644
index 0000000000..6674f1ed8d
--- /dev/null
+++ b/res/img/element-icons/i.svg
@@ -0,0 +1,3 @@
+
diff --git a/res/img/voip/more.svg b/res/img/voip/more.svg
new file mode 100644
index 0000000000..7990f6bcff
--- /dev/null
+++ b/res/img/voip/more.svg
@@ -0,0 +1,17 @@
+
diff --git a/res/img/voip/paused.svg b/res/img/voip/paused.svg
new file mode 100644
index 0000000000..a967bf8ddf
--- /dev/null
+++ b/res/img/voip/paused.svg
@@ -0,0 +1,3 @@
+
diff --git a/res/themes/dark/css/_dark.scss b/res/themes/dark/css/_dark.scss
index 76cc5e2df9..08fe2e9f57 100644
--- a/res/themes/dark/css/_dark.scss
+++ b/res/themes/dark/css/_dark.scss
@@ -108,6 +108,9 @@ $eventtile-meta-color: $roomtopic-color;
$header-divider-color: $header-panel-text-primary-color;
$composer-e2e-icon-color: $header-panel-text-primary-color;
+// this probably shouldn't have it's own colour
+$voipcall-plinth-color: #21262c;
+
// ********************
$theme-button-bg-color: #e3e8f0;
@@ -214,7 +217,7 @@ $composer-shadow-color: rgba(0, 0, 0, 0.28);
/* align images in buttons (eg spinners) */
vertical-align: middle;
border: 0px;
- border-radius: 4px;
+ border-radius: 8px;
font-family: $font-family;
font-size: $font-14px;
color: $button-fg-color;
diff --git a/res/themes/legacy-dark/css/_legacy-dark.scss b/res/themes/legacy-dark/css/_legacy-dark.scss
index 716d8c7385..3e3c299af9 100644
--- a/res/themes/legacy-dark/css/_legacy-dark.scss
+++ b/res/themes/legacy-dark/css/_legacy-dark.scss
@@ -105,6 +105,9 @@ $eventtile-meta-color: $roomtopic-color;
$header-divider-color: $header-panel-text-primary-color;
$composer-e2e-icon-color: $header-panel-text-primary-color;
+// this probably shouldn't have it's own colour
+$voipcall-plinth-color: #f2f5f8;
+
// ********************
$theme-button-bg-color: #e3e8f0;
@@ -205,7 +208,7 @@ $composer-shadow-color: tranparent;
/* align images in buttons (eg spinners) */
vertical-align: middle;
border: 0px;
- border-radius: 4px;
+ border-radius: 8px;
font-family: $font-family;
font-size: $font-14px;
color: $button-fg-color;
diff --git a/res/themes/legacy-light/css/_legacy-light.scss b/res/themes/legacy-light/css/_legacy-light.scss
index 8c42c5c97f..085d6d7f10 100644
--- a/res/themes/legacy-light/css/_legacy-light.scss
+++ b/res/themes/legacy-light/css/_legacy-light.scss
@@ -172,6 +172,9 @@ $eventtile-meta-color: $roomtopic-color;
$composer-e2e-icon-color: #91a1c0;
$header-divider-color: #91a1c0;
+// this probably shouldn't have it's own colour
+$voipcall-plinth-color: #f2f5f8;
+
// ********************
$theme-button-bg-color: #e3e8f0;
@@ -328,7 +331,7 @@ $composer-shadow-color: tranparent;
/* align images in buttons (eg spinners) */
vertical-align: middle;
border: 0px;
- border-radius: 4px;
+ border-radius: 8px;
font-family: $font-family;
font-size: $font-14px;
color: $button-fg-color;
diff --git a/res/themes/light/css/_light.scss b/res/themes/light/css/_light.scss
index 5437a6de1c..4cfeeae05e 100644
--- a/res/themes/light/css/_light.scss
+++ b/res/themes/light/css/_light.scss
@@ -166,6 +166,9 @@ $eventtile-meta-color: $roomtopic-color;
$composer-e2e-icon-color: #91A1C0;
$header-divider-color: #91A1C0;
+// this probably shouldn't have it's own colour
+$voipcall-plinth-color: #f2f5f8;
+
// ********************
$theme-button-bg-color: #e3e8f0;
@@ -332,7 +335,7 @@ $composer-shadow-color: rgba(0, 0, 0, 0.04);
/* align images in buttons (eg spinners) */
vertical-align: middle;
border: 0px;
- border-radius: 4px;
+ border-radius: 8px;
font-family: $font-family;
font-size: $font-14px;
color: $button-fg-color;
diff --git a/scripts/ci/Dockerfile b/scripts/ci/Dockerfile
index c153d11cc7..5351291f29 100644
--- a/scripts/ci/Dockerfile
+++ b/scripts/ci/Dockerfile
@@ -1,7 +1,7 @@
# Update on docker hub with the following commands in the directory of this file:
-# docker build -t matrixdotorg/riotweb-ci-e2etests-env:latest .
+# docker build -t vectorim/element-web-ci-e2etests-env:latest .
# docker log
-# docker push matrixdotorg/riotweb-ci-e2etests-env:latest
+# docker push vectorim/element-web-ci-e2etests-env:latest
FROM node:10
RUN apt-get update
RUN apt-get -y install build-essential python3-dev libffi-dev python-pip python-setuptools sqlite3 libssl-dev python-virtualenv libjpeg-dev libxslt1-dev uuid-runtime
diff --git a/scripts/ci/riot-unit-tests.sh b/scripts/ci/app-tests.sh
similarity index 56%
rename from scripts/ci/riot-unit-tests.sh
rename to scripts/ci/app-tests.sh
index 337c0fe6c3..97e54dce66 100755
--- a/scripts/ci/riot-unit-tests.sh
+++ b/scripts/ci/app-tests.sh
@@ -2,11 +2,11 @@
#
# script which is run by the CI build (after `yarn test`).
#
-# clones riot-web develop and runs the tests against our version of react-sdk.
+# clones element-web develop and runs the tests against our version of react-sdk.
set -ev
-scripts/ci/layered-riot-web.sh
-cd ../riot-web
+scripts/ci/layered.sh
+cd element-web
yarn build:genfiles # so the tests can run. Faster version of `build`
yarn test
diff --git a/scripts/ci/end-to-end-tests.sh b/scripts/ci/end-to-end-tests.sh
index 7a62c03b12..edb8870d8e 100755
--- a/scripts/ci/end-to-end-tests.sh
+++ b/scripts/ci/end-to-end-tests.sh
@@ -2,7 +2,7 @@
#
# script which is run by the CI build (after `yarn test`).
#
-# clones riot-web develop and runs the tests against our version of react-sdk.
+# clones element-web develop and runs the tests against our version of react-sdk.
set -ev
@@ -14,20 +14,20 @@ handle_error() {
trap 'handle_error' ERR
echo "--- Building Element"
-scripts/ci/layered-riot-web.sh
-cd ../riot-web
-riot_web_dir=`pwd`
+scripts/ci/layered.sh
+cd element-web
+element_web_dir=`pwd`
CI_PACKAGE=true yarn build
-cd ../matrix-react-sdk
+cd ..
# run end to end tests
pushd test/end-to-end-tests
-ln -s $riot_web_dir riot/riot-web
+ln -s $element_web_dir element/element-web
# PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=true ./install.sh
# CHROME_PATH=$(which google-chrome-stable) ./run.sh
echo "--- Install synapse & other dependencies"
./install.sh
-# install static webserver to server symlinked local copy of riot
-./riot/install-webserver.sh
+# install static webserver to server symlinked local copy of element
+./element/install-webserver.sh
rm -r logs || true
mkdir logs
echo "+++ Running end-to-end tests"
diff --git a/scripts/ci/layered-riot-web.sh b/scripts/ci/layered-riot-web.sh
deleted file mode 100755
index f58794b451..0000000000
--- a/scripts/ci/layered-riot-web.sh
+++ /dev/null
@@ -1,31 +0,0 @@
-#!/bin/bash
-
-# Creates an environment similar to one that riot-web would expect for
-# development. This means going one directory up (and assuming we're in
-# a directory like /workdir/matrix-react-sdk) and putting riot-web and
-# the js-sdk there.
-
-cd ../ # Assume we're at something like /workdir/matrix-react-sdk
-
-# Set up the js-sdk first
-matrix-react-sdk/scripts/fetchdep.sh matrix-org matrix-js-sdk
-pushd matrix-js-sdk
-yarn link
-yarn install
-popd
-
-# Now set up the react-sdk
-pushd matrix-react-sdk
-yarn link matrix-js-sdk
-yarn link
-yarn install
-popd
-
-# Finally, set up riot-web
-matrix-react-sdk/scripts/fetchdep.sh vector-im riot-web
-pushd riot-web
-yarn link matrix-js-sdk
-yarn link matrix-react-sdk
-yarn install
-yarn build:res
-popd
diff --git a/scripts/ci/layered.sh b/scripts/ci/layered.sh
new file mode 100755
index 0000000000..306f9c9974
--- /dev/null
+++ b/scripts/ci/layered.sh
@@ -0,0 +1,31 @@
+#!/bin/bash
+
+# Creates a layered environment with the full repo for the app and SDKs cloned
+# and linked.
+
+# Note that this style is different from the recommended developer setup: this
+# file nests js-sdk and element-web inside react-sdk, while the local
+# development setup places them all at the same level. We are nesting them here
+# because some CI systems do not allow moving to a directory above the checkout
+# for the primary repo (react-sdk in this case).
+
+# Set up the js-sdk first
+scripts/fetchdep.sh matrix-org matrix-js-sdk
+pushd matrix-js-sdk
+yarn link
+yarn install
+popd
+
+# Now set up the react-sdk
+yarn link matrix-js-sdk
+yarn link
+yarn install
+
+# Finally, set up element-web
+scripts/fetchdep.sh vector-im element-web
+pushd element-web
+yarn link matrix-js-sdk
+yarn link matrix-react-sdk
+yarn install
+yarn build:res
+popd
diff --git a/scripts/fetchdep.sh b/scripts/fetchdep.sh
index 0142305797..850eef25ec 100755
--- a/scripts/fetchdep.sh
+++ b/scripts/fetchdep.sh
@@ -34,7 +34,7 @@ elif [[ "${#BUILDKITE_BRANCH_ARRAY[@]}" == "2" ]]; then
fi
# Try the target branch of the push or PR.
clone $deforg $defrepo $BUILDKITE_PULL_REQUEST_BASE_BRANCH
-# Try the current branch from Jenkins.
-clone $deforg $defrepo `"echo $GIT_BRANCH" | sed -e 's/^origin\///'`
+# Try HEAD which is the branch name in Netlify (not BRANCH which is pull/xxxx/head for PR builds)
+clone $deforg $defrepo $HEAD
# Use the default branch as the last resort.
clone $deforg $defrepo $defbranch
diff --git a/src/BasePlatform.ts b/src/BasePlatform.ts
index 0a1f06f0b3..c301aa6a10 100644
--- a/src/BasePlatform.ts
+++ b/src/BasePlatform.ts
@@ -18,6 +18,7 @@ limitations under the License.
*/
import {MatrixClient} from "matrix-js-sdk/src/client";
+import {encodeUnpaddedBase64} from "matrix-js-sdk/src/crypto/olmlib";
import dis from './dispatcher/dispatcher';
import BaseEventIndexManager from './indexing/BaseEventIndexManager';
import {ActionPayload} from "./dispatcher/payloads";
@@ -25,6 +26,7 @@ import {CheckUpdatesPayload} from "./dispatcher/payloads/CheckUpdatesPayload";
import {Action} from "./dispatcher/actions";
import {hideToast as hideUpdateToast} from "./toasts/UpdateToast";
import {MatrixClientPeg} from "./MatrixClientPeg";
+import {idbLoad, idbSave, idbDelete} from "./utils/StorageManager";
export const SSO_HOMESERVER_URL_KEY = "mx_sso_hs_url";
export const SSO_ID_SERVER_URL_KEY = "mx_sso_is_url";
@@ -248,15 +250,16 @@ export default abstract class BasePlatform {
* @param {MatrixClient} mxClient the matrix client using which we should start the flow
* @param {"sso"|"cas"} loginType the type of SSO it is, CAS/SSO.
* @param {string} fragmentAfterLogin the hash to pass to the app during sso callback.
+ * @param {string} idpId The ID of the Identity Provider being targeted, optional.
*/
- startSingleSignOn(mxClient: MatrixClient, loginType: "sso" | "cas", fragmentAfterLogin: string) {
+ startSingleSignOn(mxClient: MatrixClient, loginType: "sso" | "cas", fragmentAfterLogin: string, idpId?: string) {
// persist hs url and is url for when the user is returned to the app with the login token
localStorage.setItem(SSO_HOMESERVER_URL_KEY, mxClient.getHomeserverUrl());
if (mxClient.getIdentityServerUrl()) {
localStorage.setItem(SSO_ID_SERVER_URL_KEY, mxClient.getIdentityServerUrl());
}
const callbackUrl = this.getSSOCallbackUrl(fragmentAfterLogin);
- window.location.href = mxClient.getSsoLoginUrl(callbackUrl.toString(), loginType); // redirect to SSO
+ window.location.href = mxClient.getSsoLoginUrl(callbackUrl.toString(), loginType, idpId); // redirect to SSO
}
onKeyDown(ev: KeyboardEvent): boolean {
@@ -272,7 +275,40 @@ export default abstract class BasePlatform {
* pickle key has been stored.
*/
async getPickleKey(userId: string, deviceId: string): Promise {
- return null;
+ if (!window.crypto || !window.crypto.subtle) {
+ return null;
+ }
+ let data;
+ try {
+ data = await idbLoad("pickleKey", [userId, deviceId]);
+ } catch (e) {}
+ if (!data) {
+ return null;
+ }
+ if (!data.encrypted || !data.iv || !data.cryptoKey) {
+ console.error("Badly formatted pickle key");
+ return null;
+ }
+
+ const additionalData = new Uint8Array(userId.length + deviceId.length + 1);
+ for (let i = 0; i < userId.length; i++) {
+ additionalData[i] = userId.charCodeAt(i);
+ }
+ additionalData[userId.length] = 124; // "|"
+ for (let i = 0; i < deviceId.length; i++) {
+ additionalData[userId.length + 1 + i] = deviceId.charCodeAt(i);
+ }
+
+ try {
+ const key = await crypto.subtle.decrypt(
+ {name: "AES-GCM", iv: data.iv, additionalData}, data.cryptoKey,
+ data.encrypted,
+ );
+ return encodeUnpaddedBase64(key);
+ } catch (e) {
+ console.error("Error decrypting pickle key");
+ return null;
+ }
}
/**
@@ -283,7 +319,37 @@ export default abstract class BasePlatform {
* support storing pickle keys.
*/
async createPickleKey(userId: string, deviceId: string): Promise {
- return null;
+ if (!window.crypto || !window.crypto.subtle) {
+ return null;
+ }
+ const crypto = window.crypto;
+ const randomArray = new Uint8Array(32);
+ crypto.getRandomValues(randomArray);
+ const cryptoKey = await crypto.subtle.generateKey(
+ {name: "AES-GCM", length: 256}, false, ["encrypt", "decrypt"],
+ );
+ const iv = new Uint8Array(32);
+ crypto.getRandomValues(iv);
+
+ const additionalData = new Uint8Array(userId.length + deviceId.length + 1);
+ for (let i = 0; i < userId.length; i++) {
+ additionalData[i] = userId.charCodeAt(i);
+ }
+ additionalData[userId.length] = 124; // "|"
+ for (let i = 0; i < deviceId.length; i++) {
+ additionalData[userId.length + 1 + i] = deviceId.charCodeAt(i);
+ }
+
+ const encrypted = await crypto.subtle.encrypt(
+ {name: "AES-GCM", iv, additionalData}, cryptoKey, randomArray,
+ );
+
+ try {
+ await idbSave("pickleKey", [userId, deviceId], {encrypted, iv, cryptoKey});
+ } catch (e) {
+ return null;
+ }
+ return encodeUnpaddedBase64(randomArray);
}
/**
@@ -292,5 +358,8 @@ export default abstract class BasePlatform {
* @param {string} userId the device ID that the pickle key is for.
*/
async destroyPickleKey(userId: string, deviceId: string): Promise {
+ try {
+ await idbDelete("pickleKey", [userId, deviceId]);
+ } catch (e) {}
}
}
diff --git a/src/CallHandler.tsx b/src/CallHandler.tsx
index 3be203ab98..fac4d6fc4e 100644
--- a/src/CallHandler.tsx
+++ b/src/CallHandler.tsx
@@ -80,6 +80,8 @@ import { MatrixCall, CallErrorCode, CallState, CallEvent, CallParty, CallType }
import Analytics from './Analytics';
import CountlyAnalytics from "./CountlyAnalytics";
import {UIFeature} from "./settings/UIFeature";
+import { CallError } from "matrix-js-sdk/src/webrtc/call";
+import { logger } from 'matrix-js-sdk/src/logger';
enum AudioID {
Ring = 'ringAudio',
@@ -114,8 +116,9 @@ function getRemoteAudioElement(): HTMLAudioElement {
}
export default class CallHandler {
- private calls = new Map();
+ private calls = new Map(); // roomId -> call
private audioPromises = new Map>();
+ private dispatcherRef: string = null;
static sharedInstance() {
if (!window.mxCallHandler) {
@@ -126,7 +129,7 @@ export default class CallHandler {
}
start() {
- dis.register(this.onAction);
+ this.dispatcherRef = dis.register(this.onAction);
// add empty handlers for media actions, otherwise the media keys
// end up causing the audio elements with our ring/ringback etc
// audio clips in to play.
@@ -149,6 +152,10 @@ export default class CallHandler {
if (cli) {
cli.removeListener('Call.incoming', this.onCallIncoming);
}
+ if (this.dispatcherRef !== null) {
+ dis.unregister(this.dispatcherRef);
+ this.dispatcherRef = null;
+ }
}
private onCallIncoming = (call) => {
@@ -174,6 +181,28 @@ export default class CallHandler {
return null;
}
+ getAllActiveCalls() {
+ const activeCalls = [];
+
+ for (const call of this.calls.values()) {
+ if (call.state !== CallState.Ended && call.state !== CallState.Ringing) {
+ activeCalls.push(call);
+ }
+ }
+ return activeCalls;
+ }
+
+ getAllActiveCallsNotInRoom(notInThisRoomId) {
+ const callsNotInThatRoom = [];
+
+ for (const [roomId, call] of this.calls.entries()) {
+ if (roomId !== notInThisRoomId && call.state !== CallState.Ended) {
+ callsNotInThatRoom.push(call);
+ }
+ }
+ return callsNotInThatRoom;
+ }
+
play(audioId: AudioID) {
// TODO: Attach an invisible element for this instead
// which listens?
@@ -226,11 +255,17 @@ export default class CallHandler {
}
private setCallListeners(call: MatrixCall) {
- call.on(CallEvent.Error, (err) => {
+ call.on(CallEvent.Error, (err: CallError) => {
if (!this.matchesCallForThisRoom(call)) return;
- Analytics.trackEvent('voip', 'callError', 'error', err);
+ Analytics.trackEvent('voip', 'callError', 'error', err.toString());
console.error("Call error:", err);
+
+ if (err.code === CallErrorCode.NoUserMedia) {
+ this.showMediaCaptureError(call);
+ return;
+ }
+
if (
MatrixClientPeg.get().getTurnServers().length === 0 &&
SettingsStore.getValue("fallbackICEServerAllowed") === null
@@ -299,8 +334,9 @@ export default class CallHandler {
Modal.createTrackedDialog('Call Handler', 'Call Failed', ErrorDialog, {
title, description,
});
- } else if (call.hangupReason === CallErrorCode.AnsweredElsewhere) {
- this.play(AudioID.Busy);
+ } else if (
+ call.hangupReason === CallErrorCode.AnsweredElsewhere && oldState === CallState.Connecting
+ ) {
Modal.createTrackedDialog('Call Handler', 'Call Failed', ErrorDialog, {
title: _t("Answered Elsewhere"),
description: _t("The call was answered on another device."),
@@ -377,6 +413,34 @@ export default class CallHandler {
}, null, true);
}
+ private showMediaCaptureError(call: MatrixCall) {
+ let title;
+ let description;
+
+ if (call.type === CallType.Voice) {
+ title = _t("Unable to access microphone");
+ description =
+ {_t(
+ "Call failed because microphone could not be accessed. " +
+ "Check that a microphone is plugged in and set up correctly.",
+ )}
+
;
+ } else if (call.type === CallType.Video) {
+ title = _t("Unable to access webcam / microphone");
+ description =
+ {_t("Call failed because webcam or microphone could not be accessed. Check that:")}
+
+
{_t("A microphone and webcam are plugged in and set up correctly")}
+
{_t("Permission is granted to use the webcam")}
+
{_t("No other application is using the webcam")}
+
+
;
+ }
+
+ Modal.createTrackedDialog('Media capture failed', '', ErrorDialog, {
+ title, description,
+ }, null, true);
+ }
private placeCall(
roomId: string, type: PlaceCallType,
@@ -389,6 +453,8 @@ export default class CallHandler {
this.setCallListeners(call);
this.setCallAudioElement(call);
+ this.setActiveCallRoomId(roomId);
+
if (type === PlaceCallType.Voice) {
call.placeVoiceCall();
} else if (type === 'video') {
@@ -417,14 +483,6 @@ export default class CallHandler {
switch (payload.action) {
case 'place_call':
{
- if (this.getAnyActiveCall()) {
- Modal.createTrackedDialog('Call Handler', 'Existing Call', ErrorDialog, {
- title: _t('Existing Call'),
- description: _t('You are already in a call.'),
- });
- return; // don't allow >1 call to be placed.
- }
-
// if the runtime env doesn't do VoIP, whine.
if (!MatrixClientPeg.get().supportsVoip()) {
Modal.createTrackedDialog('Call Handler', 'VoIP is unsupported', ErrorDialog, {
@@ -434,6 +492,15 @@ export default class CallHandler {
return;
}
+ // don't allow > 2 calls to be placed.
+ if (this.getAllActiveCalls().length > 1) {
+ Modal.createTrackedDialog('Call Handler', 'Existing Call', ErrorDialog, {
+ title: _t('Too Many Calls'),
+ description: _t("You've reached the maximum number of simultaneous calls."),
+ });
+ return;
+ }
+
const room = MatrixClientPeg.get().getRoom(payload.room_id);
if (!room) {
console.error("Room %s does not exist.", payload.room_id);
@@ -477,24 +544,21 @@ export default class CallHandler {
break;
case 'incoming_call':
{
- if (this.getAnyActiveCall()) {
- // ignore multiple incoming calls. in future, we may want a line-1/line-2 setup.
- // we avoid rejecting with "busy" in case the user wants to answer it on a different device.
- // in future we could signal a "local busy" as a warning to the caller.
- // see https://github.com/vector-im/vector-web/issues/1964
- return;
- }
-
// if the runtime env doesn't do VoIP, stop here.
if (!MatrixClientPeg.get().supportsVoip()) {
return;
}
const call = payload.call as MatrixCall;
+
+ if (this.getCallForRoom(call.roomId)) {
+ // ignore multiple incoming calls to the same room
+ return;
+ }
+
Analytics.trackEvent('voip', 'receiveCall', 'type', call.type);
this.calls.set(call.roomId, call)
this.setCallListeners(call);
- this.setCallAudioElement(call);
}
break;
case 'hangup':
@@ -507,14 +571,26 @@ export default class CallHandler {
} else {
this.calls.get(payload.room_id).hangup(CallErrorCode.UserHangup, false);
}
- this.removeCallForRoom(payload.room_id);
+ // don't remove the call yet: let the hangup event handler do it (otherwise it will throw
+ // the hangup event away)
break;
case 'answer': {
if (!this.calls.has(payload.room_id)) {
return; // no call to answer
}
+
+ if (this.getAllActiveCalls().length > 1) {
+ Modal.createTrackedDialog('Call Handler', 'Existing Call', ErrorDialog, {
+ title: _t('Too Many Calls'),
+ description: _t("You've reached the maximum number of simultaneous calls."),
+ });
+ return;
+ }
+
const call = this.calls.get(payload.room_id);
call.answer();
+ this.setCallAudioElement(call);
+ this.setActiveCallRoomId(payload.room_id);
CountlyAnalytics.instance.trackJoinCall(payload.room_id, call.type === CallType.Video, false);
dis.dispatch({
action: "view_room",
@@ -525,6 +601,21 @@ export default class CallHandler {
}
}
+ setActiveCallRoomId(activeCallRoomId: string) {
+ logger.info("Setting call in room " + activeCallRoomId + " active");
+
+ for (const [roomId, call] of this.calls.entries()) {
+ if (call.state === CallState.Ended) continue;
+
+ if (roomId === activeCallRoomId) {
+ call.setRemoteOnHold(false);
+ } else {
+ logger.info("Holding call in room " + roomId + " because another call is being set active");
+ call.setRemoteOnHold(true);
+ }
+ }
+ }
+
private async startCallApp(roomId: string, type: string) {
dis.dispatch({
action: 'appsDrawer',
diff --git a/src/Lifecycle.ts b/src/Lifecycle.ts
index ac96d59b09..f87af1a791 100644
--- a/src/Lifecycle.ts
+++ b/src/Lifecycle.ts
@@ -21,6 +21,7 @@ limitations under the License.
import Matrix from 'matrix-js-sdk';
import { InvalidStoreError } from "matrix-js-sdk/src/errors";
import { MatrixClient } from "matrix-js-sdk/src/client";
+import {decryptAES, encryptAES} from "matrix-js-sdk/src/crypto/aes";
import {IMatrixClientCreds, MatrixClientPeg} from './MatrixClientPeg';
import SecurityCustomisations from "./customisations/Security";
@@ -147,20 +148,13 @@ export async function loadSession(opts: ILoadSessionOpts = {}): Promise
* Gets the user ID of the persisted session, if one exists. This does not validate
* that the user's credentials still work, just that they exist and that a user ID
* is associated with them. The session is not loaded.
- * @returns {String} The persisted session's owner, if an owner exists. Null otherwise.
+ * @returns {[String, bool]} The persisted session's owner and whether the stored
+ * session is for a guest user, if an owner exists. If there is no stored session,
+ * return [null, null].
*/
-export function getStoredSessionOwner(): string {
- const {hsUrl, userId, accessToken} = getLocalStorageSessionVars();
- return hsUrl && userId && accessToken ? userId : null;
-}
-
-/**
- * @returns {bool} True if the stored session is for a guest user or false if it is
- * for a real user. If there is no stored session, return null.
- */
-export function getStoredSessionIsGuest(): boolean {
- const sessVars = getLocalStorageSessionVars();
- return sessVars.hsUrl && sessVars.userId && sessVars.accessToken ? sessVars.isGuest : null;
+export async function getStoredSessionOwner(): Promise<[string, boolean]> {
+ const {hsUrl, userId, hasAccessToken, isGuest} = await getStoredSessionVars();
+ return hsUrl && userId && hasAccessToken ? [userId, isGuest] : [null, null];
}
/**
@@ -197,8 +191,8 @@ export function attemptTokenLogin(
},
).then(function(creds) {
console.log("Logged in with token");
- return clearStorage().then(() => {
- persistCredentialsToLocalStorage(creds);
+ return clearStorage().then(async () => {
+ await persistCredentials(creds);
// remember that we just logged in
sessionStorage.setItem("mx_fresh_login", String(true));
return true;
@@ -276,24 +270,42 @@ function registerAsGuest(
});
}
-export interface ILocalStorageSession {
+export interface IStoredSession {
hsUrl: string;
isUrl: string;
- accessToken: string;
+ hasAccessToken: boolean;
+ accessToken: string | object;
userId: string;
deviceId: string;
isGuest: boolean;
}
/**
- * Retrieves information about the stored session in localstorage. The session
+ * Retrieves information about the stored session from the browser's storage. The session
* may not be valid, as it is not tested for consistency here.
* @returns {Object} Information about the session - see implementation for variables.
*/
-export function getLocalStorageSessionVars(): ILocalStorageSession {
+export async function getStoredSessionVars(): Promise {
const hsUrl = localStorage.getItem(HOMESERVER_URL_KEY);
const isUrl = localStorage.getItem(ID_SERVER_URL_KEY);
- const accessToken = localStorage.getItem("mx_access_token");
+ let accessToken;
+ try {
+ accessToken = await StorageManager.idbLoad("account", "mx_access_token");
+ } catch (e) {}
+ if (!accessToken) {
+ accessToken = localStorage.getItem("mx_access_token");
+ if (accessToken) {
+ try {
+ // try to migrate access token to IndexedDB if we can
+ await StorageManager.idbSave("account", "mx_access_token", accessToken);
+ localStorage.removeItem("mx_access_token");
+ } catch (e) {}
+ }
+ }
+ // if we pre-date storing "mx_has_access_token", but we retrieved an access
+ // token, then we should say we have an access token
+ const hasAccessToken =
+ (localStorage.getItem("mx_has_access_token") === "true") || !!accessToken;
const userId = localStorage.getItem("mx_user_id");
const deviceId = localStorage.getItem("mx_device_id");
@@ -305,7 +317,43 @@ export function getLocalStorageSessionVars(): ILocalStorageSession {
isGuest = localStorage.getItem("matrix-is-guest") === "true";
}
- return {hsUrl, isUrl, accessToken, userId, deviceId, isGuest};
+ return {hsUrl, isUrl, hasAccessToken, accessToken, userId, deviceId, isGuest};
+}
+
+// The pickle key is a string of unspecified length and format. For AES, we
+// need a 256-bit Uint8Array. So we HKDF the pickle key to generate the AES
+// key. The AES key should be zeroed after it is used.
+async function pickleKeyToAesKey(pickleKey: string): Promise {
+ const pickleKeyBuffer = new Uint8Array(pickleKey.length);
+ for (let i = 0; i < pickleKey.length; i++) {
+ pickleKeyBuffer[i] = pickleKey.charCodeAt(i);
+ }
+ const hkdfKey = await window.crypto.subtle.importKey(
+ "raw", pickleKeyBuffer, "HKDF", false, ["deriveBits"],
+ );
+ pickleKeyBuffer.fill(0);
+ return new Uint8Array(await window.crypto.subtle.deriveBits(
+ {
+ name: "HKDF", hash: "SHA-256",
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
+ // @ts-ignore: https://github.com/microsoft/TypeScript-DOM-lib-generator/pull/879
+ salt: new Uint8Array(32), info: new Uint8Array(0),
+ },
+ hkdfKey,
+ 256,
+ ));
+}
+
+async function abortLogin() {
+ const signOut = await showStorageEvictedDialog();
+ if (signOut) {
+ await clearStorage();
+ // This error feels a bit clunky, but we want to make sure we don't go any
+ // further and instead head back to sign in.
+ throw new AbortLoginAndRebuildStorage(
+ "Aborting login in progress because of storage inconsistency",
+ );
+ }
}
// returns a promise which resolves to true if a session is found in
@@ -325,7 +373,11 @@ async function restoreFromLocalStorage(opts?: { ignoreGuest?: boolean }): Promis
return false;
}
- const {hsUrl, isUrl, accessToken, userId, deviceId, isGuest} = getLocalStorageSessionVars();
+ const {hsUrl, isUrl, hasAccessToken, accessToken, userId, deviceId, isGuest} = await getStoredSessionVars();
+
+ if (hasAccessToken && !accessToken) {
+ abortLogin();
+ }
if (accessToken && userId && hsUrl) {
if (ignoreGuest && isGuest) {
@@ -333,9 +385,15 @@ async function restoreFromLocalStorage(opts?: { ignoreGuest?: boolean }): Promis
return false;
}
+ let decryptedAccessToken = accessToken;
const pickleKey = await PlatformPeg.get().getPickleKey(userId, deviceId);
if (pickleKey) {
console.log("Got pickle key");
+ if (typeof accessToken !== "string") {
+ const encrKey = await pickleKeyToAesKey(pickleKey);
+ decryptedAccessToken = await decryptAES(accessToken, encrKey, "access_token");
+ encrKey.fill(0);
+ }
} else {
console.log("No pickle key available");
}
@@ -347,7 +405,7 @@ async function restoreFromLocalStorage(opts?: { ignoreGuest?: boolean }): Promis
await doSetLoggedIn({
userId: userId,
deviceId: deviceId,
- accessToken: accessToken,
+ accessToken: decryptedAccessToken as string,
homeserverUrl: hsUrl,
identityServerUrl: isUrl,
guest: isGuest,
@@ -486,15 +544,7 @@ async function doSetLoggedIn(
// crypto store, we'll be generally confused when handling encrypted data.
// Show a modal recommending a full reset of storage.
if (results.dataInLocalStorage && results.cryptoInited && !results.dataInCryptoStore) {
- const signOut = await showStorageEvictedDialog();
- if (signOut) {
- await clearStorage();
- // This error feels a bit clunky, but we want to make sure we don't go any
- // further and instead head back to sign in.
- throw new AbortLoginAndRebuildStorage(
- "Aborting login in progress because of storage inconsistency",
- );
- }
+ await abortLogin();
}
Analytics.setLoggedIn(credentials.guest, credentials.homeserverUrl);
@@ -516,7 +566,7 @@ async function doSetLoggedIn(
if (localStorage) {
try {
- persistCredentialsToLocalStorage(credentials);
+ await persistCredentials(credentials);
// make sure we don't think that it's a fresh login any more
sessionStorage.removeItem("mx_fresh_login");
} catch (e) {
@@ -545,18 +595,55 @@ function showStorageEvictedDialog(): Promise {
// `instanceof`. Babel 7 supports this natively in their class handling.
class AbortLoginAndRebuildStorage extends Error { }
-function persistCredentialsToLocalStorage(credentials: IMatrixClientCreds): void {
+async function persistCredentials(credentials: IMatrixClientCreds): Promise {
localStorage.setItem(HOMESERVER_URL_KEY, credentials.homeserverUrl);
if (credentials.identityServerUrl) {
localStorage.setItem(ID_SERVER_URL_KEY, credentials.identityServerUrl);
}
localStorage.setItem("mx_user_id", credentials.userId);
- localStorage.setItem("mx_access_token", credentials.accessToken);
localStorage.setItem("mx_is_guest", JSON.stringify(credentials.guest));
+ // store whether we expect to find an access token, to detect the case
+ // where IndexedDB is blown away
+ if (credentials.accessToken) {
+ localStorage.setItem("mx_has_access_token", "true");
+ } else {
+ localStorage.deleteItem("mx_has_access_token");
+ }
+
if (credentials.pickleKey) {
+ let encryptedAccessToken;
+ try {
+ // try to encrypt the access token using the pickle key
+ const encrKey = await pickleKeyToAesKey(credentials.pickleKey);
+ encryptedAccessToken = await encryptAES(credentials.accessToken, encrKey, "access_token");
+ encrKey.fill(0);
+ } catch (e) {
+ console.warn("Could not encrypt access token", e);
+ }
+ try {
+ // save either the encrypted access token, or the plain access
+ // token if we were unable to encrypt (e.g. if the browser doesn't
+ // have WebCrypto).
+ await StorageManager.idbSave(
+ "account", "mx_access_token",
+ encryptedAccessToken || credentials.accessToken,
+ );
+ } catch (e) {
+ // if we couldn't save to indexedDB, fall back to localStorage. We
+ // store the access token unencrypted since localStorage only saves
+ // strings.
+ localStorage.setItem("mx_access_token", credentials.accessToken);
+ }
localStorage.setItem("mx_has_pickle_key", String(true));
} else {
+ try {
+ await StorageManager.idbSave(
+ "account", "mx_access_token", credentials.accessToken,
+ );
+ } catch (e) {
+ localStorage.setItem("mx_access_token", credentials.accessToken);
+ }
if (localStorage.getItem("mx_has_pickle_key")) {
console.error("Expected a pickle key, but none provided. Encryption may not work.");
}
@@ -733,6 +820,10 @@ async function clearStorage(opts?: { deleteEverything?: boolean }): Promise {
diff --git a/src/Login.ts b/src/Login.ts
index ae4aa226ed..6493b244e0 100644
--- a/src/Login.ts
+++ b/src/Login.ts
@@ -29,10 +29,23 @@ interface ILoginOptions {
}
// TODO: Move this to JS SDK
-interface ILoginFlow {
- type: string;
+interface IPasswordFlow {
+ type: "m.login.password";
}
+export interface IIdentityProvider {
+ id: string;
+ name: string;
+ icon?: string;
+}
+
+export interface ISSOFlow {
+ type: "m.login.sso" | "m.login.cas";
+ "org.matrix.msc2858.identity_providers": IIdentityProvider[]; // Unstable prefix for MSC2858
+}
+
+export type LoginFlow = ISSOFlow | IPasswordFlow;
+
// TODO: Move this to JS SDK
/* eslint-disable camelcase */
interface ILoginParams {
@@ -48,9 +61,8 @@ export default class Login {
private hsUrl: string;
private isUrl: string;
private fallbackHsUrl: string;
- private currentFlowIndex: number;
// TODO: Flows need a type in JS SDK
- private flows: Array;
+ private flows: Array;
private defaultDeviceDisplayName: string;
private tempClient: MatrixClient;
@@ -63,7 +75,6 @@ export default class Login {
this.hsUrl = hsUrl;
this.isUrl = isUrl;
this.fallbackHsUrl = fallbackHsUrl;
- this.currentFlowIndex = 0;
this.flows = [];
this.defaultDeviceDisplayName = opts.defaultDeviceDisplayName;
this.tempClient = null; // memoize
@@ -100,27 +111,13 @@ export default class Login {
});
}
- public async getFlows(): Promise> {
+ public async getFlows(): Promise> {
const client = this.createTemporaryClient();
const { flows } = await client.loginFlows();
this.flows = flows;
- this.currentFlowIndex = 0;
- // technically the UI should display options for all flows for the
- // user to then choose one, so return all the flows here.
return this.flows;
}
- public chooseFlow(flowIndex): void {
- this.currentFlowIndex = flowIndex;
- }
-
- public getCurrentFlowStep(): string {
- // technically the flow can have multiple steps, but no one does this
- // for login so we can ignore it.
- const flowStep = this.flows[this.currentFlowIndex];
- return flowStep ? flowStep.type : null;
- }
-
public loginViaPassword(
username: string,
phoneCountry: string,
diff --git a/src/PasswordReset.js b/src/PasswordReset.js
index 9472ddc633..b38a9de960 100644
--- a/src/PasswordReset.js
+++ b/src/PasswordReset.js
@@ -40,10 +40,6 @@ export default class PasswordReset {
this.identityServerDomain = identityUrl ? identityUrl.split("://")[1] : null;
}
- doesServerRequireIdServerParam() {
- return this.client.doesServerRequireIdServerParam();
- }
-
/**
* Attempt to reset the user's password. This will trigger a side-effect of
* sending an email to the provided email address.
@@ -78,9 +74,6 @@ export default class PasswordReset {
sid: this.sessionId,
client_secret: this.clientSecret,
};
- if (await this.doesServerRequireIdServerParam()) {
- creds.id_server = this.identityServerDomain;
- }
try {
await this.client.setPassword({
diff --git a/src/SlashCommands.tsx b/src/SlashCommands.tsx
index 9af5ebcbfb..e2ae875ac3 100644
--- a/src/SlashCommands.tsx
+++ b/src/SlashCommands.tsx
@@ -46,6 +46,7 @@ import { EffectiveMembership, getEffectiveMembership, leaveRoomBehaviour } from
import SdkConfig from "./SdkConfig";
import SettingsStore from "./settings/SettingsStore";
import {UIFeature} from "./settings/UIFeature";
+import {CHAT_EFFECTS} from "./effects"
import CallHandler from "./CallHandler";
// XXX: workaround for https://github.com/microsoft/TypeScript/issues/31816
@@ -78,6 +79,7 @@ export const CommandCategories = {
"actions": _td("Actions"),
"admin": _td("Admin"),
"advanced": _td("Advanced"),
+ "effects": _td("Effects"),
"other": _td("Other"),
};
@@ -1094,6 +1096,30 @@ export const Commands = [
category: CommandCategories.messages,
hideCompletionAfterSpace: true,
}),
+
+ ...CHAT_EFFECTS.map((effect) => {
+ return new Command({
+ command: effect.command,
+ description: effect.description(),
+ args: '',
+ runFn: function(roomId, args) {
+ return success((async () => {
+ if (!args) {
+ args = effect.fallbackMessage();
+ MatrixClientPeg.get().sendEmoteMessage(roomId, args);
+ } else {
+ const content = {
+ msgtype: effect.msgType,
+ body: args,
+ };
+ MatrixClientPeg.get().sendMessage(roomId, content);
+ }
+ dis.dispatch({action: `effects.${effect.command}`});
+ })());
+ },
+ category: CommandCategories.effects,
+ })
+ }),
];
// build a map from names and aliases to the Command objects.
diff --git a/src/components/structures/ContextMenu.tsx b/src/components/structures/ContextMenu.tsx
index fa0d6682dd..190b231b74 100644
--- a/src/components/structures/ContextMenu.tsx
+++ b/src/components/structures/ContextMenu.tsx
@@ -398,7 +398,7 @@ export const toRightOf = (elementRect: DOMRect, chevronOffset = 12) => {
};
// Placement method for to position context menu right-aligned and flowing to the left of elementRect
-export const aboveLeftOf = (elementRect: DOMRect, chevronFace = ChevronFace.None) => {
+export const aboveLeftOf = (elementRect: DOMRect, chevronFace = ChevronFace.None, vPadding = 0) => {
const menuOptions: IPosition & { chevronFace: ChevronFace } = { chevronFace };
const buttonRight = elementRect.right + window.pageXOffset;
@@ -408,9 +408,9 @@ export const aboveLeftOf = (elementRect: DOMRect, chevronFace = ChevronFace.None
menuOptions.right = window.innerWidth - buttonRight;
// Align the menu vertically on whichever side of the button has more space available.
if (buttonBottom < window.innerHeight / 2) {
- menuOptions.top = buttonBottom;
+ menuOptions.top = buttonBottom + vPadding;
} else {
- menuOptions.bottom = window.innerHeight - buttonTop;
+ menuOptions.bottom = (window.innerHeight - buttonTop) + vPadding;
}
return menuOptions;
diff --git a/src/components/structures/MatrixChat.tsx b/src/components/structures/MatrixChat.tsx
index 9fede15aa6..4a8d3cc718 100644
--- a/src/components/structures/MatrixChat.tsx
+++ b/src/components/structures/MatrixChat.tsx
@@ -34,7 +34,6 @@ import { DecryptionFailureTracker } from "../../DecryptionFailureTracker";
import { MatrixClientPeg, IMatrixClientCreds } from "../../MatrixClientPeg";
import PlatformPeg from "../../PlatformPeg";
import SdkConfig from "../../SdkConfig";
-import * as RoomListSorter from "../../RoomListSorter";
import dis from "../../dispatcher/dispatcher";
import Notifier from '../../Notifier';
@@ -48,7 +47,6 @@ import * as Lifecycle from '../../Lifecycle';
// LifecycleStore is not used but does listen to and dispatch actions
import '../../stores/LifecycleStore';
import PageTypes from '../../PageTypes';
-import { getHomePageUrl } from '../../utils/pages';
import createRoom from "../../createRoom";
import {_t, _td, getCurrentLanguage} from '../../languageHandler';
@@ -591,7 +589,7 @@ export default class MatrixChat extends React.PureComponent {
MatrixClientPeg.get().leave(payload.room_id).then(() => {
modal.close();
if (this.state.currentRoomId === payload.room_id) {
- dis.dispatch({action: 'view_next_room'});
+ dis.dispatch({action: 'view_home_page'});
}
}, (err) => {
modal.close();
@@ -620,9 +618,6 @@ export default class MatrixChat extends React.PureComponent {
}
break;
}
- case 'view_next_room':
- this.viewNextRoom(1);
- break;
case Action.ViewUserSettings: {
const tabPayload = payload as OpenToTabPayload;
const UserSettingsDialog = sdk.getComponent("dialogs.UserSettingsDialog");
@@ -802,35 +797,6 @@ export default class MatrixChat extends React.PureComponent {
this.notifyNewScreen('register');
}
- // TODO: Move to RoomViewStore
- private viewNextRoom(roomIndexDelta: number) {
- const allRooms = RoomListSorter.mostRecentActivityFirst(
- MatrixClientPeg.get().getRooms(),
- );
- // If there are 0 rooms or 1 room, view the home page because otherwise
- // if there are 0, we end up trying to index into an empty array, and
- // if there is 1, we end up viewing the same room.
- if (allRooms.length < 2) {
- dis.dispatch({
- action: 'view_home_page',
- });
- return;
- }
- let roomIndex = -1;
- for (let i = 0; i < allRooms.length; ++i) {
- if (allRooms[i].roomId === this.state.currentRoomId) {
- roomIndex = i;
- break;
- }
- }
- roomIndex = (roomIndex + roomIndexDelta) % allRooms.length;
- if (roomIndex < 0) roomIndex = allRooms.length - 1;
- dis.dispatch({
- action: 'view_room',
- room_id: allRooms[roomIndex].roomId,
- });
- }
-
// switch view to the given room
//
// @param {Object} roomInfo Object containing data about the room to be joined
@@ -1097,9 +1063,9 @@ export default class MatrixChat extends React.PureComponent {
private forgetRoom(roomId: string) {
MatrixClientPeg.get().forget(roomId).then(() => {
- // Switch to another room view if we're currently viewing the historical room
+ // Switch to home page if we're currently viewing the forgotten room
if (this.state.currentRoomId === roomId) {
- dis.dispatch({ action: "view_next_room" });
+ dis.dispatch({ action: "view_home_page" });
}
}).catch((err) => {
const errCode = err.errcode || _td("unknown error code");
@@ -1233,12 +1199,8 @@ export default class MatrixChat extends React.PureComponent {
} else {
if (MatrixClientPeg.get().isGuest()) {
dis.dispatch({action: 'view_welcome_page'});
- } else if (getHomePageUrl(this.props.config)) {
- dis.dispatch({action: 'view_home_page'});
} else {
- this.firstSyncPromise.promise.then(() => {
- dis.dispatch({action: 'view_next_room'});
- });
+ dis.dispatch({action: 'view_home_page'});
}
}
}
@@ -2009,6 +1971,7 @@ export default class MatrixChat extends React.PureComponent {
onLoginClick={this.onLoginClick}
onServerConfigChange={this.onServerConfigChange}
defaultDeviceDisplayName={this.props.defaultDeviceDisplayName}
+ fragmentAfterLogin={fragmentAfterLogin}
{...this.getServerProperties()}
/>
);
diff --git a/src/components/structures/RoomView.tsx b/src/components/structures/RoomView.tsx
index f4f7c6ceec..0ee847fbc9 100644
--- a/src/components/structures/RoomView.tsx
+++ b/src/components/structures/RoomView.tsx
@@ -69,11 +69,15 @@ import AuxPanel from "../views/rooms/AuxPanel";
import RoomHeader from "../views/rooms/RoomHeader";
import {XOR} from "../../@types/common";
import { IThreepidInvite } from "../../stores/ThreepidInviteStore";
+import EffectsOverlay from "../views/elements/EffectsOverlay";
+import {containsEmoji} from '../../effects/utils';
+import {CHAT_EFFECTS} from '../../effects';
import { CallState, MatrixCall } from "matrix-js-sdk/src/webrtc/call";
import WidgetStore from "../../stores/WidgetStore";
import {UPDATE_EVENT} from "../../stores/AsyncStore";
import Notifier from "../../Notifier";
import {showToast as showNotificationsToast} from "../../toasts/DesktopNotificationsToast";
+import { RoomNotificationStateStore } from "../../stores/notifications/RoomNotificationStateStore";
const DEBUG = false;
let debuglog = function(msg: string) {};
@@ -248,6 +252,8 @@ export default class RoomView extends React.Component {
this.context.on("deviceVerificationChanged", this.onDeviceVerificationChanged);
this.context.on("userTrustStatusChanged", this.onUserVerificationChanged);
this.context.on("crossSigning.keysChanged", this.onCrossSigningKeysChanged);
+ this.context.on("Event.decrypted", this.onEventDecrypted);
+ this.context.on("event", this.onEvent);
// Start listening for RoomViewStore updates
this.roomStoreToken = RoomViewStore.addListener(this.onRoomViewStoreUpdate);
this.rightPanelStoreToken = RightPanelStore.getSharedInstance().addListener(this.onRightPanelStoreUpdate);
@@ -581,6 +587,8 @@ export default class RoomView extends React.Component {
this.context.removeListener("deviceVerificationChanged", this.onDeviceVerificationChanged);
this.context.removeListener("userTrustStatusChanged", this.onUserVerificationChanged);
this.context.removeListener("crossSigning.keysChanged", this.onCrossSigningKeysChanged);
+ this.context.removeListener("Event.decrypted", this.onEventDecrypted);
+ this.context.removeListener("event", this.onEvent);
}
window.removeEventListener('beforeunload', this.onPageUnload);
@@ -781,6 +789,30 @@ export default class RoomView extends React.Component {
}
};
+ private onEventDecrypted = (ev) => {
+ if (ev.isDecryptionFailure()) return;
+ this.handleEffects(ev);
+ };
+
+ private onEvent = (ev) => {
+ if (ev.isBeingDecrypted() || ev.isDecryptionFailure()) return;
+ this.handleEffects(ev);
+ };
+
+ private handleEffects = (ev) => {
+ if (!this.state.room || !this.state.matrixClientIsReady) return; // not ready at all
+ if (ev.getRoomId() !== this.state.room.roomId) return; // not for us
+
+ const notifState = RoomNotificationStateStore.instance.getRoomState(this.state.room);
+ if (!notifState.isUnread) return;
+
+ CHAT_EFFECTS.forEach(effect => {
+ if (containsEmoji(ev.getContent(), effect.emojis) || ev.getContent().msgtype === effect.msgType) {
+ dis.dispatch({action: `effects.${effect.command}`});
+ }
+ });
+ };
+
private onRoomName = (room: Room) => {
if (this.state.room && room.roomId == this.state.room.roomId) {
this.forceUpdate();
@@ -1332,7 +1364,7 @@ export default class RoomView extends React.Component {
rejecting: true,
});
this.context.leave(this.state.roomId).then(() => {
- dis.dispatch({ action: 'view_next_room' });
+ dis.dispatch({ action: 'view_home_page' });
this.setState({
rejecting: false,
});
@@ -1366,7 +1398,7 @@ export default class RoomView extends React.Component {
await this.context.setIgnoredUsers(ignoredUsers);
await this.context.leave(this.state.roomId);
- dis.dispatch({ action: 'view_next_room' });
+ dis.dispatch({ action: 'view_home_page' });
this.setState({
rejecting: false,
});
@@ -1946,9 +1978,14 @@ export default class RoomView extends React.Component {
mx_RoomView_inCall: Boolean(activeCall),
});
+ const showChatEffects = SettingsStore.getValue('showChatEffects');
+
return (
+ {showChatEffects && this.roomView.current &&
+
+ }
{
// /read_markers API is not implemented on this HS, fallback to just RR
if (e.errcode === 'M_UNRECOGNIZED' && lastReadEvent) {
return MatrixClientPeg.get().sendReadReceipt(
lastReadEvent,
- {hidden: hiddenRR},
+ {},
).catch((e) => {
console.error(e);
this.lastRRSentEventId = undefined;
diff --git a/src/components/structures/auth/ForgotPassword.js b/src/components/structures/auth/ForgotPassword.js
index f9f5263f7e..5a39fe9fd9 100644
--- a/src/components/structures/auth/ForgotPassword.js
+++ b/src/components/structures/auth/ForgotPassword.js
@@ -21,16 +21,14 @@ import PropTypes from 'prop-types';
import { _t } from '../../../languageHandler';
import * as sdk from '../../../index';
import Modal from "../../../Modal";
-import SdkConfig from "../../../SdkConfig";
import PasswordReset from "../../../PasswordReset";
import AutoDiscoveryUtils, {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
import classNames from 'classnames';
import AuthPage from "../../views/auth/AuthPage";
import CountlyAnalytics from "../../../CountlyAnalytics";
+import ServerPicker from "../../views/elements/ServerPicker";
// Phases
-// Show controls to configure server details
-const PHASE_SERVER_DETAILS = 0;
// Show the forgot password inputs
const PHASE_FORGOT = 1;
// Email is in the process of being sent
@@ -62,7 +60,6 @@ export default class ForgotPassword extends React.Component {
serverIsAlive: true,
serverErrorIsFatal: false,
serverDeadError: "",
- serverRequiresIdServer: null,
};
constructor(props) {
@@ -93,12 +90,8 @@ export default class ForgotPassword extends React.Component {
serverConfig.isUrl,
);
- const pwReset = new PasswordReset(serverConfig.hsUrl, serverConfig.isUrl);
- const serverRequiresIdServer = await pwReset.doesServerRequireIdServerParam();
-
this.setState({
serverIsAlive: true,
- serverRequiresIdServer,
});
} catch (e) {
this.setState(AutoDiscoveryUtils.authComponentStateForError(e, "forgot_password"));
@@ -177,20 +170,6 @@ export default class ForgotPassword extends React.Component {
});
};
- onServerDetailsNextPhaseClick = async () => {
- this.setState({
- phase: PHASE_FORGOT,
- });
- };
-
- onEditServerDetailsClick = ev => {
- ev.preventDefault();
- ev.stopPropagation();
- this.setState({
- phase: PHASE_SERVER_DETAILS,
- });
- };
-
onLoginClick = ev => {
ev.preventDefault();
ev.stopPropagation();
@@ -205,24 +184,6 @@ export default class ForgotPassword extends React.Component {
});
}
- renderServerDetails() {
- const ServerConfig = sdk.getComponent("auth.ServerConfig");
-
- if (SdkConfig.get()['disable_custom_urls']) {
- return null;
- }
-
- return ;
- }
-
renderForgot() {
const Field = sdk.getComponent('elements.Field');
@@ -246,57 +207,13 @@ export default class ForgotPassword extends React.Component {
);
}
- let yourMatrixAccountText = _t('Your Matrix account on %(serverName)s', {
- serverName: this.props.serverConfig.hsName,
- });
- if (this.props.serverConfig.hsNameIsDifferent) {
- const TextWithTooltip = sdk.getComponent("elements.TextWithTooltip");
-
- yourMatrixAccountText = _t('Your Matrix account on ', {}, {
- 'underlinedServerName': () => {
- return
- {this.props.serverConfig.hsName}
- ;
- },
- });
- }
-
- // If custom URLs are allowed, wire up the server details edit link.
- let editLink = null;
- if (!SdkConfig.get()['disable_custom_urls']) {
- editLink =
- {_t('Change')}
- ;
- }
-
- if (!this.props.serverConfig.isUrl && this.state.serverRequiresIdServer) {
- return
-
- {yourMatrixAccountText}
- {editLink}
-
- {_t(
- "No identity server is configured: " +
- "add one in server settings to reset your password.",
- )}
-
- {_t('Sign in instead')}
-
-
;
- }
-
return
{errorText}
{serverDeadSection}
-
- {yourMatrixAccountText}
- {editLink}
-
+
;
} else if (SettingsStore.getValue(UIFeature.Registration)) {
footer = (
-
- { _t('Create account') }
-
+
+ {_t("New? Create account", {}, {
+ a: sub => { sub },
+ })}
+
);
}
@@ -686,8 +596,11 @@ export default class LoginComponent extends React.Component {
{ errorTextSection }
{ serverDeadSection }
- { this.renderServerComponent() }
- { this.renderLoginComponentForStep() }
+
+ { this.renderLoginComponentForFlows() }
{ footer }
diff --git a/src/components/structures/auth/Registration.tsx b/src/components/structures/auth/Registration.tsx
index f97f20cf59..095f3d3433 100644
--- a/src/components/structures/auth/Registration.tsx
+++ b/src/components/structures/auth/Registration.tsx
@@ -15,29 +15,21 @@ limitations under the License.
*/
import Matrix from 'matrix-js-sdk';
-import React, {ComponentProps, ReactNode} from 'react';
+import React, {ReactNode} from 'react';
import {MatrixClient} from "matrix-js-sdk/src/client";
import * as sdk from '../../../index';
import { _t, _td } from '../../../languageHandler';
-import SdkConfig from '../../../SdkConfig';
import { messageForResourceLimitError } from '../../../utils/ErrorUtils';
-import * as ServerType from '../../views/auth/ServerTypeSelector';
import AutoDiscoveryUtils, {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
import classNames from "classnames";
import * as Lifecycle from '../../../Lifecycle';
import {MatrixClientPeg} from "../../../MatrixClientPeg";
import AuthPage from "../../views/auth/AuthPage";
-import Login from "../../../Login";
+import Login, {ISSOFlow} from "../../../Login";
import dis from "../../../dispatcher/dispatcher";
-
-// Phases
-enum Phase {
- // Show controls to configure server details
- ServerDetails = 0,
- // Show the appropriate registration flow(s) for the server
- Registration = 1,
-}
+import SSOButtons from "../../views/elements/SSOButtons";
+import ServerPicker from '../../views/elements/ServerPicker';
interface IProps {
serverConfig: ValidatedServerConfig;
@@ -47,6 +39,7 @@ interface IProps {
clientSecret?: string;
sessionId?: string;
idSid?: string;
+ fragmentAfterLogin?: string;
// Called when the user has logged in. Params:
// - object with userId, deviceId, homeserverUrl, identityServerUrl, accessToken
@@ -92,9 +85,6 @@ interface IState {
// If set, we've registered but are not going to log
// the user in to their new account automatically.
completedNoSignin: boolean;
- serverType: ServerType.FREE | ServerType.PREMIUM | ServerType.ADVANCED;
- // Phase of the overall registration dialog.
- phase: Phase;
flows: {
stages: string[];
}[];
@@ -109,23 +99,22 @@ interface IState {
// Our matrix client - part of state because we can't render the UI auth
// component without it.
matrixClient?: MatrixClient;
- // whether the HS requires an ID server to register with a threepid
- serverRequiresIdServer?: boolean;
// The user ID we've just registered
registeredUsername?: string;
// if a different user ID to the one we just registered is logged in,
// this is the user ID that's logged in.
differentLoggedInUserId?: string;
+ // the SSO flow definition, this is fetched from /login as that's the only
+ // place it is exposed.
+ ssoFlow?: ISSOFlow;
}
-// Enable phases for registration
-const PHASES_ENABLED = true;
-
export default class Registration extends React.Component {
+ loginLogic: Login;
+
constructor(props) {
super(props);
- const serverType = ServerType.getTypeFromServerConfig(this.props.serverConfig);
this.state = {
busy: false,
errorText: null,
@@ -133,14 +122,17 @@ export default class Registration extends React.Component {
email: this.props.email,
},
doingUIAuth: Boolean(this.props.sessionId),
- serverType,
- phase: Phase.Registration,
flows: null,
completedNoSignin: false,
serverIsAlive: true,
serverErrorIsFatal: false,
serverDeadError: "",
};
+
+ const {hsUrl, isUrl} = this.props.serverConfig;
+ this.loginLogic = new Login(hsUrl, isUrl, null, {
+ defaultDeviceDisplayName: "Element login check", // We shouldn't ever be used
+ });
}
componentDidMount() {
@@ -154,61 +146,8 @@ export default class Registration extends React.Component {
newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return;
this.replaceClient(newProps.serverConfig);
-
- // Handle cases where the user enters "https://matrix.org" for their server
- // from the advanced option - we should default to FREE at that point.
- const serverType = ServerType.getTypeFromServerConfig(newProps.serverConfig);
- if (serverType !== this.state.serverType) {
- // Reset the phase to default phase for the server type.
- this.setState({
- serverType,
- phase: Registration.getDefaultPhaseForServerType(serverType),
- });
- }
}
- private static getDefaultPhaseForServerType(type: IState["serverType"]) {
- switch (type) {
- case ServerType.FREE: {
- // Move directly to the registration phase since the server
- // details are fixed.
- return Phase.Registration;
- }
- case ServerType.PREMIUM:
- case ServerType.ADVANCED:
- return Phase.ServerDetails;
- }
- }
-
- private onServerTypeChange = (type: IState["serverType"]) => {
- this.setState({
- serverType: type,
- });
-
- // When changing server types, set the HS / IS URLs to reasonable defaults for the
- // the new type.
- switch (type) {
- case ServerType.FREE: {
- const { serverConfig } = ServerType.TYPES.FREE;
- this.props.onServerConfigChange(serverConfig);
- break;
- }
- case ServerType.PREMIUM:
- // We can accept whatever server config was the default here as this essentially
- // acts as a slightly different "custom server"/ADVANCED option.
- break;
- case ServerType.ADVANCED:
- // Use the default config from the config
- this.props.onServerConfigChange(SdkConfig.get()["validated_server_config"]);
- break;
- }
-
- // Reset the phase to default phase for the server type.
- this.setState({
- phase: Registration.getDefaultPhaseForServerType(type),
- });
- };
-
private async replaceClient(serverConfig: ValidatedServerConfig) {
this.setState({
errorText: null,
@@ -245,16 +184,20 @@ export default class Registration extends React.Component {
idBaseUrl: isUrl,
});
- let serverRequiresIdServer = true;
+ this.loginLogic.setHomeserverUrl(hsUrl);
+ this.loginLogic.setIdentityServerUrl(isUrl);
+
+ let ssoFlow: ISSOFlow;
try {
- serverRequiresIdServer = await cli.doesServerRequireIdServerParam();
+ const loginFlows = await this.loginLogic.getFlows();
+ ssoFlow = loginFlows.find(f => f.type === "m.login.sso" || f.type === "m.login.cas") as ISSOFlow;
} catch (e) {
- console.log("Unable to determine is server needs id_server param", e);
+ console.error("Failed to get login flows to check for SSO support", e);
}
this.setState({
matrixClient: cli,
- serverRequiresIdServer,
+ ssoFlow,
busy: false,
});
const showGenericError = (e) => {
@@ -282,26 +225,16 @@ export default class Registration extends React.Component {
// At this point registration is pretty much disabled, but before we do that let's
// quickly check to see if the server supports SSO instead. If it does, we'll send
// the user off to the login page to figure their account out.
- try {
- const loginLogic = new Login(hsUrl, isUrl, null, {
- defaultDeviceDisplayName: "Element login check", // We shouldn't ever be used
+ if (ssoFlow) {
+ // Redirect to login page - server probably expects SSO only
+ dis.dispatch({action: 'start_login'});
+ } else {
+ this.setState({
+ serverErrorIsFatal: true, // fatal because user cannot continue on this server
+ errorText: _t("Registration has been disabled on this homeserver."),
+ // add empty flows array to get rid of spinner
+ flows: [],
});
- const flows = await loginLogic.getFlows();
- const hasSsoFlow = flows.find(f => f.type === 'm.login.sso' || f.type === 'm.login.cas');
- if (hasSsoFlow) {
- // Redirect to login page - server probably expects SSO only
- dis.dispatch({action: 'start_login'});
- } else {
- this.setState({
- serverErrorIsFatal: true, // fatal because user cannot continue on this server
- errorText: _t("Registration has been disabled on this homeserver."),
- // add empty flows array to get rid of spinner
- flows: [],
- });
- }
- } catch (e) {
- console.error("Failed to get login flows to check for SSO support", e);
- showGenericError(e);
}
} else {
console.log("Unable to query for supported registration methods.", e);
@@ -365,6 +298,8 @@ export default class Registration extends React.Component {
if (!msisdnAvailable) {
msg = _t('This server does not support authentication with a phone number.');
}
+ } else if (response.errcode === "M_USER_IN_USE") {
+ msg = _t("That username already exists, please try another.");
}
this.setState({
busy: false,
@@ -390,8 +325,7 @@ export default class Registration extends React.Component {
// isn't a guest user since we'll usually have set a guest user session before
// starting the registration process. This isn't perfect since it's possible
// the user had a separate guest session they didn't actually mean to replace.
- const sessionOwner = Lifecycle.getStoredSessionOwner();
- const sessionIsGuest = Lifecycle.getStoredSessionIsGuest();
+ const [sessionOwner, sessionIsGuest] = await Lifecycle.getStoredSessionOwner();
if (sessionOwner && !sessionIsGuest && sessionOwner !== response.userId) {
console.log(
`Found a session for ${sessionOwner} but ${response.userId} has just registered.`,
@@ -453,21 +387,6 @@ export default class Registration extends React.Component {
this.setState({
busy: false,
doingUIAuth: false,
- phase: Phase.Registration,
- });
- };
-
- private onServerDetailsNextPhaseClick = async () => {
- this.setState({
- phase: Phase.Registration,
- });
- };
-
- private onEditServerDetailsClick = ev => {
- ev.preventDefault();
- ev.stopPropagation();
- this.setState({
- phase: Phase.ServerDetails,
});
};
@@ -516,77 +435,7 @@ export default class Registration extends React.Component {
}
};
- private renderServerComponent() {
- const ServerTypeSelector = sdk.getComponent("auth.ServerTypeSelector");
- const ServerConfig = sdk.getComponent("auth.ServerConfig");
- const ModularServerConfig = sdk.getComponent("auth.ModularServerConfig");
-
- if (SdkConfig.get()['disable_custom_urls']) {
- return null;
- }
-
- // Hide the server picker once the user is doing UI Auth unless encountered a fatal server error
- if (this.state.phase !== Phase.ServerDetails && this.state.doingUIAuth && !this.state.serverErrorIsFatal) {
- return null;
- }
-
- // If we're on a different phase, we only show the server type selector,
- // which is always shown if we allow custom URLs at all.
- // (if there's a fatal server error, we need to show the full server
- // config as the user may need to change servers to resolve the error).
- if (PHASES_ENABLED && this.state.phase !== Phase.ServerDetails && !this.state.serverErrorIsFatal) {
- return
;
- }
-
private renderRegisterComponent() {
- if (PHASES_ENABLED && this.state.phase !== Phase.Registration) {
- return null;
- }
-
const InteractiveAuth = sdk.getComponent('structures.InteractiveAuth');
const Spinner = sdk.getComponent('elements.Spinner');
const RegistrationForm = sdk.getComponent('auth.RegistrationForm');
@@ -610,18 +459,47 @@ export default class Registration extends React.Component {
;
} else if (this.state.flows.length) {
- return ;
+ let ssoSection;
+ if (this.state.ssoFlow) {
+ let continueWithSection;
+ const providers = this.state.ssoFlow["org.matrix.msc2858.identity_providers"] || [];
+ // when there is only a single (or 0) providers we show a wide button with `Continue with X` text
+ if (providers.length > 1) {
+ // i18n: ssoButtons is a placeholder to help translators understand context
+ continueWithSection =
}
+
{ this.renderRegisterComponent() }
{ goBack }
{ signIn }
diff --git a/src/components/structures/auth/SoftLogout.js b/src/components/structures/auth/SoftLogout.js
index a539c8c9ee..fdc1aec96d 100644
--- a/src/components/structures/auth/SoftLogout.js
+++ b/src/components/structures/auth/SoftLogout.js
@@ -24,8 +24,8 @@ import Modal from '../../../Modal';
import {MatrixClientPeg} from "../../../MatrixClientPeg";
import {sendLoginRequest} from "../../../Login";
import AuthPage from "../../views/auth/AuthPage";
-import SSOButton from "../../views/elements/SSOButton";
import {SSO_HOMESERVER_URL_KEY, SSO_ID_SERVER_URL_KEY} from "../../../BasePlatform";
+import SSOButtons from "../../views/elements/SSOButtons";
const LOGIN_VIEW = {
LOADING: 1,
@@ -101,10 +101,11 @@ export default class SoftLogout extends React.Component {
// Note: we don't use the existing Login class because it is heavily flow-based. We don't
// care about login flows here, unless it is the single flow we support.
const client = MatrixClientPeg.get();
- const loginViews = (await client.loginFlows()).flows.map(f => FLOWS_TO_VIEWS[f.type]);
+ const flows = (await client.loginFlows()).flows;
+ const loginViews = flows.map(f => FLOWS_TO_VIEWS[f.type]);
const chosenView = loginViews.filter(f => !!f)[0] || LOGIN_VIEW.UNSUPPORTED;
- this.setState({loginView: chosenView});
+ this.setState({ flows, loginView: chosenView });
}
onPasswordChange = (ev) => {
@@ -240,13 +241,18 @@ export default class SoftLogout extends React.Component {
introText = _t("Sign in and regain access to your account.");
} // else we already have a message and should use it (key backup warning)
+ const loginType = this.state.loginView === LOGIN_VIEW.CAS ? "cas" : "sso";
+ const flow = this.state.flows.find(flow => flow.type === "m.login." + loginType);
+
return (
{introText}
- flow.type === "m.login.password")}
/>
);
diff --git a/src/components/views/auth/CustomServerDialog.js b/src/components/views/auth/CustomServerDialog.js
deleted file mode 100644
index 138f8c4689..0000000000
--- a/src/components/views/auth/CustomServerDialog.js
+++ /dev/null
@@ -1,47 +0,0 @@
-/*
-Copyright 2015, 2016 OpenMarket 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.
-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 SdkConfig from '../../../SdkConfig';
-
-export default class CustomServerDialog extends React.Component {
- render() {
- const brand = SdkConfig.get().brand;
- return (
-
-
- { _t("Custom Server Options") }
-
-
-
{_t(
- "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.",
- { brand },
- )}
-
-
-
-
-
- );
- }
-}
diff --git a/src/components/views/auth/InteractiveAuthEntryComponents.js b/src/components/views/auth/InteractiveAuthEntryComponents.js
index 6628ca7120..60e57afc98 100644
--- a/src/components/views/auth/InteractiveAuthEntryComponents.js
+++ b/src/components/views/auth/InteractiveAuthEntryComponents.js
@@ -18,7 +18,6 @@ limitations under the License.
import React, {createRef} from 'react';
import PropTypes from 'prop-types';
-import url from 'url';
import classnames from 'classnames';
import * as sdk from '../../../index';
@@ -500,17 +499,11 @@ export class MsisdnAuthEntry extends React.Component {
});
try {
- const requiresIdServerParam =
- await this.props.matrixClient.doesServerRequireIdServerParam();
let result;
if (this._submitUrl) {
result = await this.props.matrixClient.submitMsisdnTokenOtherUrl(
this._submitUrl, this._sid, this.props.clientSecret, this.state.token,
);
- } else if (requiresIdServerParam) {
- result = await this.props.matrixClient.submitMsisdnToken(
- this._sid, this.props.clientSecret, this.state.token,
- );
} else {
throw new Error("The registration with MSISDN flow is misconfigured");
}
@@ -519,12 +512,6 @@ export class MsisdnAuthEntry extends React.Component {
sid: this._sid,
client_secret: this.props.clientSecret,
};
- if (requiresIdServerParam) {
- const idServerParsedUrl = url.parse(
- this.props.matrixClient.getIdentityServerUrl(),
- );
- creds.id_server = idServerParsedUrl.host;
- }
this.props.submitAuthDict({
type: MsisdnAuthEntry.LOGIN_TYPE,
// TODO: Remove `threepid_creds` once servers support proper UIA
diff --git a/src/components/views/auth/ModularServerConfig.js b/src/components/views/auth/ModularServerConfig.js
deleted file mode 100644
index 28fd16379d..0000000000
--- a/src/components/views/auth/ModularServerConfig.js
+++ /dev/null
@@ -1,124 +0,0 @@
-/*
-Copyright 2019 New Vector Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-import React from 'react';
-import * as sdk from '../../../index';
-import { _t } from '../../../languageHandler';
-import {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
-import SdkConfig from "../../../SdkConfig";
-import AutoDiscoveryUtils from "../../../utils/AutoDiscoveryUtils";
-import * as ServerType from '../../views/auth/ServerTypeSelector';
-import ServerConfig from "./ServerConfig";
-
-const MODULAR_URL = 'https://element.io/matrix-services' +
- '?utm_source=element-web&utm_medium=web&utm_campaign=element-web-authentication';
-
-// TODO: TravisR - Can this extend ServerConfig for most things?
-
-/*
- * Configure the Modular server name.
- *
- * This is a variant of ServerConfig with only the HS field and different body
- * text that is specific to the Modular case.
- */
-export default class ModularServerConfig extends ServerConfig {
- static propTypes = ServerConfig.propTypes;
-
- async validateAndApplyServer(hsUrl, isUrl) {
- // Always try and use the defaults first
- const defaultConfig: ValidatedServerConfig = SdkConfig.get()["validated_server_config"];
- if (defaultConfig.hsUrl === hsUrl && defaultConfig.isUrl === isUrl) {
- this.setState({busy: false, errorText: ""});
- this.props.onServerConfigChange(defaultConfig);
- return defaultConfig;
- }
-
- this.setState({
- hsUrl,
- isUrl,
- busy: true,
- errorText: "",
- });
-
- try {
- const result = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl, isUrl);
- this.setState({busy: false, errorText: ""});
- this.props.onServerConfigChange(result);
- return result;
- } catch (e) {
- console.error(e);
- let message = _t("Unable to validate homeserver/identity server");
- if (e.translatedMessage) {
- message = e.translatedMessage;
- }
- this.setState({
- busy: false,
- errorText: message,
- });
-
- return null;
- }
- }
-
- async validateServer() {
- // TODO: Do we want to support .well-known lookups here?
- // If for some reason someone enters "matrix.org" for a URL, we could do a lookup to
- // find their homeserver without demanding they use "https://matrix.org"
- return this.validateAndApplyServer(this.state.hsUrl, ServerType.TYPES.PREMIUM.identityServerUrl);
- }
-
- render() {
- const Field = sdk.getComponent('elements.Field');
- const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
-
- const submitButton = this.props.submitText
- ? {this.props.submitText}
- : null;
-
- return (
-
-
{_t("Your server")}
- {_t(
- "Enter the location of your Element Matrix Services homeserver. It may use your own " +
- "domain name or be a subdomain of element.io.",
- {}, {
- a: sub =>
- {sub}
- ,
- },
- )}
-
-
- );
- }
-}
diff --git a/src/components/views/auth/PasswordLogin.tsx b/src/components/views/auth/PasswordLogin.tsx
index fced2e08d0..84e583c3a5 100644
--- a/src/components/views/auth/PasswordLogin.tsx
+++ b/src/components/views/auth/PasswordLogin.tsx
@@ -1,5 +1,5 @@
/*
-Copyright 2015, 2016, 2017, 2019 New Vector Ltd.
+Copyright 2015, 2016, 2017, 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.
@@ -26,7 +26,6 @@ import withValidation from "../elements/Validation";
import * as Email from "../../../email";
import Field from "../elements/Field";
import CountryDropdown from "./CountryDropdown";
-import SignInToText from "./SignInToText";
// For validating phone numbers without country codes
const PHONE_NUMBER_REGEX = /^[0-9()\-\s]*$/;
@@ -47,7 +46,6 @@ interface IProps {
onUsernameBlur?(username: string): void;
onPhoneCountryChanged?(phoneCountry: string): void;
onPhoneNumberChanged?(phoneNumber: string): void;
- onEditServerDetailsClick?(): void;
onForgotPasswordClick?(): void;
}
@@ -70,7 +68,6 @@ enum LoginField {
*/
export default class PasswordLogin extends React.PureComponent {
static defaultProps = {
- onEditServerDetailsClick: null,
onUsernameChanged: function() {},
onUsernameBlur: function() {},
onPhoneCountryChanged: function() {},
@@ -296,7 +293,7 @@ export default class PasswordLogin extends React.PureComponent {
}, {
key: "number",
test: ({ value }) => !value || PHONE_NUMBER_REGEX.test(value),
- invalid: () => _t("Doesn't look like a valid phone number"),
+ invalid: () => _t("That phone number doesn't look quite right, please check and try again"),
},
],
});
@@ -357,6 +354,7 @@ export default class PasswordLogin extends React.PureComponent {
key="username_input"
type="text"
label={_t("Username")}
+ placeholder={_t("Username").toLocaleLowerCase()}
value={this.props.username}
onChange={this.onUsernameChanged}
onFocus={this.onUsernameFocus}
@@ -410,20 +408,14 @@ export default class PasswordLogin extends React.PureComponent {
let forgotPasswordJsx;
if (this.props.onForgotPasswordClick) {
- forgotPasswordJsx =
- {_t('Not sure of your password? Set a new one', {}, {
- a: sub => (
-
- {sub}
-
- ),
- })}
- ;
+ forgotPasswordJsx =
+ {_t("Forgot password?")}
+ ;
}
const pwFieldClass = classNames({
@@ -465,8 +457,6 @@ export default class PasswordLogin extends React.PureComponent {
return (
-
;
} else {
emailHelperText =
- {_t(
- "Set an email for account recovery. " +
- "Use email to optionally be discoverable by existing contacts.",
- )}
+ {
+ _t("Add an email to be able to reset your password.")
+ } {
+ _t("Use email to optionally be discoverable by existing contacts.")
+ }
diff --git a/src/components/views/auth/ServerConfig.js b/src/components/views/auth/ServerConfig.js
deleted file mode 100644
index e04bf9e25a..0000000000
--- a/src/components/views/auth/ServerConfig.js
+++ /dev/null
@@ -1,291 +0,0 @@
-/*
-Copyright 2015, 2016 OpenMarket Ltd
-Copyright 2019 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 PropTypes from 'prop-types';
-import Modal from '../../../Modal';
-import * as sdk from '../../../index';
-import { _t } from '../../../languageHandler';
-import {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
-import AutoDiscoveryUtils from "../../../utils/AutoDiscoveryUtils";
-import SdkConfig from "../../../SdkConfig";
-import { createClient } from 'matrix-js-sdk/src/matrix';
-import classNames from 'classnames';
-import CountlyAnalytics from "../../../CountlyAnalytics";
-
-/*
- * A pure UI component which displays the HS and IS to use.
- */
-
-export default class ServerConfig extends React.PureComponent {
- static propTypes = {
- onServerConfigChange: PropTypes.func.isRequired,
-
- // The current configuration that the user is expecting to change.
- serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired,
-
- delayTimeMs: PropTypes.number, // time to wait before invoking onChanged
-
- // Called after the component calls onServerConfigChange
- onAfterSubmit: PropTypes.func,
-
- // Optional text for the submit button. If falsey, no button will be shown.
- submitText: PropTypes.string,
-
- // Optional class for the submit button. Only applies if the submit button
- // is to be rendered.
- submitClass: PropTypes.string,
-
- // Whether the flow this component is embedded in requires an identity
- // server when the homeserver says it will need one. Default false.
- showIdentityServerIfRequiredByHomeserver: PropTypes.bool,
- };
-
- static defaultProps = {
- onServerConfigChange: function() {},
- delayTimeMs: 0,
- };
-
- constructor(props) {
- super(props);
-
- this.state = {
- busy: false,
- errorText: "",
- hsUrl: props.serverConfig.hsUrl,
- isUrl: props.serverConfig.isUrl,
- showIdentityServer: false,
- };
-
- CountlyAnalytics.instance.track("onboarding_custom_server");
- }
-
- // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
- UNSAFE_componentWillReceiveProps(newProps) { // eslint-disable-line camelcase
- if (newProps.serverConfig.hsUrl === this.state.hsUrl &&
- newProps.serverConfig.isUrl === this.state.isUrl) return;
-
- this.validateAndApplyServer(newProps.serverConfig.hsUrl, newProps.serverConfig.isUrl);
- }
-
- async validateServer() {
- // TODO: Do we want to support .well-known lookups here?
- // If for some reason someone enters "matrix.org" for a URL, we could do a lookup to
- // find their homeserver without demanding they use "https://matrix.org"
- const result = this.validateAndApplyServer(this.state.hsUrl, this.state.isUrl);
- if (!result) {
- return result;
- }
-
- // If the UI flow this component is embedded in requires an identity
- // server when the homeserver says it will need one, check first and
- // reveal this field if not already shown.
- // XXX: This a backward compatibility path for homeservers that require
- // an identity server to be passed during certain flows.
- // See also https://github.com/matrix-org/synapse/pull/5868.
- if (
- this.props.showIdentityServerIfRequiredByHomeserver &&
- !this.state.showIdentityServer &&
- await this.isIdentityServerRequiredByHomeserver()
- ) {
- this.setState({
- showIdentityServer: true,
- });
- return null;
- }
-
- return result;
- }
-
- async validateAndApplyServer(hsUrl, isUrl) {
- // Always try and use the defaults first
- const defaultConfig: ValidatedServerConfig = SdkConfig.get()["validated_server_config"];
- if (defaultConfig.hsUrl === hsUrl && defaultConfig.isUrl === isUrl) {
- this.setState({
- hsUrl: defaultConfig.hsUrl,
- isUrl: defaultConfig.isUrl,
- busy: false,
- errorText: "",
- });
- this.props.onServerConfigChange(defaultConfig);
- return defaultConfig;
- }
-
- this.setState({
- hsUrl,
- isUrl,
- busy: true,
- errorText: "",
- });
-
- try {
- const result = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl, isUrl);
- this.setState({busy: false, errorText: ""});
- this.props.onServerConfigChange(result);
- return result;
- } catch (e) {
- console.error(e);
-
- const stateForError = AutoDiscoveryUtils.authComponentStateForError(e);
- if (!stateForError.isFatalError) {
- this.setState({
- busy: false,
- });
- // carry on anyway
- const result = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl, isUrl, true);
- this.props.onServerConfigChange(result);
- return result;
- } else {
- let message = _t("Unable to validate homeserver/identity server");
- if (e.translatedMessage) {
- message = e.translatedMessage;
- }
- this.setState({
- busy: false,
- errorText: message,
- });
-
- return null;
- }
- }
- }
-
- async isIdentityServerRequiredByHomeserver() {
- // XXX: We shouldn't have to create a whole new MatrixClient just to
- // check if the homeserver requires an identity server... Should it be
- // extracted to a static utils function...?
- return createClient({
- baseUrl: this.state.hsUrl,
- }).doesServerRequireIdServerParam();
- }
-
- onHomeserverBlur = (ev) => {
- this._hsTimeoutId = this._waitThenInvoke(this._hsTimeoutId, () => {
- this.validateServer();
- });
- };
-
- onHomeserverChange = (ev) => {
- const hsUrl = ev.target.value;
- this.setState({ hsUrl });
- };
-
- onIdentityServerBlur = (ev) => {
- this._isTimeoutId = this._waitThenInvoke(this._isTimeoutId, () => {
- this.validateServer();
- });
- };
-
- onIdentityServerChange = (ev) => {
- const isUrl = ev.target.value;
- this.setState({ isUrl });
- };
-
- onSubmit = async (ev) => {
- ev.preventDefault();
- ev.stopPropagation();
- const result = await this.validateServer();
- if (!result) return; // Do not continue.
-
- if (this.props.onAfterSubmit) {
- this.props.onAfterSubmit();
- }
- };
-
- _waitThenInvoke(existingTimeoutId, fn) {
- if (existingTimeoutId) {
- clearTimeout(existingTimeoutId);
- }
- return setTimeout(fn.bind(this), this.props.delayTimeMs);
- }
-
- showHelpPopup = () => {
- const CustomServerDialog = sdk.getComponent('auth.CustomServerDialog');
- Modal.createTrackedDialog('Custom Server Dialog', '', CustomServerDialog);
- };
-
- _renderHomeserverSection() {
- const Field = sdk.getComponent('elements.Field');
- return
;
- }
-
- render() {
- const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
-
- const errorText = this.state.errorText
- ? {this.state.errorText}
- : null;
-
- const submitButton = this.props.submitText
- ? {this.props.submitText}
- : null;
-
- return (
-
- );
- }
-}
diff --git a/src/components/views/auth/ServerTypeSelector.js b/src/components/views/auth/ServerTypeSelector.js
deleted file mode 100644
index 71e7ac7f0e..0000000000
--- a/src/components/views/auth/ServerTypeSelector.js
+++ /dev/null
@@ -1,153 +0,0 @@
-/*
-Copyright 2019 New Vector Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-import React from 'react';
-import PropTypes from 'prop-types';
-import { _t } from '../../../languageHandler';
-import * as sdk from '../../../index';
-import classnames from 'classnames';
-import {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
-import {makeType} from "../../../utils/TypeUtils";
-
-const MODULAR_URL = 'https://element.io/matrix-services' +
- '?utm_source=element-web&utm_medium=web&utm_campaign=element-web-authentication';
-
-export const FREE = 'Free';
-export const PREMIUM = 'Premium';
-export const ADVANCED = 'Advanced';
-
-export const TYPES = {
- FREE: {
- id: FREE,
- label: () => _t('Free'),
- logo: () => ,
- description: () => _t('Join millions for free on the largest public server'),
- serverConfig: makeType(ValidatedServerConfig, {
- hsUrl: "https://matrix-client.matrix.org",
- hsName: "matrix.org",
- hsNameIsDifferent: false,
- isUrl: "https://vector.im",
- }),
- },
- PREMIUM: {
- id: PREMIUM,
- label: () => _t('Premium'),
- logo: () => ,
- description: () => _t('Premium hosting for organisations Learn more', {}, {
- a: sub =>
- {sub}
- ,
- }),
- identityServerUrl: "https://vector.im",
- },
- ADVANCED: {
- id: ADVANCED,
- label: () => _t('Advanced'),
- logo: () =>
-
- {_t('Other')}
-
,
- description: () => _t('Find other public servers or use a custom server'),
- },
-};
-
-export function getTypeFromServerConfig(config) {
- const {hsUrl} = config;
- if (!hsUrl) {
- return null;
- } else if (hsUrl === TYPES.FREE.serverConfig.hsUrl) {
- return FREE;
- } else if (new URL(hsUrl).hostname.endsWith('.modular.im')) {
- // This is an unlikely case to reach, as Modular defaults to hiding the
- // server type selector.
- return PREMIUM;
- } else {
- return ADVANCED;
- }
-}
-
-export default class ServerTypeSelector extends React.PureComponent {
- static propTypes = {
- // The default selected type.
- selected: PropTypes.string,
- // Handler called when the selected type changes.
- onChange: PropTypes.func.isRequired,
- };
-
- constructor(props) {
- super(props);
-
- const {
- selected,
- } = props;
-
- this.state = {
- selected,
- };
- }
-
- updateSelectedType(type) {
- if (this.state.selected === type) {
- return;
- }
- this.setState({
- selected: type,
- });
- if (this.props.onChange) {
- this.props.onChange(type);
- }
- }
-
- onClick = (e) => {
- e.stopPropagation();
- const type = e.currentTarget.dataset.id;
- this.updateSelectedType(type);
- };
-
- render() {
- const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
-
- const serverTypes = [];
- for (const type of Object.values(TYPES)) {
- const { id, label, logo, description } = type;
- const classes = classnames(
- "mx_ServerTypeSelector_type",
- `mx_ServerTypeSelector_type_${id}`,
- {
- "mx_ServerTypeSelector_type_selected": id === this.state.selected,
- },
- );
-
- serverTypes.push(
-
- {label()}
-
-
-
- {logo()}
-
-
- {description()}
-
-
-
);
- }
-
- return
- {serverTypes}
-
;
- }
-}
diff --git a/src/components/views/auth/SignInToText.js b/src/components/views/auth/SignInToText.js
deleted file mode 100644
index 7564096b7d..0000000000
--- a/src/components/views/auth/SignInToText.js
+++ /dev/null
@@ -1,62 +0,0 @@
-/*
-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 {_t} from "../../../languageHandler";
-import * as sdk from "../../../index";
-import PropTypes from "prop-types";
-import {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
-
-export default class SignInToText extends React.PureComponent {
- static propTypes = {
- serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired,
- onEditServerDetailsClick: PropTypes.func,
- };
-
- render() {
- let signInToText = _t('Sign in to your Matrix account on %(serverName)s', {
- serverName: this.props.serverConfig.hsName,
- });
- if (this.props.serverConfig.hsNameIsDifferent) {
- const TextWithTooltip = sdk.getComponent("elements.TextWithTooltip");
-
- signInToText = _t('Sign in to your Matrix account on ', {}, {
- 'underlinedServerName': () => {
- return
- {this.props.serverConfig.hsName}
- ;
- },
- });
- }
-
- let editLink = null;
- if (this.props.onEditServerDetailsClick) {
- editLink =
- {_t('Change')}
- ;
- }
-
- return
- {signInToText}
- {editLink}
-
;
- }
-}
diff --git a/src/components/views/context_menus/CallContextMenu.tsx b/src/components/views/context_menus/CallContextMenu.tsx
new file mode 100644
index 0000000000..336b72cebf
--- /dev/null
+++ b/src/components/views/context_menus/CallContextMenu.tsx
@@ -0,0 +1,59 @@
+/*
+Copyright 2020 New Vector Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+import React from 'react';
+import PropTypes from 'prop-types';
+import { _t } from '../../../languageHandler';
+import { ContextMenu, IProps as IContextMenuProps, MenuItem } from '../../structures/ContextMenu';
+import { MatrixCall } from 'matrix-js-sdk/src/webrtc/call';
+import CallHandler from '../../../CallHandler';
+
+interface IProps extends IContextMenuProps {
+ call: MatrixCall;
+}
+
+export default class CallContextMenu extends React.Component {
+ static propTypes = {
+ // js-sdk User object. Not required because it might not exist.
+ user: PropTypes.object,
+ };
+
+ constructor(props) {
+ super(props);
+ }
+
+ onHoldClick = () => {
+ this.props.call.setRemoteOnHold(true);
+ this.props.onFinished();
+ }
+
+ onUnholdClick = () => {
+ CallHandler.sharedInstance().setActiveCallRoomId(this.props.call.roomId);
+
+ this.props.onFinished();
+ }
+
+ render() {
+ const holdUnholdCaption = this.props.call.isRemoteOnHold() ? _t("Resume") : _t("Hold");
+ const handler = this.props.call.isRemoteOnHold() ? this.onUnholdClick : this.onHoldClick;
+
+ return
+
+ ;
+ }
+}
diff --git a/src/components/views/context_menus/MessageContextMenu.js b/src/components/views/context_menus/MessageContextMenu.js
index bc4514f8a6..6b871e4f24 100644
--- a/src/components/views/context_menus/MessageContextMenu.js
+++ b/src/components/views/context_menus/MessageContextMenu.js
@@ -149,7 +149,7 @@ export default class MessageContextMenu extends React.Component {
onRedactClick = () => {
const ConfirmRedactDialog = sdk.getComponent("dialogs.ConfirmRedactDialog");
Modal.createTrackedDialog('Confirm Redact Dialog', '', ConfirmRedactDialog, {
- onFinished: async (proceed) => {
+ onFinished: async (proceed, reason) => {
if (!proceed) return;
const cli = MatrixClientPeg.get();
@@ -157,6 +157,8 @@ export default class MessageContextMenu extends React.Component {
await cli.redactEvent(
this.props.mxEvent.getRoomId(),
this.props.mxEvent.getId(),
+ undefined,
+ reason ? { reason } : {},
);
} catch (e) {
const code = e.errcode || e.statusCode;
diff --git a/src/components/views/context_menus/WidgetContextMenu.tsx b/src/components/views/context_menus/WidgetContextMenu.tsx
index 7656e70341..8026942038 100644
--- a/src/components/views/context_menus/WidgetContextMenu.tsx
+++ b/src/components/views/context_menus/WidgetContextMenu.tsx
@@ -57,7 +57,7 @@ const WidgetContextMenu: React.FC = ({
let unpinButton;
if (showUnpin) {
const onUnpinClick = () => {
- WidgetStore.instance.unpinWidget(app.id);
+ WidgetStore.instance.unpinWidget(room.roomId, app.id);
onFinished();
};
@@ -143,7 +143,7 @@ const WidgetContextMenu: React.FC = ({
let moveLeftButton;
if (showUnpin && widgetIndex > 0) {
const onClick = () => {
- WidgetStore.instance.movePinnedWidget(app.id, -1);
+ WidgetStore.instance.movePinnedWidget(roomId, app.id, -1);
onFinished();
};
@@ -153,7 +153,7 @@ const WidgetContextMenu: React.FC = ({
let moveRightButton;
if (showUnpin && widgetIndex < pinnedWidgets.length - 1) {
const onClick = () => {
- WidgetStore.instance.movePinnedWidget(app.id, 1);
+ WidgetStore.instance.movePinnedWidget(roomId, app.id, 1);
onFinished();
};
diff --git a/src/components/views/dialogs/ConfirmRedactDialog.js b/src/components/views/dialogs/ConfirmRedactDialog.js
index 3106df1d5b..2216f9a93a 100644
--- a/src/components/views/dialogs/ConfirmRedactDialog.js
+++ b/src/components/views/dialogs/ConfirmRedactDialog.js
@@ -23,15 +23,17 @@ import { _t } from '../../../languageHandler';
*/
export default class ConfirmRedactDialog extends React.Component {
render() {
- const QuestionDialog = sdk.getComponent('views.dialogs.QuestionDialog');
+ const TextInputDialog = sdk.getComponent('views.dialogs.TextInputDialog');
return (
-
-
+
);
}
}
diff --git a/src/components/views/dialogs/InfoDialog.js b/src/components/views/dialogs/InfoDialog.js
index 8125bc3edd..97ae968ff3 100644
--- a/src/components/views/dialogs/InfoDialog.js
+++ b/src/components/views/dialogs/InfoDialog.js
@@ -31,6 +31,7 @@ export default class InfoDialog extends React.Component {
onFinished: PropTypes.func,
hasCloseButton: PropTypes.bool,
onKeyDown: PropTypes.func,
+ fixedWidth: PropTypes.bool,
};
static defaultProps = {
@@ -54,6 +55,7 @@ export default class InfoDialog extends React.Component {
contentId='mx_Dialog_content'
hasCancel={this.props.hasCloseButton}
onKeyDown={this.props.onKeyDown}
+ fixedWidth={this.props.fixedWidth}
>
{ this.props.description }
diff --git a/src/components/views/dialogs/InviteDialog.js b/src/components/views/dialogs/InviteDialog.tsx
similarity index 94%
rename from src/components/views/dialogs/InviteDialog.js
rename to src/components/views/dialogs/InviteDialog.tsx
index c039c191c5..8ccbbe473c 100644
--- a/src/components/views/dialogs/InviteDialog.js
+++ b/src/components/views/dialogs/InviteDialog.tsx
@@ -15,13 +15,12 @@ limitations under the License.
*/
import React, {createRef} from 'react';
-import PropTypes from 'prop-types';
import {_t} from "../../../languageHandler";
import * as sdk from "../../../index";
import {MatrixClientPeg} from "../../../MatrixClientPeg";
import {makeRoomPermalink, makeUserPermalink} from "../../../utils/permalinks/Permalinks";
import DMRoomMap from "../../../utils/DMRoomMap";
-import {RoomMember} from "matrix-js-sdk/src/matrix";
+import {RoomMember} from "matrix-js-sdk/src/models/room-member";
import SdkConfig from "../../../SdkConfig";
import {getHttpUriForMxc} from "matrix-js-sdk/src/content-repo";
import * as Email from "../../../email";
@@ -132,12 +131,12 @@ class ThreepidMember extends Member {
}
}
-class DMUserTile extends React.PureComponent {
- static propTypes = {
- member: PropTypes.object.isRequired, // Should be a Member (see interface above)
- onRemove: PropTypes.func, // takes 1 argument, the member being removed
- };
+interface IDMUserTileProps {
+ member: RoomMember;
+ onRemove: (RoomMember) => any;
+}
+class DMUserTile extends React.PureComponent {
_onRemove = (e) => {
// Stop the browser from highlighting text
e.preventDefault();
@@ -173,7 +172,9 @@ class DMUserTile extends React.PureComponent {
className='mx_InviteDialog_userTile_remove'
onClick={this._onRemove}
>
-
+
);
}
@@ -190,15 +191,15 @@ class DMUserTile extends React.PureComponent {
}
}
-class DMRoomTile extends React.PureComponent {
- static propTypes = {
- member: PropTypes.object.isRequired, // Should be a Member (see interface above)
- lastActiveTs: PropTypes.number,
- onToggle: PropTypes.func.isRequired, // takes 1 argument, the member being toggled
- highlightWord: PropTypes.string,
- isSelected: PropTypes.bool,
- };
+interface IDMRoomTileProps {
+ member: RoomMember;
+ lastActiveTs: number;
+ onToggle: (RoomMember) => any;
+ highlightWord: string;
+ isSelected: boolean;
+}
+class DMRoomTile extends React.PureComponent {
_onClick = (e) => {
// Stop the browser from highlighting text
e.preventDefault();
@@ -298,28 +299,45 @@ class DMRoomTile extends React.PureComponent {
}
}
-export default class InviteDialog extends React.PureComponent {
- static propTypes = {
- // Takes an array of user IDs/emails to invite.
- onFinished: PropTypes.func.isRequired,
+interface IInviteDialogProps {
+ // Takes an array of user IDs/emails to invite.
+ onFinished: (toInvite?: string[]) => any;
- // The kind of invite being performed. Assumed to be KIND_DM if
- // not provided.
- kind: PropTypes.string,
+ // The kind of invite being performed. Assumed to be KIND_DM if
+ // not provided.
+ kind: string,
- // The room ID this dialog is for. Only required for KIND_INVITE.
- roomId: PropTypes.string,
+ // The room ID this dialog is for. Only required for KIND_INVITE.
+ roomId: string,
- // Initial value to populate the filter with
- initialText: PropTypes.string,
- };
+ // Initial value to populate the filter with
+ initialText: string,
+}
+interface IInviteDialogState {
+ targets: RoomMember[]; // array of Member objects (see interface above)
+ filterText: string;
+ recents: { user: Member, userId: string }[];
+ numRecentsShown: number;
+ suggestions: { user: Member, userId: string }[];
+ numSuggestionsShown: number;
+ serverResultsMixin: { user: Member, userId: string }[];
+ threepidResultsMixin: { user: Member, userId: string}[];
+ canUseIdentityServer: boolean;
+ tryingIdentityServer: boolean;
+
+ // These two flags are used for the 'Go' button to communicate what is going on.
+ busy: boolean,
+ errorText: string,
+}
+
+export default class InviteDialog extends React.PureComponent {
static defaultProps = {
kind: KIND_DM,
initialText: "",
};
- _debounceTimer: number = null;
+ _debounceTimer: NodeJS.Timeout = null; // actually number because we're in the browser
_editorRef: any = null;
constructor(props) {
@@ -348,8 +366,8 @@ export default class InviteDialog extends React.PureComponent {
numRecentsShown: INITIAL_ROOMS_SHOWN,
suggestions: this._buildSuggestions(alreadyInvited),
numSuggestionsShown: INITIAL_ROOMS_SHOWN,
- serverResultsMixin: [], // { user: DirectoryMember, userId: string }[], like recents and suggestions
- threepidResultsMixin: [], // { user: ThreepidMember, userId: string}[], like recents and suggestions
+ serverResultsMixin: [],
+ threepidResultsMixin: [],
canUseIdentityServer: !!MatrixClientPeg.get().getIdentityServerUrl(),
tryingIdentityServer: false,
@@ -367,7 +385,7 @@ export default class InviteDialog extends React.PureComponent {
}
}
- static buildRecents(excludedTargetIds: Set): {userId: string, user: RoomMember, lastActive: number} {
+ static buildRecents(excludedTargetIds: Set): {userId: string, user: RoomMember, lastActive: number}[] {
const rooms = DMRoomMap.shared().getUniqueRoomsWithIndividuals(); // map of userId => js-sdk Room
// Also pull in all the rooms tagged as DefaultTagID.DM so we don't miss anything. Sometimes the
@@ -430,7 +448,7 @@ export default class InviteDialog extends React.PureComponent {
return recents;
}
- _buildSuggestions(excludedTargetIds: Set): {userId: string, user: RoomMember} {
+ _buildSuggestions(excludedTargetIds: Set): {userId: string, user: RoomMember}[] {
const maxConsideredMembers = 200;
const joinedRooms = MatrixClientPeg.get().getRooms()
.filter(r => r.getMyMembership() === 'join' && r.getJoinedMemberCount() <= maxConsideredMembers);
@@ -470,7 +488,7 @@ export default class InviteDialog extends React.PureComponent {
}, {});
// Generates { userId: {member, numRooms, score} }
- const memberScores = Object.values(memberRooms).reduce((scores, entry) => {
+ const memberScores = Object.values(memberRooms).reduce((scores, entry: {member: RoomMember, rooms: Room[]}) => {
const numMembersTotal = entry.rooms.reduce((c, r) => c + r.getJoinedMemberCount(), 0);
const maxRange = maxConsideredMembers * entry.rooms.length;
scores[entry.member.userId] = {
@@ -603,7 +621,7 @@ export default class InviteDialog extends React.PureComponent {
return;
}
- const createRoomOptions = {inlineErrors: true};
+ const createRoomOptions = {inlineErrors: true} as any; // XXX: Type out `createRoomOptions`
if (privateShouldBeEncrypted()) {
// Check whether all users have uploaded device keys before.
@@ -620,7 +638,7 @@ export default class InviteDialog extends React.PureComponent {
// Check if it's a traditional DM and create the room if required.
// TODO: [Canonical DMs] Remove this check and instead just create the multi-person DM
- let createRoomPromise = Promise.resolve();
+ let createRoomPromise = Promise.resolve(null) as Promise;
const isSelf = targetIds.length === 1 && targetIds[0] === MatrixClientPeg.get().getUserId();
if (targetIds.length === 1 && !isSelf) {
createRoomOptions.dmUserId = targetIds[0];
@@ -990,7 +1008,8 @@ export default class InviteDialog extends React.PureComponent {
const hasMixins = this.state.serverResultsMixin || this.state.threepidResultsMixin;
if (this.state.filterText && hasMixins && kind === 'suggestions') {
// We don't want to duplicate members though, so just exclude anyone we've already seen.
- const notAlreadyExists = (u: Member): boolean => {
+ // The type of u is a pain to define but members of both mixins have the 'userId' property
+ const notAlreadyExists = (u: any): boolean => {
return !sourceMembers.some(m => m.userId === u.userId)
&& !priorityAdditionalMembers.some(m => m.userId === u.userId)
&& !otherAdditionalMembers.some(m => m.userId === u.userId);
@@ -1169,7 +1188,8 @@ export default class InviteDialog extends React.PureComponent {
if (CommunityPrototypeStore.instance.getSelectedCommunityId()) {
const communityName = CommunityPrototypeStore.instance.getSelectedCommunityName();
- const inviteText = _t("This won't invite them to %(communityName)s. " +
+ const inviteText = _t(
+ "This won't invite them to %(communityName)s. " +
"To invite someone to %(communityName)s, click here",
{communityName}, {
userId: () => {
@@ -1209,7 +1229,9 @@ export default class InviteDialog extends React.PureComponent {
userId: () =>
{userId},
a: (sub) =>
- {sub},
+
+ {sub}
+ ,
},
);
} else {
@@ -1220,7 +1242,9 @@ export default class InviteDialog extends React.PureComponent {
userId: () =>
{userId},
a: (sub) =>
- {sub},
+
+ {sub}
+ ,
},
);
}
diff --git a/src/components/views/dialogs/RegistrationEmailPromptDialog.tsx b/src/components/views/dialogs/RegistrationEmailPromptDialog.tsx
new file mode 100644
index 0000000000..b7cc81c113
--- /dev/null
+++ b/src/components/views/dialogs/RegistrationEmailPromptDialog.tsx
@@ -0,0 +1,96 @@
+/*
+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 { _t } from '../../../languageHandler';
+import { IDialogProps } from "./IDialogProps";
+import {useRef, useState} from "react";
+import Field from "../elements/Field";
+import CountlyAnalytics from "../../../CountlyAnalytics";
+import withValidation from "../elements/Validation";
+import * as Email from "../../../email";
+import BaseDialog from "./BaseDialog";
+import DialogButtons from "../elements/DialogButtons";
+
+interface IProps extends IDialogProps {
+ onFinished(continued: boolean, email?: string): void;
+}
+
+const validation = withValidation({
+ rules: [
+ {
+ key: "email",
+ test: ({ value }) => !value || Email.looksValid(value),
+ invalid: () => _t("Doesn't look like a valid email address"),
+ },
+ ],
+});
+
+const RegistrationEmailPromptDialog: React.FC = ({onFinished}) => {
+ const [email, setEmail] = useState("");
+ const fieldRef = useRef();
+
+ const onSubmit = async () => {
+ if (email) {
+ const valid = await fieldRef.current.validate({ allowEmpty: false });
+
+ if (!valid) {
+ fieldRef.current.focus();
+ fieldRef.current.validate({ allowEmpty: false, focused: true });
+ return;
+ }
+ }
+
+ onFinished(true, email);
+ };
+
+ return onFinished(false)}
+ fixedWidth={false}
+ >
+
+
{_t("Just a heads up, if you don't add an email and forget your password, you could " +
+ "permanently lose access to your account.", {}, {
+ b: sub => {sub},
+ })}
+
+
+
+ ;
+};
+
+export default RegistrationEmailPromptDialog;
diff --git a/src/components/views/dialogs/RoomSettingsDialog.js b/src/components/views/dialogs/RoomSettingsDialog.js
index a43b284c42..9d9313f08f 100644
--- a/src/components/views/dialogs/RoomSettingsDialog.js
+++ b/src/components/views/dialogs/RoomSettingsDialog.js
@@ -53,9 +53,9 @@ export default class RoomSettingsDialog extends React.Component {
}
_onAction = (payload) => {
- // When room changes below us, close the room settings
+ // When view changes below us, close the room settings
// whilst the modal is open this can only be triggered when someone hits Leave Room
- if (payload.action === 'view_next_room') {
+ if (payload.action === 'view_home_page') {
this.props.onFinished();
}
};
diff --git a/src/components/views/dialogs/ServerPickerDialog.tsx b/src/components/views/dialogs/ServerPickerDialog.tsx
new file mode 100644
index 0000000000..65d53f0870
--- /dev/null
+++ b/src/components/views/dialogs/ServerPickerDialog.tsx
@@ -0,0 +1,235 @@
+/*
+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, {createRef} from "react";
+import {AutoDiscovery} from "matrix-js-sdk/src/autodiscovery";
+
+import AutoDiscoveryUtils, {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
+import BaseDialog from './BaseDialog';
+import { _t } from '../../../languageHandler';
+import AccessibleButton from "../elements/AccessibleButton";
+import SdkConfig from "../../../SdkConfig";
+import Field from "../elements/Field";
+import StyledRadioButton from "../elements/StyledRadioButton";
+import TextWithTooltip from "../elements/TextWithTooltip";
+import withValidation, {IFieldState} from "../elements/Validation";
+
+interface IProps {
+ title?: string;
+ serverConfig: ValidatedServerConfig;
+ onFinished(config?: ValidatedServerConfig): void;
+}
+
+interface IState {
+ defaultChosen: boolean;
+ otherHomeserver: string;
+}
+
+export default class ServerPickerDialog extends React.PureComponent {
+ private readonly defaultServer: ValidatedServerConfig;
+ private readonly fieldRef = createRef();
+ private validatedConf: ValidatedServerConfig;
+
+ constructor(props) {
+ super(props);
+
+ const config = SdkConfig.get();
+ this.defaultServer = config["validated_server_config"] as ValidatedServerConfig;
+ const { serverConfig } = this.props;
+
+ let otherHomeserver = "";
+ if (!serverConfig.isDefault) {
+ if (serverConfig.isNameResolvable && serverConfig.hsName) {
+ otherHomeserver = serverConfig.hsName;
+ } else {
+ otherHomeserver = serverConfig.hsUrl;
+ }
+ }
+
+ this.state = {
+ defaultChosen: serverConfig.isDefault,
+ otherHomeserver,
+ };
+ }
+
+ private onDefaultChosen = () => {
+ this.setState({ defaultChosen: true });
+ };
+
+ private onOtherChosen = () => {
+ this.setState({ defaultChosen: false });
+ };
+
+ private onHomeserverChange = (ev) => {
+ this.setState({ otherHomeserver: ev.target.value });
+ };
+
+ // TODO: Do we want to support .well-known lookups here?
+ // If for some reason someone enters "matrix.org" for a URL, we could do a lookup to
+ // find their homeserver without demanding they use "https://matrix.org"
+ private validate = withValidation({
+ deriveData: async ({ value }) => {
+ let hsUrl = value.trim(); // trim to account for random whitespace
+
+ // if the URL has no protocol, try validate it as a serverName via well-known
+ if (!hsUrl.includes("://")) {
+ try {
+ const discoveryResult = await AutoDiscovery.findClientConfig(hsUrl);
+ this.validatedConf = AutoDiscoveryUtils.buildValidatedConfigFromDiscovery(hsUrl, discoveryResult);
+ return {}; // we have a validated config, we don't need to try the other paths
+ } catch (e) {
+ console.error(`Attempted ${hsUrl} as a server_name but it failed`, e);
+ }
+ }
+
+ // if we got to this stage then either the well-known failed or the URL had a protocol specified,
+ // so validate statically only. If the URL has no protocol, default to https.
+ if (!hsUrl.includes("://")) {
+ hsUrl = "https://" + hsUrl;
+ }
+
+ try {
+ this.validatedConf = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl);
+ return {};
+ } catch (e) {
+ console.error(e);
+
+ const stateForError = AutoDiscoveryUtils.authComponentStateForError(e);
+ if (stateForError.isFatalError) {
+ let error = _t("Unable to validate homeserver");
+ if (e.translatedMessage) {
+ error = e.translatedMessage;
+ }
+ return { error };
+ }
+
+ // try to carry on anyway
+ try {
+ this.validatedConf = await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(hsUrl, null, true);
+ return {};
+ } catch (e) {
+ console.error(e);
+ return { error: _t("Invalid URL") };
+ }
+ }
+ },
+ rules: [
+ {
+ key: "required",
+ test: ({ value, allowEmpty }) => allowEmpty || !!value,
+ invalid: () => _t("Specify a homeserver"),
+ }, {
+ key: "valid",
+ test: async function({ value }, { error }) {
+ if (!value) return true;
+ return !error;
+ },
+ invalid: function({ error }) {
+ return error;
+ },
+ },
+ ],
+ });
+
+ private onHomeserverValidate = (fieldState: IFieldState) => this.validate(fieldState);
+
+ private onSubmit = async (ev) => {
+ ev.preventDefault();
+
+ const valid = await this.fieldRef.current.validate({ allowEmpty: false });
+
+ if (!valid) {
+ this.fieldRef.current.focus();
+ this.fieldRef.current.validate({ allowEmpty: false, focused: true });
+ return;
+ }
+
+ this.props.onFinished(this.validatedConf);
+ };
+
+ public render() {
+ let text;
+ if (this.defaultServer.hsName === "matrix.org") {
+ text = _t("Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.");
+ }
+
+ let defaultServerName = this.defaultServer.hsName;
+ if (this.defaultServer.hsNameIsDifferent) {
+ defaultServerName = (
+
+ {this.defaultServer.hsName}
+
+ );
+ }
+
+ return
+
+ ;
+ }
+}
diff --git a/src/components/views/dialogs/ShareDialog.tsx b/src/components/views/dialogs/ShareDialog.tsx
index 1569977d58..5264031cc6 100644
--- a/src/components/views/dialogs/ShareDialog.tsx
+++ b/src/components/views/dialogs/ShareDialog.tsx
@@ -146,7 +146,7 @@ export default class ShareDialog extends React.PureComponent {
const events = this.props.target.getLiveTimeline().getEvents();
matrixToUrl = this.state.permalinkCreator.forEvent(events[events.length - 1].getId());
} else {
- matrixToUrl = this.state.permalinkCreator.forRoom();
+ matrixToUrl = this.state.permalinkCreator.forShareableRoom();
}
} else if (this.props.target instanceof User || this.props.target instanceof RoomMember) {
matrixToUrl = makeUserPermalink(this.props.target.userId);
diff --git a/src/components/views/elements/AppTile.js b/src/components/views/elements/AppTile.js
index b862a1e912..7e0ae965bb 100644
--- a/src/components/views/elements/AppTile.js
+++ b/src/components/views/elements/AppTile.js
@@ -375,17 +375,20 @@ export default class AppTile extends React.Component {
);
- // all widgets can theoretically be allowed to remain on screen, so we wrap
- // them all in a PersistedElement from the get-go. If we wait, the iframe will
- // be re-mounted later, which means the widget has to start over, which is bad.
+ if (!this.props.userWidget) {
+ // All room widgets can theoretically be allowed to remain on screen, so we
+ // wrap them all in a PersistedElement from the get-go. If we wait, the iframe
+ // will be re-mounted later, which means the widget has to start over, which is
+ // bad.
- // Also wrap the PersistedElement in a div to fix the height, otherwise
- // AppTile's border is in the wrong place
- appTileBody =
-
- {appTileBody}
-
-
;
+ // Also wrap the PersistedElement in a div to fix the height, otherwise
+ // AppTile's border is in the wrong place
+ appTileBody =
+
+ {appTileBody}
+
+
;
+ }
}
}
diff --git a/src/components/views/elements/EffectsOverlay.tsx b/src/components/views/elements/EffectsOverlay.tsx
new file mode 100644
index 0000000000..38be8da9a8
--- /dev/null
+++ b/src/components/views/elements/EffectsOverlay.tsx
@@ -0,0 +1,94 @@
+/*
+ Copyright 2020 Nurjin Jafar
+ Copyright 2020 Nordeck IT + Consulting GmbH.
+
+ 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, { FunctionComponent, useEffect, useRef } from 'react';
+import dis from '../../../dispatcher/dispatcher';
+import ICanvasEffect from '../../../effects/ICanvasEffect';
+import {CHAT_EFFECTS} from '../../../effects'
+
+interface IProps {
+ roomWidth: number;
+}
+
+const EffectsOverlay: FunctionComponent = ({ roomWidth }) => {
+ const canvasRef = useRef(null);
+ const effectsRef = useRef
;
- const memberCount = useMemberCount(room);
+ const memberCount = useRoomMemberCount(room);
return
diff --git a/src/components/views/right_panel/WidgetCard.tsx b/src/components/views/right_panel/WidgetCard.tsx
index c1753e90e3..593bd0dde7 100644
--- a/src/components/views/right_panel/WidgetCard.tsx
+++ b/src/components/views/right_panel/WidgetCard.tsx
@@ -42,7 +42,7 @@ const WidgetCard: React.FC = ({ room, widgetId, onClose }) => {
const apps = useWidgets(room);
const app = apps.find(a => a.id === widgetId);
- const isPinned = app && WidgetStore.instance.isPinned(app.id);
+ const isPinned = app && WidgetStore.instance.isPinned(room.roomId, app.id);
const [menuDisplayed, handle, openMenu, closeMenu] = useContextMenu();
diff --git a/src/components/views/rooms/AuxPanel.tsx b/src/components/views/rooms/AuxPanel.tsx
index 465c9c749a..7966643084 100644
--- a/src/components/views/rooms/AuxPanel.tsx
+++ b/src/components/views/rooms/AuxPanel.tsx
@@ -26,9 +26,9 @@ import classNames from 'classnames';
import RateLimitedFunc from '../../../ratelimitedfunc';
import SettingsStore from "../../../settings/SettingsStore";
import AutoHideScrollbar from "../../structures/AutoHideScrollbar";
-import CallView from "../voip/CallView";
import {UIFeature} from "../../../settings/UIFeature";
import { ResizeNotifier } from "../../../utils/ResizeNotifier";
+import CallViewForRoom from '../voip/CallViewForRoom';
interface IProps {
// js-sdk room object
@@ -166,8 +166,8 @@ export default class AuxPanel extends React.Component {
}
const callView = (
-
diff --git a/src/components/views/rooms/BasicMessageComposer.tsx b/src/components/views/rooms/BasicMessageComposer.tsx
index 311a4734fd..2ececdeaed 100644
--- a/src/components/views/rooms/BasicMessageComposer.tsx
+++ b/src/components/views/rooms/BasicMessageComposer.tsx
@@ -47,6 +47,7 @@ import AutocompleteWrapperModel from "../../../editor/autocomplete";
import DocumentPosition from "../../../editor/position";
import {ICompletion} from "../../../autocomplete/Autocompleter";
+// matches emoticons which follow the start of a line or whitespace
const REGEX_EMOTICON_WHITESPACE = new RegExp('(?:^|\\s)(' + EMOTICON_REGEX.source + ')\\s$');
const IS_MAC = navigator.platform.indexOf("Mac") !== -1;
@@ -524,7 +525,7 @@ export default class BasicMessageEditor extends React.Component
const position = model.positionForOffset(caret.offset, caret.atNodeEnd);
const range = model.startRange(position);
range.expandBackwardsWhile((index, offset, part) => {
- return part.text[offset] !== " " && (
+ return part.text[offset] !== " " && part.text[offset] !== "+" && (
part.type === "plain" ||
part.type === "pill-candidate" ||
part.type === "command"
diff --git a/src/components/views/rooms/EventTile.js b/src/components/views/rooms/EventTile.js
index c358ef610d..11277daa57 100644
--- a/src/components/views/rooms/EventTile.js
+++ b/src/components/views/rooms/EventTile.js
@@ -745,13 +745,22 @@ export default class EventTile extends React.Component {
}
if (this.props.mxEvent.sender && avatarSize) {
+ let member;
+ // set member to receiver (target) if it is a 3PID invite
+ // so that the correct avatar is shown as the text is
+ // `$target accepted the invitation for $email`
+ if (this.props.mxEvent.getContent().third_party_invite) {
+ member = this.props.mxEvent.target;
+ } else {
+ member = this.props.mxEvent.sender;
+ }
avatar = (
-
;
+ if (this.state.showMoreMenu) {
+ contextMenu = ;
}
+ const micClasses = classNames({
+ mx_CallView_callControls_button: true,
+ mx_CallView_callControls_button_micOn: !this.state.micMuted,
+ mx_CallView_callControls_button_micOff: this.state.micMuted,
+ });
+
+ const vidClasses = classNames({
+ mx_CallView_callControls_button: true,
+ mx_CallView_callControls_button_vidOn: !this.state.vidMuted,
+ mx_CallView_callControls_button_vidOff: this.state.vidMuted,
+ });
+
+ // Put the other states of the mic/video icons in the document to make sure they're cached
+ // (otherwise the icon disappears briefly when toggled)
+ const micCacheClasses = classNames({
+ mx_CallView_callControls_button: true,
+ mx_CallView_callControls_button_micOn: this.state.micMuted,
+ mx_CallView_callControls_button_micOff: !this.state.micMuted,
+ mx_CallView_callControls_button_invisible: true,
+ });
+
+ const vidCacheClasses = classNames({
+ mx_CallView_callControls_button: true,
+ mx_CallView_callControls_button_vidOn: this.state.micMuted,
+ mx_CallView_callControls_button_vidOff: !this.state.micMuted,
+ mx_CallView_callControls_button_invisible: true,
+ });
+
+ const callControlsClasses = classNames({
+ mx_CallView_callControls: true,
+ mx_CallView_callControls_hidden: !this.state.controlsVisible,
+ });
+
+ const vidMuteButton = this.props.call.type === CallType.Video ? : null;
+
+ // The 'more' button actions are only relevant in a connected call
+ // When not connected, we have to put something there to make the flexbox alignment correct
+ const contextMenuButton = this.state.callState === CallState.Connected ? : ;
+
+ // in the near future, the dial pad button will go on the left. For now, it's the nothing button
+ // because something needs to have margin-right: auto to make the alignment correct.
+ const callControls =
;
+
// The 'content' for the call, ie. the videos for a video call and profile picture
// for voice calls (fills the bg)
let contentView: React.ReactNode;
- if (this.state.call.type === CallType.Video) {
+ const isOnHold = this.state.isLocalOnHold || this.state.isRemoteOnHold;
+ let onHoldText = null;
+ if (this.state.isRemoteOnHold) {
+ onHoldText = _t("You held the call Resume", {}, {
+ a: sub =>
+ {sub}
+ ,
+ });
+ } else if (this.state.isLocalOnHold) {
+ onHoldText = _t("%(peerName)s held the call", {
+ peerName: this.props.call.getOpponentMember().name,
+ });
+ }
+
+ if (this.props.call.type === CallType.Video) {
+ let onHoldContent = null;
+ let onHoldBackground = null;
+ const backgroundStyle: CSSProperties = {};
+ const containerClasses = classNames({
+ mx_CallView_video: true,
+ mx_CallView_video_hold: isOnHold,
+ });
+ if (isOnHold) {
+ onHoldContent =
+ {onHoldText}
+
;
+ const backgroundAvatarUrl = avatarUrlForMember(
+ // is it worth getting the size of the div to pass here?
+ this.props.call.getOpponentMember(), 1024, 1024, 'crop',
+ );
+ backgroundStyle.backgroundImage = 'url(' + backgroundAvatarUrl + ')';
+ onHoldBackground = ;
+ }
+
// if we're fullscreen, we don't want to set a maxHeight on the video element.
- const maxVideoHeight = getFullScreenElement() ? null : this.props.maxVideoHeight - HEADER_HEIGHT;
- contentView =
;
}
}
diff --git a/src/components/views/voip/CallViewForRoom.tsx b/src/components/views/voip/CallViewForRoom.tsx
new file mode 100644
index 0000000000..4cb4e66fbe
--- /dev/null
+++ b/src/components/views/voip/CallViewForRoom.tsx
@@ -0,0 +1,87 @@
+/*
+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 { CallState, MatrixCall } from 'matrix-js-sdk/src/webrtc/call';
+import React from 'react';
+import CallHandler from '../../../CallHandler';
+import CallView from './CallView';
+import dis from '../../../dispatcher/dispatcher';
+
+interface IProps {
+ // What room we should display the call for
+ roomId: string,
+
+ // maxHeight style attribute for the video panel
+ maxVideoHeight?: number;
+
+ // a callback which is called when the content in the callview changes
+ // in a way that is likely to cause a resize.
+ onResize?: any;
+}
+
+interface IState {
+ call: MatrixCall,
+}
+
+/*
+ * Wrapper for CallView that always display the call in a given room,
+ * or nothing if there is no call in that room.
+ */
+export default class CallViewForRoom extends React.Component {
+ private dispatcherRef: string;
+
+ constructor(props: IProps) {
+ super(props);
+ this.state = {
+ call: this.getCall(),
+ };
+ }
+
+ public componentDidMount() {
+ this.dispatcherRef = dis.register(this.onAction);
+ }
+
+ public componentWillUnmount() {
+ dis.unregister(this.dispatcherRef);
+ }
+
+ private onAction = (payload) => {
+ switch (payload.action) {
+ case 'call_state': {
+ const newCall = this.getCall();
+ if (newCall !== this.state.call) {
+ this.setState({call: newCall});
+ }
+ break;
+ }
+ }
+ };
+
+ private getCall(): MatrixCall {
+ const call = CallHandler.sharedInstance().getCallForRoom(this.props.roomId);
+
+ if (call && [CallState.Ended, CallState.Ringing].includes(call.state)) return null;
+ return call;
+ }
+
+ public render() {
+ if (!this.state.call) return null;
+
+ return ;
+ }
+}
diff --git a/src/components/views/voip/VideoFeed.tsx b/src/components/views/voip/VideoFeed.tsx
index 5fb71a6d69..5210f28eb1 100644
--- a/src/components/views/voip/VideoFeed.tsx
+++ b/src/components/views/voip/VideoFeed.tsx
@@ -42,10 +42,12 @@ export default class VideoFeed extends React.Component {
componentDidMount() {
this.vid.current.addEventListener('resize', this.onResize);
- if (this.props.type === VideoFeedType.Local) {
- this.props.call.setLocalVideoElement(this.vid.current);
- } else {
- this.props.call.setRemoteVideoElement(this.vid.current);
+ this.setVideoElement();
+ }
+
+ componentDidUpdate(prevProps) {
+ if (this.props.call !== prevProps.call) {
+ this.setVideoElement();
}
}
@@ -53,6 +55,14 @@ export default class VideoFeed extends React.Component {
this.vid.current.removeEventListener('resize', this.onResize);
}
+ private setVideoElement() {
+ if (this.props.type === VideoFeedType.Local) {
+ this.props.call.setLocalVideoElement(this.vid.current);
+ } else {
+ this.props.call.setRemoteVideoElement(this.vid.current);
+ }
+ }
+
onResize = (e) => {
if (this.props.onResize) {
this.props.onResize(e);
diff --git a/src/customisations/RoomList.ts b/src/customisations/RoomList.ts
new file mode 100644
index 0000000000..758b212aa2
--- /dev/null
+++ b/src/customisations/RoomList.ts
@@ -0,0 +1,45 @@
+/*
+ * 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 { Room } from "matrix-js-sdk/src/models/room";
+
+// Populate this file with the details of your customisations when copying it.
+
+/**
+ * Determines if a room is visible in the room list or not. By default,
+ * all rooms are visible. Where special handling is performed by Element,
+ * those rooms will not be able to override their visibility in the room
+ * list - Element will make the decision without calling this function.
+ *
+ * This function should be as fast as possible to avoid slowing down the
+ * client.
+ * @param {Room} room The room to check the visibility of.
+ * @returns {boolean} True if the room should be visible, false otherwise.
+ */
+function isRoomVisible(room: Room): boolean {
+ return true;
+}
+
+// This interface summarises all available customisation points and also marks
+// them all as optional. This allows customisers to only define and export the
+// customisations they need while still maintaining type safety.
+export interface IRoomListCustomisations {
+ isRoomVisible?: typeof isRoomVisible;
+}
+
+// A real customisation module will define and export one or more of the
+// customisation points that make up the interface above.
+export const RoomListCustomisations: IRoomListCustomisations = {};
diff --git a/src/editor/parts.ts b/src/editor/parts.ts
index 5ed0c0529f..8a7ccfcb7b 100644
--- a/src/editor/parts.ts
+++ b/src/editor/parts.ts
@@ -190,7 +190,9 @@ abstract class PlainBasePart extends BasePart {
return true;
}
// only split if the previous character is a space
- return this._text[offset - 1] !== " ";
+ // or if it is a + and this is a :
+ return this._text[offset - 1] !== " " &&
+ (this._text[offset - 1] !== "+" || chr !== ":");
}
return true;
}
diff --git a/src/effects/ICanvasEffect.ts b/src/effects/ICanvasEffect.ts
new file mode 100644
index 0000000000..9bf3e9293d
--- /dev/null
+++ b/src/effects/ICanvasEffect.ts
@@ -0,0 +1,47 @@
+/*
+ Copyright 2020 Nurjin Jafar
+ Copyright 2020 Nordeck IT + Consulting GmbH.
+
+ 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.
+ */
+/**
+ * Defines the constructor of a canvas based room effect
+ */
+export interface ICanvasEffectConstructable {
+ /**
+ * @param {{[key:string]:any}} options? Optional animation options
+ * @returns ICanvasEffect Returns a new instance of the canvas effect
+ */
+ new(options?: { [key: string]: any }): ICanvasEffect;
+}
+
+/**
+ * Defines the interface of a canvas based room effect
+ */
+export default interface ICanvasEffect {
+ /**
+ * @param {HTMLCanvasElement} canvas The canvas instance as the render target of the animation
+ * @param {number} timeout? A timeout that defines the runtime of the animation (defaults to false)
+ */
+ start: (canvas: HTMLCanvasElement, timeout?: number) => Promise;
+
+ /**
+ * Stops the current animation
+ */
+ stop: () => Promise;
+
+ /**
+ * Returns a value that defines if the animation is currently running
+ */
+ isRunning: boolean;
+}
diff --git a/src/effects/confetti/index.ts b/src/effects/confetti/index.ts
new file mode 100644
index 0000000000..8c4b2d2616
--- /dev/null
+++ b/src/effects/confetti/index.ts
@@ -0,0 +1,191 @@
+/*
+ Copyright 2020 Nurjin Jafar
+ Copyright 2020 Nordeck IT + Consulting GmbH.
+
+ 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 ICanvasEffect from '../ICanvasEffect';
+
+export type ConfettiOptions = {
+ /**
+ * max confetti count
+ */
+ maxCount: number,
+ /**
+ * particle animation speed
+ */
+ speed: number,
+ /**
+ * the confetti animation frame interval in milliseconds
+ */
+ frameInterval: number,
+ /**
+ * the alpha opacity of the confetti (between 0 and 1, where 1 is opaque and 0 is invisible)
+ */
+ alpha: number,
+ /**
+ * use gradient instead of solid particle color
+ */
+ gradient: boolean,
+}
+
+type ConfettiParticle = {
+ color: string,
+ color2: string,
+ x: number,
+ y: number,
+ diameter: number,
+ tilt: number,
+ tiltAngleIncrement: number,
+ tiltAngle: number,
+}
+
+export const DefaultOptions: ConfettiOptions = {
+ maxCount: 150,
+ speed: 3,
+ frameInterval: 15,
+ alpha: 1.0,
+ gradient: false,
+};
+
+export default class Confetti implements ICanvasEffect {
+ private readonly options: ConfettiOptions;
+
+ constructor(options: { [key: string]: any }) {
+ this.options = {...DefaultOptions, ...options};
+ }
+
+ private context: CanvasRenderingContext2D | null = null;
+ private supportsAnimationFrame = window.requestAnimationFrame;
+ private colors = ['rgba(30,144,255,', 'rgba(107,142,35,', 'rgba(255,215,0,',
+ 'rgba(255,192,203,', 'rgba(106,90,205,', 'rgba(173,216,230,',
+ 'rgba(238,130,238,', 'rgba(152,251,152,', 'rgba(70,130,180,',
+ 'rgba(244,164,96,', 'rgba(210,105,30,', 'rgba(220,20,60,'];
+
+ private lastFrameTime = Date.now();
+ private particles: Array = [];
+ private waveAngle = 0;
+
+ public isRunning: boolean;
+
+ public start = async (canvas: HTMLCanvasElement, timeout = 3000) => {
+ if (!canvas) {
+ return;
+ }
+ this.context = canvas.getContext('2d');
+ this.particles = [];
+ const count = this.options.maxCount;
+ while (this.particles.length < count) {
+ this.particles.push(this.resetParticle({} as ConfettiParticle, canvas.width, canvas.height));
+ }
+ this.isRunning = true;
+ this.runAnimation();
+ if (timeout) {
+ window.setTimeout(this.stop, timeout);
+ }
+ }
+
+ public stop = async () => {
+ this.isRunning = false;
+ }
+
+ private resetParticle = (particle: ConfettiParticle, width: number, height: number): ConfettiParticle => {
+ particle.color = this.colors[(Math.random() * this.colors.length) | 0] + (this.options.alpha + ')');
+ if (this.options.gradient) {
+ particle.color2 = this.colors[(Math.random() * this.colors.length) | 0] + (this.options.alpha + ')');
+ } else {
+ particle.color2 = particle.color;
+ }
+ particle.x = Math.random() * width;
+ particle.y = Math.random() * -height;
+ particle.diameter = Math.random() * 10 + 5;
+ particle.tilt = Math.random() * -10;
+ particle.tiltAngleIncrement = Math.random() * 0.07 + 0.05;
+ particle.tiltAngle = Math.random() * Math.PI;
+ return particle;
+ }
+
+ private runAnimation = (): void => {
+ if (!this.context || !this.context.canvas) {
+ return;
+ }
+ if (this.particles.length === 0) {
+ this.context.clearRect(0, 0, this.context.canvas.width, this.context.canvas.height);
+ } else {
+ const now = Date.now();
+ const delta = now - this.lastFrameTime;
+ if (!this.supportsAnimationFrame || delta > this.options.frameInterval) {
+ this.context.clearRect(0, 0, this.context.canvas.width, this.context.canvas.height);
+ this.updateParticles();
+ this.drawParticles(this.context);
+ this.lastFrameTime = now - (delta % this.options.frameInterval);
+ }
+ requestAnimationFrame(this.runAnimation);
+ }
+ }
+
+
+ private drawParticles = (context: CanvasRenderingContext2D): void => {
+ if (!this.context || !this.context.canvas) {
+ return;
+ }
+ let x; let x2; let y2;
+ for (const particle of this.particles) {
+ this.context.beginPath();
+ context.lineWidth = particle.diameter;
+ x2 = particle.x + particle.tilt;
+ x = x2 + particle.diameter / 2;
+ y2 = particle.y + particle.tilt + particle.diameter / 2;
+ if (this.options.gradient) {
+ const gradient = context.createLinearGradient(x, particle.y, x2, y2);
+ gradient.addColorStop(0, particle.color);
+ gradient.addColorStop(1.0, particle.color2);
+ context.strokeStyle = gradient;
+ } else {
+ context.strokeStyle = particle.color;
+ }
+ context.moveTo(x, particle.y);
+ context.lineTo(x2, y2);
+ context.stroke();
+ }
+ }
+
+ private updateParticles = () => {
+ if (!this.context || !this.context.canvas) {
+ return;
+ }
+ const width = this.context.canvas.width;
+ const height = this.context.canvas.height;
+ let particle: ConfettiParticle;
+ this.waveAngle += 0.01;
+ for (let i = 0; i < this.particles.length; i++) {
+ particle = this.particles[i];
+ if (!this.isRunning && particle.y < -15) {
+ particle.y = height + 100;
+ } else {
+ particle.tiltAngle += particle.tiltAngleIncrement;
+ particle.x += Math.sin(this.waveAngle) - 0.5;
+ particle.y += (Math.cos(this.waveAngle) + particle.diameter + this.options.speed) * 0.5;
+ particle.tilt = Math.sin(particle.tiltAngle) * 15;
+ }
+ if (particle.x > width + 20 || particle.x < -20 || particle.y > height) {
+ if (this.isRunning && this.particles.length <= this.options.maxCount) {
+ this.resetParticle(particle, width, height);
+ } else {
+ this.particles.splice(i, 1);
+ i--;
+ }
+ }
+ }
+ }
+}
diff --git a/src/effects/index.ts b/src/effects/index.ts
new file mode 100644
index 0000000000..16a0851070
--- /dev/null
+++ b/src/effects/index.ts
@@ -0,0 +1,89 @@
+/*
+ Copyright 2020 Nurjin Jafar
+ Copyright 2020 Nordeck IT + Consulting GmbH.
+
+ 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 { _t, _td } from "../languageHandler";
+
+export type Effect = {
+ /**
+ * one or more emojis that will trigger this effect
+ */
+ emojis: Array;
+ /**
+ * the matrix message type that will trigger this effect
+ */
+ msgType: string;
+ /**
+ * the room command to trigger this effect
+ */
+ command: string;
+ /**
+ * a function that returns the translated description of the effect
+ */
+ description: () => string;
+ /**
+ * a function that returns the translated fallback message. this message will be shown if the user did not provide a custom message
+ */
+ fallbackMessage: () => string;
+ /**
+ * animation options
+ */
+ options: TOptions;
+}
+
+type ConfettiOptions = {
+ /**
+ * max confetti count
+ */
+ maxCount: number,
+ /**
+ * particle animation speed
+ */
+ speed: number,
+ /**
+ * the confetti animation frame interval in milliseconds
+ */
+ frameInterval: number,
+ /**
+ * the alpha opacity of the confetti (between 0 and 1, where 1 is opaque and 0 is invisible)
+ */
+ alpha: number,
+ /**
+ * use gradient instead of solid particle color
+ */
+ gradient: boolean,
+}
+
+/**
+ * This configuration defines room effects that can be triggered by custom message types and emojis
+ */
+export const CHAT_EFFECTS: Array> = [
+ {
+ emojis: ['🎊', '🎉'],
+ msgType: 'nic.custom.confetti',
+ command: 'confetti',
+ description: () => _td("Sends the given message with confetti"),
+ fallbackMessage: () => _t("sends confetti") + " 🎉",
+ options: {
+ maxCount: 150,
+ speed: 3,
+ frameInterval: 15,
+ alpha: 1.0,
+ gradient: false,
+ },
+ } as Effect,
+];
+
+
diff --git a/src/effects/utils.ts b/src/effects/utils.ts
new file mode 100644
index 0000000000..c2b499b154
--- /dev/null
+++ b/src/effects/utils.ts
@@ -0,0 +1,24 @@
+/*
+ Copyright 2020 Nurjin Jafar
+ Copyright 2020 Nordeck IT + Consulting GmbH.
+
+ 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.
+ */
+/**
+ * Checks a message if it contains one of the provided emojis
+ * @param {Object} content The message
+ * @param {Array} emojis The list of emojis to check for
+ */
+export const containsEmoji = (content: { msgtype: string, body: string }, emojis: Array): boolean => {
+ return emojis.some((emoji) => content.body && content.body.includes(emoji));
+}
diff --git a/src/hooks/useRoomMembers.ts b/src/hooks/useRoomMembers.ts
new file mode 100644
index 0000000000..e25436045e
--- /dev/null
+++ b/src/hooks/useRoomMembers.ts
@@ -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 {useState} from "react";
+import {Room} from "matrix-js-sdk/src/models/room";
+import {RoomMember} from "matrix-js-sdk/src/models/room-member";
+
+import {useEventEmitter} from "./useEventEmitter";
+import {throttle} from "lodash";
+
+// Hook to simplify watching Matrix Room joined members
+export const useRoomMembers = (room: Room, throttleWait = 250) => {
+ const [members, setMembers] = useState(room.getJoinedMembers());
+ useEventEmitter(room.currentState, "RoomState.members", throttle(() => {
+ setMembers(room.getJoinedMembers());
+ }, throttleWait, {leading: true, trailing: true}));
+ return members;
+};
+
+// Hook to simplify watching Matrix Room joined member count
+export const useRoomMemberCount = (room: Room, throttleWait = 250) => {
+ const [count, setCount] = useState(room.getJoinedMemberCount());
+ useEventEmitter(room.currentState, "RoomState.members", throttle(() => {
+ setCount(room.getJoinedMemberCount());
+ }, throttleWait, {leading: true, trailing: true}));
+ return count;
+};
diff --git a/src/i18n/strings/ar.json b/src/i18n/strings/ar.json
index fec025da8d..5da09afd23 100644
--- a/src/i18n/strings/ar.json
+++ b/src/i18n/strings/ar.json
@@ -1422,5 +1422,71 @@
"This will end the conference for everyone. Continue?": "هذا سينهي المؤتمر للجميع. استمر؟",
"The call was answered on another device.": "تم الرد على المكالمة على جهاز آخر.",
"The call could not be established": "تعذر إجراء المكالمة",
- "Call Declined": "رُفض الاتصال"
+ "Call Declined": "رُفض الاتصال",
+ "See videos posted to your active room": "أظهر الفيديوهات المرسلة إلى هذه غرفتك النشطة",
+ "See videos posted to this room": "أظهر الفيديوهات المرسلة إلى هذه الغرفة",
+ "Send videos as you in your active room": "أرسل الفيديوهات بهويتك في غرفتك النشطة",
+ "Send videos as you in this room": "أرسل الفيديوهات بهويتك في هذه الغرفة",
+ "See messages posted to this room": "أرسل الرسائل المرسلة إلى هذه الغرفة",
+ "See images posted to your active room": "أظهر الصور المرسلة إلى غرفتك النشطة",
+ "See images posted to this room": "أظهر الصور المرسلة إلى هذه الغرفة",
+ "Send images as you in your active room": "أرسل الصور بهويتك في غرفتك النشطة",
+ "Send images as you in this room": "أرسل الصور بهويتك في هذه الغرفة",
+ "See emotes posted to your active room": "أظهر الرموز التعبيرية المرسلة لغرفتك النشطة",
+ "See emotes posted to this room": "أظهر الرموز التعبيرية المرسلة إلى هذه الغرفة",
+ "Send emotes as you in your active room": "أظهر الرموز التعبيرية بهويتك في غرفتك النشطة",
+ "Send emotes as you in this room": "أرسل الرموز التعبيرية بهويتك",
+ "See text messages posted to your active room": "أظهر الرسائل النصية المرسلة إلى غرفتك النشطة",
+ "See text messages posted to this room": "أظهر الرسائل النصية المرسلة إلى هذه الغرفة",
+ "Send text messages as you in your active room": "أرسل الرسائل النصية بهويتك في غرفتك النشطة",
+ "Send text messages as you in this room": "أرسل الرسائل النصية بهويتك في هذه الغرفة",
+ "See messages posted to your active room": "أظهر الرسائل المرسلة إلى غرفتك النشطة",
+ "Send messages as you in your active room": "أرسل رسائل بهويتك في غرفتك النشطة",
+ "Send messages as you in this room": "أرسل رسائل بهويتك في هذه الغرفة",
+ "The %(capability)s capability": "القدرة %(capability)s",
+ "See %(eventType)s events posted to your active room": "أظهر أحداث %(eventType)s المنشورة بغرفة النشطة",
+ "Send %(eventType)s events as you in your active room": "أرسل أحداث %(eventType)s بهويتك في غرفتك النشطة",
+ "See %(eventType)s events posted to this room": "أظهر أحداث %(eventType)s المنشورة في هذه الغرفة",
+ "Send %(eventType)s events as you in this room": "أرسل أحداث %(eventType)s بهويتك في هذه الغرفة",
+ "with state key %(stateKey)s": "مع مفتاح الحالة %(stateKey)s",
+ "with an empty state key": "بمفتاح حالة فارغ",
+ "See when anyone posts a sticker to your active room": "أظهر وضع أي أحد للملصقات لغرفتك النشطة",
+ "Send stickers to your active room as you": "أرسل ملصقات لغرفتك النشطة بهويتك",
+ "See when a sticker is posted in this room": "أظهر وضع الملصقات في هذه الغرفة",
+ "Send stickers to this room as you": "أرسل ملصقات لهذه الغرفة بهويتك",
+ "See when the avatar changes in your active room": "أظهر تغييرات صورة غرفتك النشطة",
+ "Change the avatar of your active room": "غير صورة غرفتك النشطة",
+ "See when the avatar changes in this room": "أظهر تغييرات الصورة في هذه الغرفة",
+ "Change the avatar of this room": "غير صورة هذه الغرفة",
+ "See when the name changes in your active room": "أظهر تغييرات الاسم في غرفتك النشطة",
+ "Change the name of your active room": "غير اسم غرفتك النشطة",
+ "See when the name changes in this room": "أظهر تغييرات الاسم في هذه الغرفة",
+ "Change the name of this room": "غير اسم هذه الغرفة",
+ "See when the topic changes in your active room": "أظهر تغيير موضوع غرفتك النشطة",
+ "Change the topic of your active room": "غير موضوع غرفتك النشطة",
+ "See when the topic changes in this room": "أظهر تغير موضوع هذه الغرفة",
+ "Change the topic of this room": "تغيير موضوع هذه الغرفة",
+ "Change which room you're viewing": "تغيير الغرفة التي تشاهدها",
+ "Send stickers into your active room": "أرسل ملصقات إلى غرفتك النشطة",
+ "Send stickers into this room": "أرسل ملصقات إلى هذه الغرفة",
+ "Remain on your screen while running": "ابقَ على شاشتك أثناء إجراء",
+ "Remain on your screen when viewing another room, when running": "ابقَ على شاشتك عند مشاهدة غرفة أخرى أثناء إجراء",
+ "%(senderName)s created a rule banning users matching %(glob)s for %(reason)s": "%(senderName)s حدَّث قاعدة حظر المستخدمين المطابقة %(glob)s بسبب %(reason)s",
+ "%(senderName)s updated the rule banning servers matching %(glob)s for %(reason)s": "%(senderName)s حدَّث قاعدة حظر الخوادم المطابقة %(glob)s بسبب %(reason)s",
+ "%(senderName)s updated the rule banning rooms matching %(glob)s for %(reason)s": "%(senderName)s حدَّث قاعدة حظر الغرفة المطابقة %(glob)s بسبب %(reason)s",
+ "%(senderName)s declined the call.": "%(senderName)s رفض المكالمة.",
+ "(an error occurred)": "(حدث خطأ)",
+ "(their device couldn't start the camera / microphone)": "(تعذر على جهازهم بدء تشغيل الكاميرا / الميكروفون)",
+ "(connection failed)": "(فشل الاتصال)",
+ "🎉 All servers are banned from participating! This room can no longer be used.": "🎉 جميع الخوادم ممنوعة من المشاركة! لم يعد من الممكن استخدام هذه الغرفة.",
+ "Takes the call in the current room off hold": "يوقف المكالمة في الغرفة الحالية",
+ "Places the call in the current room on hold": "يضع المكالمة في الغرفة الحالية قيد الانتظار",
+ "Prepends ( ͡° ͜ʖ ͡°) to a plain-text message": "يلصق (͡ ° ͜ʖ ͡ °) أوَّل رسالة نصية عادية",
+ "No other application is using the webcam": "لا يوجد تطبيق آخر يستخدم كاميرا الويب",
+ "Permission is granted to use the webcam": "منح الإذن باستخدام كاميرا الويب",
+ "A microphone and webcam are plugged in and set up correctly": "الميكروفون وكاميرا ويب موصولان ومعدان بشكل صحيح",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "فشلت المكالمة نظرًا لتعذر الوصول إلى كاميرا الويب أو الميكروفون. تحقق مما يلي:",
+ "Unable to access webcam / microphone": "تعذر الوصول إلى كاميرا الويب / الميكروفون",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "فشلت المكالمة لأنه لا يمكن الوصول إلى ميكروفون. تحقق من توصيل الميكروفون وإعداده بشكل صحيح.",
+ "Unable to access microphone": "تعذر الوصول إلى الميكروفون"
}
diff --git a/src/i18n/strings/cs.json b/src/i18n/strings/cs.json
index 4d9d27363f..3f9d231317 100644
--- a/src/i18n/strings/cs.json
+++ b/src/i18n/strings/cs.json
@@ -119,7 +119,7 @@
"Error: Problem communicating with the given homeserver.": "Chyba: problém v komunikaci s daným domovským serverem.",
"Existing Call": "Probíhající hovor",
"Export": "Exportovat",
- "Export E2E room keys": "Exportovat end-to-end klíče místnosti",
+ "Export E2E room keys": "Exportovat end-to-end klíče místností",
"Failed to ban user": "Nepodařilo se vykázat uživatele",
"Failed to join room": "Vstup do místnosti se nezdařil",
"Failed to kick": "Vykopnutí se nezdařilo",
@@ -131,14 +131,14 @@
"Failed to send request.": "Odeslání žádosti se nezdařilo.",
"Failed to set display name": "Nepodařilo se nastavit zobrazované jméno",
"Failed to unban": "Přijetí zpět se nezdařilo",
- "Failed to upload profile picture!": "Nahrání profilového obrázku se nezdařilo",
+ "Failed to upload profile picture!": "Nahrání profilového obrázku se nezdařilo!",
"Failure to create room": "Vytvoření místnosti se nezdařilo",
"Forget room": "Zapomenout místnost",
"For security, this session has been signed out. Please sign in again.": "Z bezpečnostních důvodů bylo toto přihlášení ukončeno. Přihlašte se prosím znovu.",
"and %(count)s others...|other": "a %(count)s další...",
"%(widgetName)s widget modified by %(senderName)s": "Uživatel %(senderName)s upravil widget %(widgetName)s",
"%(widgetName)s widget removed by %(senderName)s": "Uživatel %(senderName)s odstranil widget %(widgetName)s",
- "%(widgetName)s widget added by %(senderName)s": "Uživatel %(senderName)s přidal widget %(widgetName)s",
+ "%(widgetName)s widget added by %(senderName)s": "Uživatel %(senderName)s přidal widget %(widgetName)s",
"Automatically replace plain text Emoji": "Automaticky nahrazovat textové emoji",
"Failed to upload image": "Obrázek se nepodařilo nahrát",
"%(senderName)s answered the call.": "Uživatel %(senderName)s přijal hovor.",
@@ -219,7 +219,7 @@
"Submit": "Odeslat",
"Success": "Úspěch",
"The phone number entered looks invalid": "Zadané telefonní číslo se zdá být neplatné",
- "This email address is already in use": "Tato e-mailová adresa je již požívána",
+ "This email address is already in use": "Tato e-mailová adresa je již používána",
"This email address was not found": "Tato e-mailová adresa nebyla nalezena",
"This room has no local addresses": "Tato místnost nemá žádné místní adresy",
"This room is not recognised.": "Tato místnost nebyla rozpoznána.",
@@ -438,7 +438,7 @@
"URL previews are enabled by default for participants in this room.": "Ve výchozím nastavení jsou náhledy URL adres povolené pro členy této místnosti.",
"URL previews are disabled by default for participants in this room.": "Ve výchozím nastavení jsou náhledy URL adres zakázané pro členy této místnosti.",
"Invalid file%(extra)s": "Neplatný soubor%(extra)s",
- "You are about to be taken to a third-party site so you can authenticate your account for use with %(integrationsUrl)s. Do you wish to continue?": "Budete přesměrováni na stránku třetí strany k ověření svého účtu pro používání s %(integrationsUrl)s. Chcete pokračovat?",
+ "You are about to be taken to a third-party site so you can authenticate your account for use with %(integrationsUrl)s. Do you wish to continue?": "Budete přesměrováni na stránku třetí strany k ověření svého účtu pro používání s %(integrationsUrl)s. Chcete pokračovat?",
"Please check your email to continue registration.": "Pro pokračování v registraci prosím zkontrolujte své e-maily.",
"Token incorrect": "Neplatný token",
"A text message has been sent to %(msisdn)s": "Na číslo %(msisdn)s byla odeslána textová zpráva",
@@ -557,7 +557,7 @@
"Failed to add the following users to the summary of %(groupId)s:": "Do souhrnného seznamu skupiny %(groupId)s se nepodařilo přidat následující uživatele:",
"Add a User": "Přidat uživatele",
"Failed to remove a user from the summary of %(groupId)s": "Ze souhrnného seznamu skupiny %(groupId)s se nepodařilo odstranit uživatele",
- "The user '%(displayName)s' could not be removed from the summary.": "Nelze odstranit uživatele '%(displayName)s' ze souhrnného seznamu.",
+ "The user '%(displayName)s' could not be removed from the summary.": "Nelze odstranit uživatele '%(displayName)s' ze souhrnného seznamu.",
"Unable to accept invite": "Nelze přijmout pozvání",
"Unable to reject invite": "Nelze odmítnout pozvání",
"Community Settings": "Nastavení skupiny",
@@ -1332,7 +1332,7 @@
"The homeserver may be unavailable or overloaded.": "Domovský server je nedostupný nebo přetížený.",
"Add room": "Přidat místnost",
"You have %(count)s unread notifications in a prior version of this room.|other": "Máte %(count)s nepřečtených oznámení v předchozí verzi této místnosti.",
- "You have %(count)s unread notifications in a prior version of this room.|one": "Máte jedno nepřečtené oznámení v předchozí verzi této místnosti.",
+ "You have %(count)s unread notifications in a prior version of this room.|one": "Máte %(count)s nepřečtených oznámení v předchozí verzi této místnosti.",
"Your profile": "Váš profil",
"Your Matrix account on ": "Váš účet Matrix na serveru ",
"Failed to get autodiscovery configuration from server": "Nepovedlo se automaticky načíst konfiguraci ze serveru",
@@ -1753,7 +1753,7 @@
"Review": "Prohlédnout",
"This bridge was provisioned by .": "Toto propojení poskytuje .",
"This bridge is managed by .": "Toto propojení spravuje .",
- "Workspace: %(networkName)s": "Workspace: %(networkName)s",
+ "Workspace: %(networkName)s": "Pracovní oblast: %(networkName)s",
"Channel: %(channelName)s": "Kanál: %(channelName)s",
"Show less": "Skrýt detaily",
"Show more": "Více",
@@ -2213,5 +2213,543 @@
"Upload completed": "Nahrávání dokončeno",
"Cancelled signature upload": "Nahrávání podpisu zrušeno",
"Unable to upload": "Nelze nahrát",
- "Server isn't responding": "Server neodpovídá"
+ "Server isn't responding": "Server neodpovídá",
+ "End": "End",
+ "Space": "Mezerník",
+ "Enter": "Enter",
+ "Esc": "Esc",
+ "Page Down": "Page Down",
+ "Page Up": "Page Up",
+ "Cancel autocomplete": "Zrušit automatické doplňování",
+ "Move autocomplete selection up/down": "Posun nahoru/dolu v automatickém doplňování",
+ "Toggle this dialog": "Zobrazit/skrýt tento dialog",
+ "Activate selected button": "Aktivovat označené tlačítko",
+ "Close dialog or context menu": "Zavřít dialog nebo kontextové menu",
+ "Toggle the top left menu": "Zobrazit/skrýt menu vlevo nahoře",
+ "Scroll up/down in the timeline": "Posunous se nahoru/dolů v historii",
+ "Clear room list filter field": "Smazat filtr místností",
+ "Expand room list section": "Rozbalit seznam místností",
+ "Collapse room list section": "Sbalit seznam místností",
+ "Select room from the room list": "Vybrat místnost v seznamu",
+ "Navigate up/down in the room list": "Posouvat se nahoru/dolů v seznamu místností",
+ "Jump to room search": "Filtrovat místnosti",
+ "Toggle video on/off": "Zapnout nebo vypnout video",
+ "Toggle microphone mute": "Ztlumit nebo zapnout mikrofon",
+ "Jump to start/end of the composer": "Skočit na konec/začátek textového pole",
+ "New line": "Nový řádek",
+ "Toggle Quote": "Citace",
+ "Toggle Italics": "Kurzíva",
+ "Toggle Bold": "Tučné písmo",
+ "Ctrl": "Ctrl",
+ "Shift": "Shift",
+ "Alt Gr": "Alt Gr",
+ "Autocomplete": "Automatické doplňování",
+ "Room List": "Seznam místností",
+ "Calls": "Hovory",
+ "To set up a filter, drag a community avatar over to the filter panel on the far left hand side of the screen. You can click on an avatar in the filter panel at any time to see only the rooms and people associated with that community.": "Pro nastavení filtru přetáhněte avatar skupiny do panelu filtrování na levé straně obrazovky. Pak v tomto panelu můžete kdykoliv klepnout na avatar skupiny a uvidíte jen místnosti a lidi z dané skupiny.",
+ "Send feedback": "Odeslat zpětnou vazbu",
+ "Feedback": "Zpětná vazba",
+ "Feedback sent": "Zpětná vazba byla odeslána",
+ "Security & privacy": "Zabezpečení",
+ "All settings": "Nastavení",
+ "Start a conversation with someone using their name, email address or username (like ).": "Napište jméno nebo emailovou adresu uživatele se kterým chcete začít konverzaci (např. ).",
+ "Start a new chat": "Založit nový chat",
+ "Which officially provided instance you are using, if any": "Kterou oficiální instanci Riot.im používáte (a jestli vůbec)",
+ "Change the topic of this room": "Změnit téma této místnosti",
+ "%(senderName)s declined the call.": "%(senderName)s odmítl/a hovor.",
+ "(an error occurred)": "(došlo k chybě)",
+ "(their device couldn't start the camera / microphone)": "(zařízení druhé strany nemohlo spustit kameru / mikrofon)",
+ "(connection failed)": "(spojení selhalo)",
+ "🎉 All servers are banned from participating! This room can no longer be used.": "🎉 K místnosti nemá přístup žádný server! Místnost už nemůže být používána.",
+ "Prepends ( ͡° ͜ʖ ͡°) to a plain-text message": "Vloží ( ͡° ͜ʖ ͡°) na začátek zprávy",
+ "Czech Republic": "Česká republika",
+ "Algeria": "Alžírsko",
+ "Albania": "Albánie",
+ "Åland Islands": "Alandy",
+ "Afghanistan": "Afgánistán",
+ "United States": "Spojené Státy",
+ "United Kingdom": "Spojené Království",
+ "This will end the conference for everyone. Continue?": "Ukončíme konferenci pro všechny účastníky. Chcete pokračovat?",
+ "End conference": "Ukončit konferenci",
+ "No other application is using the webcam": "Webkamera není blokována jinou aplikací",
+ "Permission is granted to use the webcam": "Aplikace má k webkameře povolen přístup",
+ "A microphone and webcam are plugged in and set up correctly": "Mikrofon a webkamera jsou zapojeny a správně nastaveny",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Hovor selhal, protože nešlo použít mikrofon nebo webkameru. Zkontrolujte, že:",
+ "Unable to access webcam / microphone": "Není možné použít webkameru nebo mikrofon",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Hovor selhal, protože nešlo použít mikrofon. Zkontrolujte, že je mikrofon připojen a správně nastaven.",
+ "Unable to access microphone": "Není možné použít mikrofon",
+ "The call was answered on another device.": "Hovor byl přijat na jiném zařízení.",
+ "Answered Elsewhere": "Zodpovězeno jinde",
+ "The call could not be established": "Hovor se nepovedlo navázat",
+ "The other party declined the call.": "Druhá strana hovor odmítla.",
+ "Call Declined": "Hovor odmítnut",
+ "PRO TIP: If you start a bug, please submit debug logs to help us track down the problem.": "TIP pro profíky: Pokud nahlásíte chybu, odešlete prosím ladicí protokoly, které nám pomohou problém vypátrat.",
+ "Please view existing bugs on Github first. No match? Start a new one.": "Nejříve si prosím prohlédněte existující chyby na Githubu. Žádná shoda? Nahlašte novou chybu.",
+ "Report a bug": "Nahlásit chybu",
+ "Add widgets, bridges & bots": "Přidat widgety, bridge a boty",
+ "Widgets": "Widgety",
+ "Show Widgets": "Zobrazit widgety",
+ "Hide Widgets": "Skrýt widgety",
+ "Room settings": "Nastavení místnosti",
+ "Use the Desktop app to see all encrypted files": "Pomocí desktopové aplikace zobrazíte všechny šifrované soubory",
+ "Attach files from chat or just drag and drop them anywhere in a room.": "Připojte soubory z chatu nebo je jednoduše přetáhněte kamkoli do místnosti.",
+ "No files visible in this room": "V této místnosti nejsou viditelné žádné soubory",
+ "Show files": "Zobrazit soubory",
+ "%(count)s people|other": "%(count)s lidé",
+ "About": "O",
+ "You’re all caught up": "Vše vyřízeno",
+ "You have no visible notifications in this room.": "V této místnosti nemáte žádná viditelná oznámení.",
+ "Hey you. You're the best!": "Hej ty. Jsi nejlepší!",
+ "Secret storage:": "Bezpečné úložiště:",
+ "Backup key cached:": "Klíč zálohy cachován:",
+ "Backup key stored:": "Klíč zálohy uložen:",
+ "Backup version:": "Verze zálohy:",
+ "Algorithm:": "Algoritmus:",
+ "You might enable this if the room will only be used for collaborating with internal teams on your homeserver. This cannot be changed later.": "Tuto možnost můžete povolit, pokud bude místnost použita pouze pro spolupráci s interními týmy na vašem domovském serveru. Toto nelze později změnit.",
+ "Block anyone not part of %(serverName)s from ever joining this room.": "Blokovat komukoli, kdo není součástí serveru %(serverName)s, aby se nikdy nepřipojil do této místnosti.",
+ "Back up your encryption keys with your account data in case you lose access to your sessions. Your keys will be secured with a unique Recovery Key.": "Zálohujte šifrovací klíče s daty vašeho účtu pro případ, že ztratíte přístup k relacím. Vaše klíče budou zabezpečeny jedinečným klíčem pro obnovení.",
+ "Manage the names of and sign out of your sessions below or verify them in your User Profile.": "Níže můžete spravovat názvy a odhlásit se ze svých relací nebo je ověřit v uživatelském profilu.",
+ "or another cross-signing capable Matrix client": "nebo jiný Matrix klient schopný cross-signing",
+ "Cross-signing is not set up.": "Cross-signing není nastaveno.",
+ "Cross-signing is ready for use.": "Cross-signing je připraveno k použití.",
+ "Create a Group Chat": "Vytvořit skupinový chat",
+ "Send a Direct Message": "Poslat přímou zprávu",
+ "This requires the latest %(brand)s on your other devices:": "To vyžaduje nejnovější %(brand)s na vašich ostatních zařízeních:",
+ "Confirm your identity by verifying this login from one of your other sessions, granting it access to encrypted messages.": "Potvrďte svou identitu ověřením tohoto přihlášení z jedné z vašich dalších relací a udělte mu přístup k šifrovaným zprávám.",
+ "Verify this login": "Ověřte toto přihlášení",
+ "Welcome to %(appName)s": "Vítá vás %(appName)s",
+ "Liberate your communication": "Osvoboďte svou komunikaci",
+ "Navigation": "Navigace",
+ "Use the + to make a new room or explore existing ones below": "Pomocí + vytvořte novou místnost nebo prozkoumejte stávající místnosti",
+ "Secure Backup": "Zabezpečená záloha",
+ "Jump to oldest unread message": "Jít na nejstarší nepřečtenou zprávu",
+ "Upload a file": "Nahrát soubor",
+ "You've reached the maximum number of simultaneous calls.": "Dosáhli jste maximálního počtu souběžných hovorů.",
+ "Too Many Calls": "Přiliš mnoho hovorů",
+ "Community and user menu": "Nabídka komunity a uživatele",
+ "User menu": "Uživatelská nabídka",
+ "Switch theme": "Přepnout téma",
+ "Switch to dark mode": "Přepnout do tmavého režimu",
+ "Switch to light mode": "Přepnout do světlého režimu",
+ "User settings": "Uživatelská nastavení",
+ "Community settings": "Nastavení komunity",
+ "Confirm your recovery passphrase": "Potvrďte vaši frázi pro obnovení",
+ "Repeat your recovery passphrase...": "Opakujte přístupovou frázi pro obnovení...",
+ "Please enter your recovery passphrase a second time to confirm.": "Potvrďte prosím podruhé svou frázi pro obnovení.",
+ "Use a different passphrase?": "Použít jinou frázi?",
+ "Great! This recovery passphrase looks strong enough.": "Skvělé! Tato fráze pro obnovení vypadá dostatečně silně.",
+ "Enter a recovery passphrase": "Zadejte frázi pro obnovení",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s nebo %(usernamePassword)s",
+ "If you've joined lots of rooms, this might take a while": "Pokud jste se připojili ke spoustě místností, může to chvíli trvat",
+ "There was a problem communicating with the homeserver, please try again later.": "Při komunikaci s domovským serverem došlo k potížím, zkuste to prosím později.",
+ "Continue with %(ssoButtons)s": "Pokračovat s %(ssoButtons)s",
+ "Use Recovery Key": "Použít klíč pro obnovu",
+ "Use Recovery Key or Passphrase": "Použít klíč pro obnovu nebo frázi",
+ "Already have an account? Sign in here": "Máte již účet? Přihlašte se zde",
+ "Host account on": "Hostovat účet na",
+ "Signing In...": "Přihlašování...",
+ "Syncing...": "Synchronizuji...",
+ "That username already exists, please try another.": "Toto uživatelské jméno již existuje, zkuste prosím jiné.",
+ "Verify other session": "Ověření jiné relace",
+ "Filter rooms and people": "Filtrovat místnosti a lidi",
+ "Explore rooms in %(communityName)s": "Prozkoumejte místnosti v %(communityName)s",
+ "delete the address.": "smazat adresu.",
+ "Delete the room address %(alias)s and remove %(name)s from the directory?": "Smazat adresu místnosti %(alias)s a odebrat %(name)s z adresáře?",
+ "Self-verification request": "Požadavek na sebeověření",
+ "%(creator)s created this DM.": "%(creator)s vytvořil tuto přímou zprávu.",
+ "You do not have permission to create rooms in this community.": "Nemáte oprávnění k vytváření místností v této komunitě.",
+ "Cannot create rooms in this community": "V této komunitě nelze vytvořit místnosti",
+ "Great, that'll help people know it's you": "Skvělé, to pomůže lidem zjistit, že jste to vy",
+ "Add a photo so people know it's you.": "Přidejte fotku, aby lidé věděli, že jste to vy.",
+ "Explore Public Rooms": "Prozkoumat veřejné místnosti",
+ "Welcome %(name)s": "Vítejte %(name)s",
+ "Now, let's help you get started": "Nyní vám pomůžeme začít",
+ "Effects": "Efekty",
+ "Alt": "Alt",
+ "Approve": "Schválit",
+ "Looks good!": "To vypadá dobře!",
+ "Wrong file type": "Špatný typ souboru",
+ "The server has denied your request.": "Server odmítl váš požadavek.",
+ "The server is offline.": "Server je offline.",
+ "not ready": "nepřipraveno",
+ "Remove messages sent by others": "Odstranit zprávy odeslané ostatními",
+ "Decline All": "Odmítnout vše",
+ "Use the Desktop app to search encrypted messages": "K prohledávání šifrovaných zpráv použijte aplikaci pro stolní počítače",
+ "Invite someone using their name, email address, username (like ) or share this room.": "Pozvěte někoho pomocí jeho jména, e-mailové adresy, uživatelského jména (například ) nebo sdílejte tuto místnost.",
+ "Super": "Super",
+ "Toggle right panel": "Zobrazit/skrýt pravý panel",
+ "Add a photo, so people can easily spot your room.": "Přidejte fotografii, aby lidé mohli snadno najít váši místnost.",
+ "%(displayName)s created this room.": "%(displayName)s vytvořil tuto místnost.",
+ "This is the start of .": "Toto je začátek místnosti .",
+ "Topic: %(topic)s ": "Téma: %(topic)s ",
+ "Topic: %(topic)s (edit)": "Téma: %(topic)s (upravit)",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Zprávy zde jsou šifrovány end-to-end. Ověřte uživatele %(displayName)s v jeho profilu - klepněte na jeho avatar.",
+ "Enter a security phrase only you know, as it’s used to safeguard your data. To be secure, you shouldn’t re-use your account password.": "Zadejte frázi zabezpečení, kterou znáte jen vy, k ochraně vašich dat. Z důvodu bezpečnosti byste neměli znovu používat heslo k účtu.",
+ "Use a secret phrase only you know, and optionally save a Security Key to use for backup.": "Použijte tajnou frázi, kterou znáte pouze vy, a volitelně uložte bezpečnostní klíč, který použijete pro zálohování.",
+ "Enter a Security Phrase": "Zadání bezpečnostní fráze",
+ "We’ll generate a Security Key for you to store somewhere safe, like a password manager or a safe.": "Vygenerujeme bezpečnostní klíč, který můžete uložit někde v bezpečí, například ve správci hesel nebo trezoru.",
+ "Generate a Security Key": "Vygenerovat bezpečnostní klíč",
+ "Set up Secure Backup": "Nastavení zabezpečené zálohy",
+ "Safeguard against losing access to encrypted messages & data by backing up encryption keys on your server.": "Chraňte se před ztrátou přístupu k šifrovaným zprávám a datům zálohováním šifrovacích klíčů na serveru.",
+ "Safeguard against losing access to encrypted messages & data": "Zabezpečení proti ztrátě přístupu k šifrovaným zprávám a datům",
+ "This is the beginning of your direct message history with .": "Toto je začátek historie vašich přímých zpráv s uživatelem .",
+ "Only the two of you are in this conversation, unless either of you invites anyone to join.": "V této konverzaci jste pouze vy dva, dokud někdo z vás nepozve někoho dalšího.",
+ "You created this room.": "Vytvořili jste tuto místnost.",
+ "Add a topic to help people know what it is about.": "Přidejte téma, aby lidé věděli, o co jde.",
+ "Invite by email": "Pozvat emailem",
+ "Comment": "Komentář",
+ "Add comment": "Přidat komentář",
+ "Update community": "Aktualizovat komunitu",
+ "Create a room in %(communityName)s": "Vytvořit místnost v %(communityName)s",
+ "Your server requires encryption to be enabled in private rooms.": "Váš server vyžaduje povolení šifrování v soukromých místnostech.",
+ "An image will help people identify your community.": "Obrázek pomůže lidem identifikovat vaši komunitu.",
+ "Invite people to join %(communityName)s": "Pozvat lidi do %(communityName)s",
+ "Send %(count)s invites|one": "Poslat %(count)s pozvánku",
+ "Send %(count)s invites|other": "Poslat %(count)s pozvánek",
+ "People you know on %(brand)s": "Lidé, které znáte z %(brand)s",
+ "Add another email": "Přidat další emailovou adresu",
+ "Show": "Zobrazit",
+ "Reason (optional)": "Důvod (volitelné)",
+ "Add image (optional)": "Přidat obrázek (volitelné)",
+ "Private rooms can be found and joined by invitation only. Public rooms can be found and joined by anyone.": "Do soukromé místnosti se lze připojit pouze s pozvánkou. Veřejné místnosti lze najít a může se do nich připojit kdokoli.",
+ "Currently indexing: %(currentRoom)s": "Aktuálně se indexuje: %(currentRoom)s",
+ "Secure your backup with a recovery passphrase": "Zabezpečte zálohu pomocí fráze pro obnovení",
+ "%(brand)s Android": "%(brand)s Android",
+ "%(brand)s iOS": "%(brand)s iOS",
+ "%(brand)s Desktop": "%(brand)s Desktop",
+ "%(brand)s Web": "%(brand)s Web",
+ "Use email to optionally be discoverable by existing contacts.": "Pomocí e-mailu můžete být volitelně viditelní pro existující kontakty.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Použijte e-mail nebo telefon, abyste byli volitelně viditelní pro stávající kontakty.",
+ "Sign in with SSO": "Přihlásit pomocí SSO",
+ "Create community": "Vytvořit komunitu",
+ "Add an email to be able to reset your password.": "Přidejte email, abyste mohli obnovit své heslo.",
+ "That phone number doesn't look quite right, please check and try again": "Toto telefonní číslo nevypadá úplně správně, zkontrolujte ho a zkuste to znovu",
+ "Forgot password?": "Zapomenuté heslo?",
+ "Enter phone number": "Zadejte telefonní číslo",
+ "Enter email address": "Zadejte emailovou adresu",
+ "Open the link in the email to continue registration.": "Pro pokračování v registraci, otevřete odkaz v e-mailu.",
+ "A confirmation email has been sent to %(emailAddress)s": "Na adresu %(emailAddress)s byl zaslán potvrzovací e-mail",
+ "Take a picture": "Vyfotit",
+ "Hold": "Podržet",
+ "Resume": "Pokračovat",
+ "Successfully restored %(sessionCount)s keys": "Úspěšně obnoveno %(sessionCount)s klíčů",
+ "Keys restored": "Klíče byly obnoveny",
+ "You're all caught up.": "Vše vyřízeno.",
+ "There was an error updating your community. The server is unable to process your request.": "Při aktualizaci komunity došlo k chybě. Server nemůže zpracovat váš požadavek.",
+ "There are two ways you can provide feedback and help us improve %(brand)s.": "Jsou dva způsoby, jak můžete poskytnout zpětnou vazbu a pomoci nám vylepšit %(brand)s.",
+ "Rate %(brand)s": "Ohodnotit %(brand)s",
+ "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.": "V této relaci jste již dříve používali novější verzi %(brand)s. Chcete-li tuto verzi znovu použít s šifrováním, budete se muset odhlásit a znovu přihlásit.",
+ "Homeserver": "Homeserver",
+ "Continue with %(provider)s": "Pokračovat s %(provider)s",
+ "Server Options": "Možnosti serveru",
+ "This version of %(brand)s does not support viewing some encrypted files": "Tato verze %(brand)s nepodporuje zobrazení některých šifrovaných souborů",
+ "This version of %(brand)s does not support searching encrypted messages": "Tato verze %(brand)s nepodporuje hledání v šifrovaných zprávách",
+ "Information": "Informace",
+ "%(count)s results|one": "%(count)s výsledek",
+ "Explore community rooms": "Prozkoumat místnosti komunity",
+ "Role": "Role",
+ "Madagascar": "Madagaskar",
+ "Macedonia": "Makedonie",
+ "Macau": "Macao",
+ "Luxembourg": "Lucembursko",
+ "Lithuania": "Litva",
+ "Liechtenstein": "Lichtenštejnsko",
+ "Libya": "Libye",
+ "Liberia": "Libérie",
+ "Lesotho": "Lesotho",
+ "Lebanon": "Libanon",
+ "Latvia": "Lotyšsko",
+ "Laos": "Laos",
+ "Kyrgyzstan": "Kyrgyzstán",
+ "Myanmar": "Myanmar",
+ "Mozambique": "Mosambik",
+ "Morocco": "Maroko",
+ "Montserrat": "Montserrat",
+ "Montenegro": "Černá Hora",
+ "Mongolia": "Mongolsko",
+ "Monaco": "Monako",
+ "Moldova": "Moldavsko",
+ "Micronesia": "Mikronésie",
+ "Mexico": "Mexiko",
+ "Mayotte": "Mayotte",
+ "Mauritius": "Mauricius",
+ "Mauritania": "Mauretánie",
+ "Martinique": "Martinik",
+ "Marshall Islands": "Marshallovy ostrovy",
+ "Malta": "Malta",
+ "Mali": "Mali",
+ "Maldives": "Maledivy",
+ "Malaysia": "Malajsie",
+ "Malawi": "Malawi",
+ "Kuwait": "Kuwait",
+ "Kosovo": "Kosovo",
+ "Kiribati": "Kiribati",
+ "Kenya": "Keňa",
+ "Kazakhstan": "Kazachstán",
+ "Congo - Brazzaville": "Kongo - Brazzaville",
+ "Cambodia": "Kambodža",
+ "Burundi": "Burundi",
+ "Burkina Faso": "Burkina Faso",
+ "Bulgaria": "Bulharsko",
+ "Brunei": "Brunej",
+ "British Virgin Islands": "Britské indickooceánské území",
+ "British Indian Ocean Territory": "Britské indickooceánské území",
+ "Brazil": "Brazílie",
+ "Bouvet Island": "Bouvetův ostrov",
+ "Botswana": "Botswana",
+ "Bosnia": "Bosna",
+ "Bolivia": "Bolívie",
+ "Bhutan": "Bhútán",
+ "Bermuda": "Bermudy",
+ "Jordan": "Jordánsko",
+ "Jersey": "Jersey",
+ "Japan": "Japonsko",
+ "Jamaica": "Jamajka",
+ "Italy": "Itálie",
+ "Israel": "Izrael",
+ "Isle of Man": "Ostrov Man",
+ "Ireland": "Irsko",
+ "Iraq": "Irák",
+ "Iran": "Írán",
+ "Indonesia": "Indonésie",
+ "India": "Indie",
+ "Iceland": "Island",
+ "Hungary": "Maďarsko",
+ "Hong Kong": "Hong Kong",
+ "Honduras": "Honduras",
+ "Heard & McDonald Islands": "Heardovy a McDonaldovy ostrovy",
+ "Haiti": "Haiti",
+ "Guyana": "Guyana",
+ "Guinea-Bissau": "Guinea-Bissau",
+ "Guinea": "Guinea",
+ "Guernsey": "Guernsey",
+ "Guatemala": "Guatemala",
+ "Guam": "Guam",
+ "Guadeloupe": "Guadeloupe",
+ "Grenada": "Grenada",
+ "Greenland": "Grónsko",
+ "Greece": "Řecko",
+ "Gibraltar": "Gibraltar",
+ "Ghana": "Ghana",
+ "Germany": "Německo",
+ "Georgia": "Gruzie",
+ "Gambia": "Gambie",
+ "Gabon": "Gabon",
+ "French Southern Territories": "Francouzská jižní území",
+ "French Polynesia": "Francouzská Polynésie",
+ "French Guiana": "Francouzská Guyana",
+ "France": "Francie",
+ "Finland": "Finsko",
+ "Fiji": "Fiji",
+ "Faroe Islands": "Faerské ostrovy",
+ "Falkland Islands": "Falklandy",
+ "Ethiopia": "Etiopie",
+ "Estonia": "Estonsko",
+ "Eritrea": "Eritrea",
+ "Equatorial Guinea": "Rovníková Guinea",
+ "El Salvador": "El Salvador",
+ "Egypt": "Egypt",
+ "Ecuador": "Ekvádor",
+ "Dominican Republic": "Dominikánská republika",
+ "Dominica": "Dominika",
+ "Djibouti": "Džibuti",
+ "Denmark": "Dánsko",
+ "Côte d’Ivoire": "Pobřeží slonoviny",
+ "Cyprus": "Kypr",
+ "Curaçao": "Curaçao",
+ "Cuba": "Kuba",
+ "Croatia": "Chorvatsko",
+ "Costa Rica": "Kostarika",
+ "Cook Islands": "Cookovy ostrovy",
+ "Congo - Kinshasa": "Kongo - Brazzaville",
+ "Comoros": "Komory",
+ "Colombia": "Kolumbie",
+ "Cocos (Keeling) Islands": "Kokosové (Keelingovy) ostrovy",
+ "Christmas Island": "Vánoční ostrov",
+ "China": "Čína",
+ "Chile": "Chile",
+ "Chad": "Čad",
+ "Central African Republic": "Středoafrická republika",
+ "Cayman Islands": "Kajmanské ostrovy",
+ "Caribbean Netherlands": "Karibské Nizozemsko",
+ "Cape Verde": "Kapverdy",
+ "Canada": "Kanada",
+ "Cameroon": "Kamerun",
+ "Benin": "Benin",
+ "Belize": "Belize",
+ "Belgium": "Belgie",
+ "Belarus": "Bělorusko",
+ "Bangladesh": "Bangladéš",
+ "Barbados": "Barbados",
+ "Bahrain": "Bahrain",
+ "Bahamas": "Bahamy",
+ "Azerbaijan": "Ázerbajdžán",
+ "Austria": "Rakousko",
+ "Australia": "Austrálie",
+ "Aruba": "Aruba",
+ "Armenia": "Arménie",
+ "Argentina": "Argentina",
+ "Antigua & Barbuda": "Antigua a Barbuda",
+ "Antarctica": "Antarktida",
+ "Anguilla": "Anguilla",
+ "Angola": "Angola",
+ "Andorra": "Andorra",
+ "American Samoa": "Americká Samoa",
+ "Room Info": "Informace o místnosti",
+ "Enable desktop notifications": "Povolit oznámení na ploše",
+ "Invalid Recovery Key": "Neplatný klíč pro obnovení",
+ "Security Key": "Bezpečnostní klíč",
+ "Use your Security Key to continue.": "Pokračujte pomocí bezpečnostního klíče.",
+ "If they don't match, the security of your communication may be compromised.": "Pokud se neshodují, bezpečnost vaší komunikace může být kompromitována.",
+ "Confirm this user's session by comparing the following with their User Settings:": "Potvrďte relaci tohoto uživatele porovnáním následujícího s jeho uživatelským nastavením:",
+ "Confirm Security Phrase": "Potvrďte bezpečnostní frázi",
+ "Privacy": "Soukromí",
+ "Click the button below to confirm setting up encryption.": "Kliknutím na tlačítko níže potvrďte nastavení šifrování.",
+ "Confirm encryption setup": "Potvrďte nastavení šifrování",
+ "Return to call": "Návrat do hovoru",
+ "Unable to set up keys": "Nepovedlo se nastavit klíče",
+ "Save your Security Key": "Uložte svůj bezpečnostní klíč",
+ "We call the places where you can host your account ‘homeservers’.": "Místům, kde můžete hostovat svůj účet, říkáme ‘domovské servery’.",
+ "About homeservers": "O domovských serverech",
+ "Learn more": "Zjistit více",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Použijte svůj preferovaný domovský server Matrix, pokud ho máte, nebo hostujte svůj vlastní.",
+ "Other homeserver": "Jiný domovský server",
+ "Sign into your homeserver": "Přihlaste se do svého domovského serveru",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org je největší veřejný domovský server na světě, pro mnoho lidí je dobrým místem.",
+ "not found in storage": "nebylo nalezeno v úložišti",
+ "ready": "připraven",
+ "May include members not in %(communityName)s": "Může zahrnovat členy, kteří nejsou v %(communityName)s",
+ "Specify a homeserver": "Zadejte domovský server",
+ "Invalid URL": "Neplatné URL",
+ "Unable to validate homeserver": "Nelze ověřit domovský server",
+ "New? Create account": "Jste zde nový? Vytvořte si účet",
+ "Navigate recent messages to edit": "Procházet poslední zprávy k úpravám",
+ "Navigate composer history": "Procházet historii editoru",
+ "Cancel replying to a message": "Zrušení odpovědi na zprávu",
+ "Don't miss a reply": "Nezmeškejte odpovědět",
+ "Unknown App": "Neznámá aplikace",
+ "%(name)s paused": "%(name)s pozastaven",
+ "Backup could not be decrypted with this recovery key: please verify that you entered the correct recovery key.": "Zálohu nebylo možné dešifrovat pomocí tohoto klíče pro obnovení: ověřte, zda jste zadali správný klíč pro obnovení.",
+ "Move right": "Posunout doprava",
+ "Move left": "Posunout doleva",
+ "Go to Home View": "Přejít na domovské zobrazení",
+ "Dismiss read marker and jump to bottom": "Zavřít značku přečtených zpráv a skočit dolů",
+ "Previous/next room or DM": "Předchozí/další místnost nebo přímá zpráva",
+ "Previous/next unread room or DM": "Předchozí/další nepřečtená místnost nebo přímá zpráva",
+ "Not encrypted": "Není šifrováno",
+ "New here? Create an account": "Jste zde nový? Vytvořte si účet",
+ "Got an account? Sign in": "Máte již účet? Přihlásit se",
+ "This won't invite them to %(communityName)s. To invite someone to %(communityName)s, click here": "Toto je nepozve do %(communityName)s. Chcete-li někoho pozvat na %(communityName)s, klikněte sem",
+ "Approve widget permissions": "Schválit oprávnění widgetu",
+ "Enter name": "Zadejte jméno",
+ "Ignored attempt to disable encryption": "Ignorovaný pokus o deaktivaci šifrování",
+ "Show chat effects": "Zobrazit efekty chatu",
+ "%(count)s people|one": "%(count)s člověk",
+ "Takes the call in the current room off hold": "Zruší podržení hovoru v aktuální místnosti",
+ "Places the call in the current room on hold": "Podrží hovor v aktuální místnosti",
+ "Zimbabwe": "Zimbabwe",
+ "Zambia": "Zambie",
+ "Yemen": "Jemen",
+ "Western Sahara": "Západní Sahara",
+ "Wallis & Futuna": "Wallis a Futuna",
+ "Vietnam": "Vietnam",
+ "Venezuela": "Venezuela",
+ "Vatican City": "Vatikán",
+ "Vanuatu": "Vanuatu",
+ "Uzbekistan": "Uzbekistán",
+ "Uruguay": "Uruguay",
+ "United Arab Emirates": "Spojené arabské emiráty",
+ "Ukraine": "Ukrajina",
+ "Uganda": "Uganda",
+ "U.S. Virgin Islands": "Panenské ostrovy",
+ "Tuvalu": "Tuvalu",
+ "Turks & Caicos Islands": "Ostrovy Turks a Caicos",
+ "Turkmenistan": "Turkmenistán",
+ "Turkey": "Turecko",
+ "Tunisia": "Tunisko",
+ "Trinidad & Tobago": "Trinidad a Tobago",
+ "Tonga": "Tonga",
+ "Tokelau": "Tokelau",
+ "Togo": "Togo",
+ "Timor-Leste": "Východní Timor",
+ "Thailand": "Thajsko",
+ "Tanzania": "Tanzánie",
+ "Tajikistan": "Tádžikistán",
+ "Taiwan": "Taiwan",
+ "São Tomé & Príncipe": "Svatý Tomáš a Princův ostrov",
+ "Syria": "Sýrie",
+ "Switzerland": "Švýcarsko",
+ "Sweden": "Švédsko",
+ "Swaziland": "Svazijsko",
+ "Svalbard & Jan Mayen": "Špicberky a Jan Mayen",
+ "Suriname": "Surinam",
+ "Sudan": "Súdán",
+ "St. Vincent & Grenadines": "Svatý Vincenc a Grenadiny",
+ "St. Pierre & Miquelon": "Saint Pierre a Miquelon",
+ "St. Martin": "Svatý Martin",
+ "St. Lucia": "Svatá Lucie",
+ "St. Kitts & Nevis": "Svatý Kryštof a Nevis",
+ "St. Helena": "Svatá Helena",
+ "St. Barthélemy": "Svatý Bartoloměj",
+ "Sri Lanka": "Srí Lanka",
+ "Spain": "Španělsko",
+ "South Sudan": "Jižní Súdán",
+ "South Korea": "Jižní Korea",
+ "South Georgia & South Sandwich Islands": "Jižní Georgie a Jižní Sandwichovy ostrovy",
+ "South Africa": "Jižní Afrika",
+ "Somalia": "Somálsko",
+ "Solomon Islands": "Šalomounovy ostrovy",
+ "Slovenia": "Slovinsko",
+ "Slovakia": "Slovensko",
+ "Sint Maarten": "Sint Maarten",
+ "Singapore": "Singapur",
+ "Sierra Leone": "Sierra Leone",
+ "Seychelles": "Seychely",
+ "Serbia": "Srbsko",
+ "Senegal": "Senegal",
+ "Saudi Arabia": "Saudská arábie",
+ "San Marino": "San Marino",
+ "Samoa": "Samoa",
+ "Réunion": "Réunion",
+ "Rwanda": "Rwanda",
+ "Russia": "Rusko",
+ "Romania": "Rumunsko",
+ "Qatar": "Katar",
+ "Puerto Rico": "Portoriko",
+ "Portugal": "Portugalsko",
+ "Poland": "Polsko",
+ "Pitcairn Islands": "Pitcairnovy ostrovy",
+ "Philippines": "Filipíny",
+ "Peru": "Peru",
+ "Paraguay": "Paraguay",
+ "Papua New Guinea": "Papua-Nová Guinea",
+ "Panama": "Panama",
+ "Palestine": "Palestina",
+ "Palau": "Palau",
+ "Pakistan": "Pákistán",
+ "Oman": "Omán",
+ "Norway": "Norsko",
+ "Northern Mariana Islands": "Severní Mariany",
+ "North Korea": "Severní Korea",
+ "Norfolk Island": "Ostrov Norfolk",
+ "Niue": "Niue",
+ "Nigeria": "Nigérie",
+ "Niger": "Niger",
+ "Nicaragua": "Nikaragua",
+ "New Zealand": "Nový Zéland",
+ "New Caledonia": "Nová Kaledonie",
+ "Netherlands": "Holandsko",
+ "Nepal": "Nepál",
+ "Nauru": "Nauru",
+ "Namibia": "Namibie",
+ "Security Phrase": "Bezpečnostní fráze",
+ "Wrong Recovery Key": "Nesprávný klíč pro obnovení",
+ "Fetching keys from server...": "Načítání klíčů ze serveru ...",
+ "Use Command + Enter to send a message": "K odeslání zprávy použijte Command + Enter",
+ "Use Ctrl + Enter to send a message": "K odeslání zprávy použijte Ctrl + Enter",
+ "Decide where your account is hosted": "Rozhodněte, kde je váš účet hostován",
+ "Unable to query secret storage status": "Nelze zjistit stav úložiště klíčů",
+ "Update %(brand)s": "Aktualizovat %(brand)s",
+ "You can also set up Secure Backup & manage your keys in Settings.": "Zabezpečené zálohování a správu klíčů můžete také nastavit v Nastavení.",
+ "Set a Security Phrase": "Nastavit bezpečnostní frázi"
}
diff --git a/src/i18n/strings/de_DE.json b/src/i18n/strings/de_DE.json
index b2e65e3f32..8588ccb7d6 100644
--- a/src/i18n/strings/de_DE.json
+++ b/src/i18n/strings/de_DE.json
@@ -301,7 +301,7 @@
"Drop file here to upload": "Datei hier loslassen zum hochladen",
"Idle": "Untätig",
"Ongoing conference call%(supportedText)s.": "Laufendes Konferenzgespräch%(supportedText)s.",
- "You are about to be taken to a third-party site so you can authenticate your account for use with %(integrationsUrl)s. Do you wish to continue?": "Du wirst jetzt auf die Website eines Drittanbieters weitergeleitet, damit du dein Benutzerkonto für die Verwendung von %(integrationsUrl)s authentifizieren kannst. Möchtest du fortfahren?",
+ "You are about to be taken to a third-party site so you can authenticate your account for use with %(integrationsUrl)s. Do you wish to continue?": "Du wirst jetzt auf die Website eines Drittanbieters weitergeleitet, damit du dein Benutzerkonto für die Verwendung von %(integrationsUrl)s authentifizieren kannst. Möchtest du fortfahren?",
"Start automatically after system login": "Nach System-Login automatisch starten",
"Jump to first unread message.": "Zur ersten ungelesenen Nachricht springen.",
"Options": "Optionen",
@@ -604,9 +604,9 @@
"Flair": "Abzeichen",
"Showing flair for these communities:": "Abzeichen für diese Communities zeigen:",
"This room is not showing flair for any communities": "Dieser Raum zeigt für keine Communities die Abzeichen an",
- "Something went wrong when trying to get your communities.": "Beim Laden deiner Communites ist etwas schief gelaufen.",
+ "Something went wrong when trying to get your communities.": "Beim Laden deiner Communities ist etwas schief gelaufen.",
"Display your community flair in rooms configured to show it.": "Zeige deinen Community-Flair in den Räumen, die es erlauben.",
- "This homeserver doesn't offer any login flows which are supported by this client.": "Dieser Heimserver verfügt über keinen, von diesem Client unterstütztes Anmeldeverfahren.",
+ "This homeserver doesn't offer any login flows which are supported by this client.": "Dieser Heimserver verfügt über kein von diesem Client unterstütztes Anmeldeverfahren.",
"Call Failed": "Anruf fehlgeschlagen",
"Send": "Senden",
"collapse": "Verbergen",
@@ -711,7 +711,7 @@
"Enter keywords separated by a comma:": "Schlüsselwörter kommagetrennt eingeben:",
"Forward Message": "Nachricht weiterleiten",
"You have successfully set a password and an email address!": "Du hast erfolgreich ein Passwort und eine E-Mail-Adresse gesetzt!",
- "Remove %(name)s from the directory?": "Soll der Raum %(name)s aus dem Verzeichnis entfernt werden?",
+ "Remove %(name)s from the directory?": "Soll der Raum %(name)s aus dem Verzeichnis entfernt werden?",
"%(brand)s uses many advanced browser features, some of which are not available or experimental in your current browser.": "%(brand)s nutzt zahlreiche fortgeschrittene Browser-Funktionen, die teilweise in deinem aktuell verwendeten Browser noch nicht verfügbar sind oder sich noch im experimentellen Status befinden.",
"Developer Tools": "Entwicklerwerkzeuge",
"Preparing to send logs": "Senden von Logs wird vorbereitet",
@@ -1099,7 +1099,7 @@
"Anchor": "Anker",
"Headphones": "Kopfhörer",
"Folder": "Ordner",
- "Pin": "Stecknadel",
+ "Pin": "Anheften",
"Timeline": "Chatverlauf",
"Autocomplete delay (ms)": "Verzögerung zur Autovervollständigung (ms)",
"Roles & Permissions": "Rollen & Berechtigungen",
@@ -2056,7 +2056,7 @@
"Your new session is now verified. Other users will see it as trusted.": "Deine neue Sitzung ist nun verifiziert. Andere Benutzer sehen sie als vertrauenswürdig an.",
"well formed": "wohlgeformt",
"If you don't want to use to discover and be discoverable by existing contacts you know, enter another identity server below.": "Wenn du nicht verwenden willst, um Kontakte zu finden und von anderen gefunden zu werden, trage unten einen anderen Identitätsserver ein.",
- "To report a Matrix-related security issue, please read the Matrix.org Security Disclosure Policy.": "Wenn du einen sicherheitsrelevaten Fehler melden möchtest, lies bitte die Matrix.org Security Disclosure Policy.",
+ "To report a Matrix-related security issue, please read the Matrix.org Security Disclosure Policy.": "Wenn du einen sicherheitsrelevanten Fehler melden möchtest, lies bitte die Matrix.org Security Disclosure Policy.",
"An error occurred changing the room's power level requirements. Ensure you have sufficient permissions and try again.": "Beim Ändern der Anforderungen für Benutzerrechte ist ein Fehler aufgetreten. Stelle sicher dass du die nötigen Berechtigungen besitzt und versuche es erneut.",
"An error occurred changing the user's power level. Ensure you have sufficient permissions and try again.": "Beim Ändern der Benutzerrechte ist ein Fehler aufgetreten. Stelle sicher dass du die nötigen Berechtigungen besitzt und versuche es erneut.",
"Unable to share email address": "E-Mail Adresse konnte nicht geteilt werden",
@@ -2187,8 +2187,8 @@
"Font size": "Schriftgröße",
"IRC display name width": "Breite des IRC Anzeigenamens",
"Size must be a number": "Größe muss eine Zahl sein",
- "Custom font size can only be between %(min)s pt and %(max)s pt": "Eigene Schriftgröße kann nur eine Zahl zwischen %(min)s pt und %(max)s pt sein",
- "Use between %(min)s pt and %(max)s pt": "Verwende eine Zahl zwischen %(min)s pt und %(max)s pt",
+ "Custom font size can only be between %(min)s pt and %(max)s pt": "Eigene Schriftgröße kann nur eine Zahl zwischen %(min)s pt und %(max)s pt sein",
+ "Use between %(min)s pt and %(max)s pt": "Verwende eine Zahl zwischen %(min)s pt und %(max)s pt",
"Appearance": "Erscheinungsbild",
"Create room": "Raum erstellen",
"Jump to oldest unread message": "Zur ältesten ungelesenen Nachricht springen",
@@ -2552,5 +2552,403 @@
"Report a bug": "Einen Fehler melden",
"Add comment": "Kommentar hinzufügen",
"Rate %(brand)s": "%(brand)s bewerten",
- "Feedback sent": "Feedback gesendet"
+ "Feedback sent": "Feedback gesendet",
+ "Takes the call in the current room off hold": "Beendet das Halten des Anrufs",
+ "Places the call in the current room on hold": "Den aktuellen Anruf halten",
+ "Uzbekistan": "Usbekistan",
+ "Send stickers into this room": "Stickers in diesen Raum senden",
+ "Send stickers into your active room": "Stickers in deinen aktiven Raum senden",
+ "Change which room you're viewing": "Ändern welchen Raum du siehst",
+ "Change the topic of this room": "Das Thema von diesem Raum ändern",
+ "See when the topic changes in this room": "Sehen wenn sich das Thema in diesem Raum ändert",
+ "Change the topic of your active room": "Das Thema von deinem aktiven Raum ändern",
+ "See when the topic changes in your active room": "Sehen wenn sich das Thema in deinem aktiven Raum ändert",
+ "Change the name of this room": "Name von diesem Raum ändern",
+ "See when the name changes in this room": "Sehen wenn sich der Name in diesem Raum ändert",
+ "Change the name of your active room": "Den Namen deines aktiven Raums ändern",
+ "See when the name changes in your active room": "Sehen wenn der Name sich in deinem aktiven Raum ändert",
+ "Change the avatar of this room": "Avatar von diesem Raum ändern",
+ "See when the avatar changes in this room": "Sehen wenn der Avatar sich in diesem Raum ändert",
+ "Change the avatar of your active room": "Den Avatar deines aktiven Raums ändern",
+ "See when the avatar changes in your active room": "Sehen wenn ein Avatar in deinem aktiven Raum geändert wird",
+ "Send stickers to this room as you": "Einen Sticker in diesen Raum als du senden",
+ "See when a sticker is posted in this room": "Sehe wenn ein Sticker in diesen Raum gesendet wird",
+ "Send stickers to your active room as you": "Einen Sticker als du in deinen aktiven Raum senden",
+ "See when anyone posts a sticker to your active room": "Sehen wenn jemand einen Sticker in deinen aktiven Raum sendet",
+ "with an empty state key": "mit einem leeren Zustandsschlüssel",
+ "with state key %(stateKey)s": "mit Zustandsschlüssel %(stateKey)s",
+ "Send %(eventType)s events as you in this room": "%(eventType)s-Events als du in diesem Raum senden",
+ "See %(eventType)s events posted to this room": "In diesem Raum gesendete %(eventType)s-Events anzeigen",
+ "Send %(eventType)s events as you in your active room": "%(eventType)s-Events als du in deinem aktiven Raum senden",
+ "See %(eventType)s events posted to your active room": "In deinem aktiven Raum gesendete %(eventType)s-Events anzeigen",
+ "The %(capability)s capability": "Die %(capability)s Fähigkeit",
+ "Send messages as you in this room": "Nachrichten als du in diesem Raum senden",
+ "Send messages as you in your active room": "Eine Nachricht als du in deinem aktiven Raum senden",
+ "See messages posted to this room": "In diesem Raum gesendete Nachrichten anzeigen",
+ "See messages posted to your active room": "In deinem aktiven Raum gesendete Nachrichten anzeigen",
+ "Send text messages as you in this room": "Textnachrichten als du in diesem Raum senden",
+ "Send text messages as you in your active room": "Textnachrichten als du in deinem aktiven Raum senden",
+ "See text messages posted to this room": "In diesem Raum gesendete Textnachrichten anzeigen",
+ "See text messages posted to your active room": "In deinem aktiven Raum gesendete Textnachrichten anzeigen",
+ "Send emotes as you in this room": "Emojis als du in diesem Raum senden",
+ "Send emotes as you in your active room": "Emojis als du in deinem aktiven Raum senden",
+ "See emotes posted to this room": "In diesem Raum gesendete Emojis anzeigen",
+ "See emotes posted to your active room": "In deinem aktiven Raum gesendete Emojis anzeigen",
+ "See videos posted to your active room": "In deinem aktiven Raum gesendete Videos anzeigen",
+ "See videos posted to this room": "In diesem Raum gesendete Videos anzeigen",
+ "Send images as you in this room": "Bilder als du in diesem Raum senden",
+ "Send images as you in your active room": "Bilder als du in deinem aktiven Raum senden",
+ "See images posted to this room": "In diesem Raum gesendete Bilder anzeigen",
+ "See images posted to your active room": "In deinem aktiven Raum gesendete Bilder anzeigen",
+ "Send videos as you in this room": "Videos als du in diesem Raum senden",
+ "Send videos as you in your active room": "Videos als du in deinem aktiven Raum senden",
+ "Send general files as you in this room": "Allgemeine Dateien als du in diesem Raum senden",
+ "Send general files as you in your active room": "Allgemeine Dateien als du in deinem aktiven Raum senden",
+ "See general files posted to your active room": "Allgemeine in deinem aktiven Raum gesendete Dateien anzeigen",
+ "See general files posted to this room": "Allgemeine in diesem Raum gesendete Dateien anzeigen",
+ "Send %(msgtype)s messages as you in this room": "Sende %(msgtype)s Nachrichten als du in diesem Raum",
+ "Send %(msgtype)s messages as you in your active room": "Sende %(msgtype)s Nachrichten als du in deinem aktiven Raum",
+ "See %(msgtype)s messages posted to this room": "Zeige %(msgtype)s Nachrichten, welche in diesem Raum gesendet worden sind",
+ "See %(msgtype)s messages posted to your active room": "Zeige %(msgtype)s Nachrichten, welche in deinem aktiven Raum gesendet worden sind",
+ "Don't miss a reply": "Verpasse keine Antwort",
+ "Enable desktop notifications": "Aktiviere Desktopbenachrichtigungen",
+ "Update %(brand)s": "Aktualisiere %(brand)s",
+ "New version of %(brand)s is available": "Neue Version von %(brand)s verfügbar",
+ "You ended the call": "Du hast den Anruf beendet",
+ "%(senderName)s ended the call": "%(senderName)s hat den Anruf beendet",
+ "Use Command + Enter to send a message": "Benutze Betriebssystemtaste + Enter um eine Nachricht zu senden",
+ "Use Ctrl + Enter to send a message": "Benutze Strg + Enter um eine Nachricht zu senden",
+ "Call Paused": "Anruf pausiert",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Verschlüsselte Nachrichten sicher lokal zwischenspeichern um sie in Suchergebnissen finden zu können, benötigt %(size)s um die Nachrichten von den Räumen %(rooms)s zu speichern.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Verschlüsselte Nachrichten sicher lokal zwischenspeichern um sie in Suchergebnissen finden zu können, benötigt %(size)s um die Nachrichten vom Raum %(rooms)s zu speichern.",
+ "Only the two of you are in this conversation, unless either of you invites anyone to join.": "Nur ihr zwei seid in dieser Konversation, außer einer von euch lädt jemanden neues ein.",
+ "This is the beginning of your direct message history with .": "Dies ist der Beginn deiner Direktnachrichtenhistorie mit .",
+ "Topic: %(topic)s (edit)": "Thema: %(topic)s (ändern)",
+ "Topic: %(topic)s ": "Thema: %(topic)s ",
+ "Add a topic to help people know what it is about.": "Füge ein Thema hinzu um Personen zu verdeutlichen um was es in ihm geht.",
+ "You created this room.": "Du hast diesen Raum erstellt.",
+ "%(displayName)s created this room.": "%(displayName)s erstellte diesen Raum.",
+ "Add a photo, so people can easily spot your room.": "Füge ein Foto hinzu, sodass Personen deinen Raum einfach finden können.",
+ "This is the start of .": "Dies ist der Beginn von .",
+ "Start a new chat": "Starte einen neuen Chat",
+ "Role": "Rolle",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Nachrichten hier sind Ende-zu-Ende-verschlüsselt. Verifiziere %(displayName)s im deren Profil - klicke auf deren Avatar.",
+ "Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "Nachrichten in diesem Raum sind Ende-zu-Ende-verschlüsselt. Wenn Personen beitreten, kannst du sie in ihrem Profil verifizieren, klicke hierfür auf deren Avatar.",
+ "Tell us below how you feel about %(brand)s so far.": "Erzähle uns wie %(brand)s dir soweit gefällt.",
+ "Please go into as much detail as you like, so we can track down the problem.": "Bitte nenne so viele Details wie du möchtest, sodass wir das Problem finden können.",
+ "Comment": "Kommentar",
+ "There are two ways you can provide feedback and help us improve %(brand)s.": "Es gibt zwei Wege wie du Feedback geben kannst und uns helfen kannst %(brand)s zu verbessern.",
+ "Please view existing bugs on Github first. No match? Start a new one.": "Bitte wirf einen Blick auf existierende Bugs auf Github. Keinen gefunden? Erstelle einen neuen.",
+ "PRO TIP: If you start a bug, please submit debug logs to help us track down the problem.": "PRO TIPP: Wenn du einen Bug meldest, füge bitte Debug-Logs hinzu um uns zu helfen das Problem zu finden.",
+ "Invite by email": "Via Email einladen",
+ "Start a conversation with someone using their name, email address or username (like ).": "Beginne eine Konversation mit jemanden unter Benutzung des Namens, Email-Addresse oder Benutzername (siehe ).",
+ "Invite someone using their name, email address, username (like ) or share this room.": "Lade jemanden unter Benutzung seines Namens, E-Mailaddresse oder Benutzername (siehe ) ein, oder teile diesen Raum.",
+ "Approve widget permissions": "Rechte für das Widget genehmigen",
+ "This widget would like to:": "Dieses Widget würde gerne:",
+ "Approve": "Zustimmen",
+ "Decline All": "Alles ablehnen",
+ "Go to Home View": "Zur Startseite gehen",
+ "Filter rooms and people": "Räume und Personen filtern",
+ "%(creator)s created this DM.": "%(creator)s hat diese DM erstellt.",
+ "Now, let's help you get started": "Nun, lassen Sie uns Ihnen den Einstieg erleichtern",
+ "Welcome %(name)s": "Willkommen %(name)s",
+ "Add a photo so people know it's you.": "Fügen Sie ein Foto hinzu, damit die Leute wissen, dass Sie es sind.",
+ "Great, that'll help people know it's you": "Großartig, das wird den Leuten helfen, zu wissen, dass Sie es sind",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
HTML für die Seite Ihrer Community
\n
\n Verwenden Sie die ausführliche Beschreibung, um die Community neuen Mitgliedern vorzustellen,\n oder teilen Sie einige wichtige Links links\n
\n
\n Sie können sogar Bilder mit Matrix-URLs hinzufügen \n
\n",
+ "Enter phone number": "Telefonnummer eingeben",
+ "Enter email address": "E-Mail-Adresse eingeben",
+ "Open the link in the email to continue registration.": "Öffnen Sie den Link in der E-Mail, um mit der Registrierung fortzufahren.",
+ "A confirmation email has been sent to %(emailAddress)s": "Eine Bestätigungs-E-Mail wurde an %(emailAddress)s gesendet",
+ "Use the + to make a new room or explore existing ones below": "Verwenden Sie das +, um einen neuen Raum zu erstellen oder unten einen bestehenden zu erkunden",
+ "Return to call": "Zurück zum Anruf",
+ "Fill Screen": "Bildschirm ausfüllen",
+ "Voice Call": "Sprachanruf",
+ "Video Call": "Videoanruf",
+ "Remain on your screen while running": "Bleiben Sie auf Ihrem Bildschirm während der Ausführung von",
+ "Remain on your screen when viewing another room, when running": "Bleiben Sie auf Ihrem Bildschirm, während Sie einen anderen Raum betrachten, wenn Sie ausführen",
+ "Zimbabwe": "Simbabwe",
+ "Zambia": "Sambia",
+ "Yemen": "Jemen",
+ "Western Sahara": "Westsahara",
+ "Wallis & Futuna": "Wallis und Futuna",
+ "Vietnam": "Vietnam",
+ "Venezuela": "Venezuela",
+ "Vatican City": "Vatikanstadt",
+ "Vanuatu": "Vanuatu",
+ "Uruguay": "Uruguay",
+ "United Arab Emirates": "Vereinigte Arabische Emirate",
+ "Ukraine": "Ukraine",
+ "Uganda": "Uganda",
+ "U.S. Virgin Islands": "Amerikanische Jungferninseln",
+ "Tuvalu": "Tuvalu",
+ "Turks & Caicos Islands": "Turks- und Caicosinseln",
+ "Turkmenistan": "Turkmenistan",
+ "Turkey": "Türkei",
+ "Tunisia": "Tunesien",
+ "Trinidad & Tobago": "Trinidad und Tobago",
+ "Tonga": "Tonga",
+ "Tokelau": "Tokelau",
+ "Togo": "Togo",
+ "Timor-Leste": "Timor-Leste",
+ "Thailand": "Thailand",
+ "Tanzania": "Tansania",
+ "Tajikistan": "Tadschikistan",
+ "Taiwan": "Taiwan",
+ "São Tomé & Príncipe": "São Tomé und Príncipe",
+ "Syria": "Syrien",
+ "Switzerland": "Schweiz",
+ "Sweden": "Schweden",
+ "Swaziland": "Swasiland",
+ "Svalbard & Jan Mayen": "Spitzbergen & Jan Mayen",
+ "Suriname": "Surinam",
+ "Sudan": "Sudan",
+ "St. Vincent & Grenadines": "St. Vincent und die Grenadinen",
+ "St. Pierre & Miquelon": "St. Pierre & Miquelon",
+ "St. Martin": "St. Martin",
+ "St. Lucia": "St. Lucia",
+ "St. Kitts & Nevis": "St. Kitts & Nevis",
+ "St. Helena": "St. Helena",
+ "St. Barthélemy": "St. Barthélemy",
+ "Sri Lanka": "Sri Lanka",
+ "Spain": "Spanien",
+ "South Sudan": "Südsudan",
+ "South Korea": "Südkorea",
+ "South Georgia & South Sandwich Islands": "Südgeorgien & Südliche Sandwichinseln",
+ "South Africa": "Südafrika",
+ "Somalia": "Somalia",
+ "Solomon Islands": "Salomonen",
+ "Slovenia": "Slowenien",
+ "Slovakia": "Slowakei",
+ "Sint Maarten": "St. Martin",
+ "Singapore": "Singapur",
+ "Sierra Leone": "Sierra Leone",
+ "Seychelles": "Seychellen",
+ "Serbia": "Serbien",
+ "Senegal": "Senegal",
+ "Saudi Arabia": "Saudi-Arabien",
+ "San Marino": "San Marino",
+ "Samoa": "Samoa",
+ "Réunion": "Réunion",
+ "Rwanda": "Ruanda",
+ "Russia": "Russland",
+ "Romania": "Rumänien",
+ "Qatar": "Katar",
+ "Puerto Rico": "Puerto Rico",
+ "Portugal": "Portugal",
+ "Poland": "Polen",
+ "Pitcairn Islands": "Pitcairninseln",
+ "Philippines": "Philippinen",
+ "Peru": "Peru",
+ "Paraguay": "Paraguay",
+ "Papua New Guinea": "Papua-Neuguinea",
+ "Panama": "Panama",
+ "Palestine": "Palästina",
+ "Palau": "Palau",
+ "Pakistan": "Pakistan",
+ "Oman": "Oman",
+ "Norway": "Norwegen",
+ "Northern Mariana Islands": "Nördliche Marianeninseln",
+ "North Korea": "Nordkorea",
+ "Norfolk Island": "Norfolkinsel",
+ "Niue": "Niue",
+ "Nigeria": "Nigeria",
+ "Niger": "Niger",
+ "Nicaragua": "Nicaragua",
+ "New Zealand": "Neuseeland",
+ "New Caledonia": "Neukaledonien",
+ "Netherlands": "Niederlande",
+ "Nepal": "Nepal",
+ "Nauru": "Nauru",
+ "Namibia": "Namibia",
+ "Myanmar": "Myanmar",
+ "Mozambique": "Mosambik",
+ "Morocco": "Marokko",
+ "Montserrat": "Montserrat",
+ "Montenegro": "Montenegro",
+ "Mongolia": "Mongolei",
+ "Czech Republic": "Tschechische Republik",
+ "Monaco": "Monaco",
+ "Moldova": "Moldawien",
+ "Micronesia": "Mikronesien",
+ "Mexico": "Mexiko",
+ "Mayotte": "Mayotte",
+ "Mauritius": "Mauritius",
+ "Mauritania": "Mauretanien",
+ "Martinique": "Martinique",
+ "Marshall Islands": "Marshallinseln",
+ "Malta": "Malta",
+ "Mali": "Mali",
+ "Maldives": "Malediven",
+ "Malaysia": "Malaysia",
+ "Malawi": "Malawi",
+ "Madagascar": "Madagaskar",
+ "Macedonia": "Mazedonien",
+ "Macau": "Macau",
+ "Luxembourg": "Luxemburg",
+ "Lithuania": "Litauen",
+ "Liechtenstein": "Liechtenstein",
+ "Libya": "Libyen",
+ "Liberia": "Liberia",
+ "Lesotho": "Lesotho",
+ "Lebanon": "Libanon",
+ "Latvia": "Lettland",
+ "Laos": "Laos",
+ "Kyrgyzstan": "Kirgisistan",
+ "Kuwait": "Kuwait",
+ "Kosovo": "Kosovo",
+ "Kiribati": "Kiribati",
+ "Kenya": "Kenia",
+ "Kazakhstan": "Kasachstan",
+ "Jordan": "Jordanien",
+ "Jersey": "Jersey",
+ "Japan": "Japan",
+ "Jamaica": "Jamaika",
+ "Italy": "Italien",
+ "Israel": "Israel",
+ "Isle of Man": "Insel Man",
+ "Ireland": "Irland",
+ "Iraq": "Irak",
+ "Iran": "Iran",
+ "Indonesia": "Indonesien",
+ "India": "Indien",
+ "Iceland": "Island",
+ "Hungary": "Ungarn",
+ "Hong Kong": "Hongkong",
+ "Honduras": "Honduras",
+ "Heard & McDonald Islands": "Heard & McDonald-Inseln",
+ "Haiti": "Haiti",
+ "Guyana": "Guyana",
+ "Guinea-Bissau": "Guinea-Bissau",
+ "Guinea": "Guinea",
+ "Guernsey": "Guernsey",
+ "Guatemala": "Guatemala",
+ "Guam": "Guam",
+ "Guadeloupe": "Guadeloupe",
+ "Grenada": "Grenada",
+ "Greenland": "Grönland",
+ "Greece": "Griechenland",
+ "Gibraltar": "Gibraltar",
+ "Ghana": "Ghana",
+ "Germany": "Deutschland",
+ "Georgia": "Georgien",
+ "Gambia": "Gambia",
+ "Gabon": "Gabun",
+ "French Southern Territories": "Französische Süd-Territorien",
+ "French Polynesia": "Französisch-Polynesien",
+ "French Guiana": "Französisch-Guayana",
+ "France": "Frankreich",
+ "Finland": "Finnland",
+ "Fiji": "Fidschi",
+ "Faroe Islands": "Färöer-Inseln",
+ "Falkland Islands": "Falklandinseln",
+ "Ethiopia": "Äthiopien",
+ "Estonia": "Estland",
+ "Eritrea": "Eritrea",
+ "Equatorial Guinea": "Äquatorialguinea",
+ "El Salvador": "El Salvador",
+ "Egypt": "Ägypten",
+ "Ecuador": "Ecuador",
+ "Dominican Republic": "Dominikanische Republik",
+ "Dominica": "Dominica",
+ "Djibouti": "Dschibuti",
+ "Denmark": "Dänemark",
+ "Côte d’Ivoire": "Tschechische Republik",
+ "Cyprus": "Zypern",
+ "Curaçao": "Curaçao",
+ "Cuba": "Kuba",
+ "Croatia": "Kroatien",
+ "Cook Islands": "Cookinseln",
+ "Costa Rica": "Costa Rica",
+ "Congo - Kinshasa": "Republik Kongo - Kinshasa",
+ "Congo - Brazzaville": "Republik Kongo - Brazzaville",
+ "Comoros": "Komoren",
+ "Colombia": "Kolumbien",
+ "Cocos (Keeling) Islands": "Kokosinseln (Keeling)",
+ "Christmas Island": "Weihnachtsinsel",
+ "China": "China",
+ "Chile": "Chile",
+ "Chad": "Tschad",
+ "Central African Republic": "Zentralafrikanische Republik",
+ "Cayman Islands": "Kaimaninseln",
+ "Caribbean Netherlands": "Karibische Niederlande",
+ "Cape Verde": "Kap Verde",
+ "Canada": "Kanada",
+ "Cameroon": "Kamerun",
+ "Cambodia": "Kambodscha",
+ "Burundi": "Burundi",
+ "Burkina Faso": "Burkina Faso",
+ "Bulgaria": "Bulgarien",
+ "Brunei": "Brunei",
+ "British Virgin Islands": "Britische Jungferninseln",
+ "British Indian Ocean Territory": "Britisches Territorium im Indischen Ozean",
+ "Brazil": "Brasilien",
+ "Bouvet Island": "Bouvetinsel",
+ "Botswana": "Botswana",
+ "Bosnia": "Bosnien",
+ "Bolivia": "Bolivien",
+ "Bhutan": "Bhutan",
+ "Bermuda": "Bermuda",
+ "Benin": "Benin",
+ "Belize": "Belize",
+ "Belgium": "Belgien",
+ "Belarus": "Weißrussland",
+ "Barbados": "Barbados",
+ "Bangladesh": "Bangladesch",
+ "Bahrain": "Bahrain",
+ "Bahamas": "Bahamas",
+ "Azerbaijan": "Aserbaidschan",
+ "Austria": "Österreich",
+ "Australia": "Australien",
+ "Aruba": "Aruba",
+ "Armenia": "Armenien",
+ "Argentina": "Argentinien",
+ "Antigua & Barbuda": "Antigua und Barbuda",
+ "Antarctica": "Antarktis",
+ "Anguilla": "Anguilla",
+ "Angola": "Angola",
+ "Andorra": "Andorra",
+ "American Samoa": "Amerikanisch-Samoa",
+ "Algeria": "Algerien",
+ "Albania": "Albanien",
+ "Åland Islands": "Äland-Inseln",
+ "Afghanistan": "Afghanistan",
+ "United States": "Vereinigte Staaten",
+ "United Kingdom": "Großbritannien",
+ "We call the places you where you can host your account ‘homeservers’.": "Orte, an denen du dein Benutzerkonto hosten kannst, nennen wir \"Homeserver\".",
+ "Specify a homeserver": "Gib einen Homeserver an",
+ "Render LaTeX maths in messages": "Zeige LaTeX-Matheformeln in Nachrichten an",
+ "Decide where your account is hosted": "Gib an wo dein Benutzerkonto gehostet werden soll",
+ "Already have an account? Sign in here": "Hast du schon ein Benutzerkonto? Melde dich hier an",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s oder %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "Mit %(ssoButtons)s fortfahren",
+ "That username already exists, please try another.": "Dieser Benutzername existiert schon. Bitte versuche es mit einem anderen.",
+ "New? Create account": "Neu? Erstelle ein Benutzerkonto",
+ "There was a problem communicating with the homeserver, please try again later.": "Es gab ein Problem bei der Kommunikation mit dem Homseserver. Bitte versuche es später erneut.",
+ "New here? Create an account": "Neu hier? Erstelle ein Benutzerkonto",
+ "Got an account? Sign in": "Hast du schon ein Benutzerkonto? Melde dich an",
+ "Use email to optionally be discoverable by existing contacts.": "Nutze optional eine E-Mail-Adresse, um von Nutzern gefunden werden zu können.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Nutze optional eine E-Mail-Adresse oder Telefonnummer, um von Nutzern gefunden werden zu können.",
+ "Add an email to be able to reset your password.": "Füge eine E-Mail-Adresse hinzu, um dein Passwort zurücksetzen zu können.",
+ "Forgot password?": "Passwort vergessen?",
+ "That phone number doesn't look quite right, please check and try again": "Diese Telefonummer sieht nicht ganz richtig aus. Bitte überprüfe deine Eingabe und versuche es erneut",
+ "About homeservers": "Über Homeserver",
+ "Learn more": "Mehr dazu",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Verwende einen Matrix-Homeserver deiner Wahl oder hoste deinen eigenen.",
+ "Other homeserver": "Anderer Homeserver",
+ "Sign into your homeserver": "Melde dich bei deinem Homeserver an",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org ist der größte öffentliche Homeserver der Welt und daher ein guter Ort für viele.",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Aufgepasst: Wenn du keine E-Mail-Adresse angibst und dein Passwort vergisst, kannst du den Zugriff auf deinen Account dauerhaft verlieren.",
+ "Continuing without email": "Ohne E-Mail fortfahren",
+ "Reason (optional)": "Grund (optional)",
+ "Continue with %(provider)s": "Mit %(provider)s fortfahren",
+ "Homeserver": "Heimserver",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Du kannst in den benutzerdefinierten Serveroptionen eine andere Heimserver-URL angeben, um dich bei anderen Matrixservern anzumelden.",
+ "Server Options": "Servereinstellungen",
+ "No other application is using the webcam": "Keine andere Anwendung auf die Webcam zugreift",
+ "Permission is granted to use the webcam": "Auf die Webcam zugegriffen werden darf",
+ "A microphone and webcam are plugged in and set up correctly": "Mikrofon und Webcam eingesteckt und richtig eingerichtet sind",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Der Anruf ist fehlgeschlagen weil nicht auf das Mikrofon zugegriffen werden konnte. Stelle sicher, dass das Mikrofon richtig eingesteckt und eingerichtet ist.",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "Der Anruf ist fehlgeschlagen weil nicht auf das Mikrofon oder die Webcam zugegriffen werden konnte. Stelle sicher, dass:",
+ "Unable to access webcam / microphone": "Auf Webcam / Mikrofon konnte nicht zugegriffen werden",
+ "Unable to access microphone": "Es konnte nicht auf das Mikrofon zugegriffen werden"
}
diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json
index 75877019d7..83e8846f31 100644
--- a/src/i18n/strings/en_EN.json
+++ b/src/i18n/strings/en_EN.json
@@ -46,11 +46,18 @@
"Alternatively, you can try to use the public server at turn.matrix.org, but this will not be as reliable, and it will share your IP address with that server. You can also manage this in Settings.": "Alternatively, you can try to use the public server at turn.matrix.org, but this will not be as reliable, and it will share your IP address with that server. You can also manage this in Settings.",
"Try using turn.matrix.org": "Try using turn.matrix.org",
"OK": "OK",
+ "Unable to access microphone": "Unable to access microphone",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.",
+ "Unable to access webcam / microphone": "Unable to access webcam / microphone",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Call failed because webcam or microphone could not be accessed. Check that:",
+ "A microphone and webcam are plugged in and set up correctly": "A microphone and webcam are plugged in and set up correctly",
+ "Permission is granted to use the webcam": "Permission is granted to use the webcam",
+ "No other application is using the webcam": "No other application is using the webcam",
"Unable to capture screen": "Unable to capture screen",
- "Existing Call": "Existing Call",
- "You are already in a call.": "You are already in a call.",
"VoIP is unsupported": "VoIP is unsupported",
"You cannot place VoIP calls in this browser.": "You cannot place VoIP calls in this browser.",
+ "Too Many Calls": "Too Many Calls",
+ "You've reached the maximum number of simultaneous calls.": "You've reached the maximum number of simultaneous calls.",
"You cannot place a call with yourself.": "You cannot place a call with yourself.",
"Call in Progress": "Call in Progress",
"A call is currently being placed!": "A call is currently being placed!",
@@ -399,6 +406,7 @@
"Messages": "Messages",
"Actions": "Actions",
"Advanced": "Advanced",
+ "Effects": "Effects",
"Other": "Other",
"Command error": "Command error",
"Usage": "Usage",
@@ -812,13 +820,13 @@
"Show hidden events in timeline": "Show hidden events in timeline",
"Low bandwidth mode": "Low bandwidth mode",
"Allow fallback call assist server turn.matrix.org when your homeserver does not offer one (your IP address would be shared during a call)": "Allow fallback call assist server turn.matrix.org when your homeserver does not offer one (your IP address would be shared during a call)",
- "Send read receipts for messages (requires compatible homeserver to disable)": "Send read receipts for messages (requires compatible homeserver to disable)",
"Show previews/thumbnails for images": "Show previews/thumbnails for images",
"Enable message search in encrypted rooms": "Enable message search in encrypted rooms",
"How fast should messages be downloaded.": "How fast should messages be downloaded.",
"Manually verify all remote sessions": "Manually verify all remote sessions",
"IRC display name width": "IRC display name width",
"Enable experimental, compact IRC style layout": "Enable experimental, compact IRC style layout",
+ "Show chat effects": "Show chat effects",
"Collecting app version information": "Collecting app version information",
"Collecting logs": "Collecting logs",
"Uploading logs": "Uploading logs",
@@ -837,10 +845,15 @@
"When rooms are upgraded": "When rooms are upgraded",
"My Ban List": "My Ban List",
"This is your list of users/servers you have blocked - don't leave the room!": "This is your list of users/servers you have blocked - don't leave the room!",
+ "Sends the given message with confetti": "Sends the given message with confetti",
+ "sends confetti": "sends confetti",
+ "You held the call Resume": "You held the call Resume",
+ "%(peerName)s held the call": "%(peerName)s held the call",
"Video Call": "Video Call",
"Voice Call": "Voice Call",
"Fill Screen": "Fill Screen",
"Return to call": "Return to call",
+ "%(name)s paused": "%(name)s paused",
"Unknown caller": "Unknown caller",
"Incoming voice call": "Incoming voice call",
"Incoming video call": "Incoming video call",
@@ -1129,6 +1142,8 @@
"Message layout": "Message layout",
"Compact": "Compact",
"Modern": "Modern",
+ "Hide advanced": "Hide advanced",
+ "Show advanced": "Show advanced",
"Set the name of a font installed on your system & %(brand)s will attempt to use it.": "Set the name of a font installed on your system & %(brand)s will attempt to use it.",
"Customise your appearance": "Customise your appearance",
"Appearance Settings only affect this %(brand)s session.": "Appearance Settings only affect this %(brand)s session.",
@@ -1888,6 +1903,11 @@
"This address is available to use": "This address is available to use",
"This address is already in use": "This address is already in use",
"Room directory": "Room directory",
+ "Server Options": "Server Options",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.",
+ "Join millions for free on the largest public server": "Join millions for free on the largest public server",
+ "Homeserver": "Homeserver",
+ "Continue with %(provider)s": "Continue with %(provider)s",
"Sign in with single sign-on": "Sign in with single sign-on",
"And %(count)s more...|other": "And %(count)s more...",
"Home": "Home",
@@ -1948,6 +1968,7 @@
"Removing…": "Removing…",
"Confirm Removal": "Confirm Removal",
"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.": "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.",
+ "Reason (optional)": "Reason (optional)",
"Clear all data in this session?": "Clear all data in this session?",
"Clearing all data from this session is permanent. Encrypted messages will be lost unless their keys have been backed up.": "Clearing all data from this session is permanent. Encrypted messages will be lost unless their keys have been backed up.",
"Clear all data": "Clear all data",
@@ -1982,8 +2003,6 @@
"Name": "Name",
"Topic (optional)": "Topic (optional)",
"Make this room public": "Make this room public",
- "Hide advanced": "Hide advanced",
- "Show advanced": "Show advanced",
"Block anyone not part of %(serverName)s from ever joining this room.": "Block anyone not part of %(serverName)s from ever joining this room.",
"Create Room": "Create Room",
"Sign out": "Sign out",
@@ -2106,6 +2125,10 @@
"Use this session to verify your new one, granting it access to encrypted messages:": "Use this session to verify your new one, granting it access to encrypted messages:",
"If you didn’t sign in to this session, your account may be compromised.": "If you didn’t sign in to this session, your account may be compromised.",
"This wasn't me": "This wasn't me",
+ "Doesn't look like a valid email address": "Doesn't look like a valid email address",
+ "Continuing without email": "Continuing without email",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.",
+ "Email (optional)": "Email (optional)",
"Please fill why you're reporting.": "Please fill why you're reporting.",
"Report Content to Your Homeserver Administrator": "Report Content to Your Homeserver Administrator",
"Reporting this message will send its unique 'event ID' to the administrator of your homeserver. If messages in this room are encrypted, your homeserver administrator will not be able to read the message text or view any files or images.": "Reporting this message will send its unique 'event ID' to the administrator of your homeserver. If messages in this room are encrypted, your homeserver administrator will not be able to read the message text or view any files or images.",
@@ -2139,6 +2162,16 @@
"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",
+ "Unable to validate homeserver": "Unable to validate homeserver",
+ "Invalid URL": "Invalid URL",
+ "Specify a homeserver": "Specify a homeserver",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.",
+ "Sign into your homeserver": "Sign into your homeserver",
+ "We call the places where you can host your account ‘homeservers’.": "We call the places where you can host your account ‘homeservers’.",
+ "Other homeserver": "Other homeserver",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Use your preferred Matrix homeserver if you have one, or host your own.",
+ "Learn more": "Learn more",
+ "About homeservers": "About homeservers",
"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",
@@ -2232,6 +2265,8 @@
"Warning: You should only set up key backup from a trusted computer.": "Warning: You should only set up key backup from a trusted computer.",
"Access your secure message history and set up secure messaging by entering your 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 ": "If you've forgotten your recovery key you can ",
+ "Resume": "Resume",
+ "Hold": "Hold",
"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",
@@ -2267,8 +2302,6 @@
"powered by Matrix": "powered by Matrix",
"This homeserver would like to make sure you are not a robot.": "This homeserver would like to make sure you are not a robot.",
"Country Dropdown": "Country Dropdown",
- "Custom Server Options": "Custom Server Options",
- "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.": "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.",
"Confirm your identity by entering your account password below.": "Confirm your identity by entering your account password below.",
"Password": "Password",
"Missing captcha public key in homeserver configuration. Please report this to your homeserver administrator.": "Missing captcha public key in homeserver configuration. Please report this to your homeserver administrator.",
@@ -2282,48 +2315,30 @@
"Code": "Code",
"Submit": "Submit",
"Start authentication": "Start authentication",
- "Unable to validate homeserver/identity server": "Unable to validate homeserver/identity server",
- "Enter the location of your Element Matrix Services homeserver. It may use your own domain name or be a subdomain of element.io.": "Enter the location of your Element Matrix Services homeserver. It may use your own domain name or be a subdomain of element.io.",
- "Server Name": "Server Name",
"Enter password": "Enter password",
"Nice, strong password!": "Nice, strong password!",
"Password is allowed, but unsafe": "Password is allowed, but unsafe",
"Keep going...": "Keep going...",
"Enter username": "Enter username",
"Enter email address": "Enter email address",
- "Doesn't look like a valid email address": "Doesn't look like a valid email address",
"Enter phone number": "Enter phone number",
- "Doesn't look like a valid phone number": "Doesn't look like a valid phone number",
+ "That phone number doesn't look quite right, please check and try again": "That phone number doesn't look quite right, please check and try again",
"Email": "Email",
"Username": "Username",
"Phone": "Phone",
- "Not sure of your password? Set a new one": "Not sure of your password? Set a new one",
+ "Forgot password?": "Forgot password?",
"Sign in with": "Sign in with",
"Sign in": "Sign in",
- "No identity server is configured so you cannot add an email address in order to reset your password in the future.": "No identity server is configured so you cannot add an email address in order to reset your password in the future.",
- "If you don't specify an email address, you won't be able to reset your password. Are you sure?": "If you don't specify an email address, you won't be able to reset your password. Are you sure?",
"Use an email address to recover your account": "Use an email address to recover your account",
"Enter email address (required on this homeserver)": "Enter email address (required on this homeserver)",
"Other users can invite you to rooms using your contact details": "Other users can invite you to rooms using your contact details",
"Enter phone number (required on this homeserver)": "Enter phone number (required on this homeserver)",
"Use lowercase letters, numbers, dashes and underscores only": "Use lowercase letters, numbers, dashes and underscores only",
- "Email (optional)": "Email (optional)",
"Phone (optional)": "Phone (optional)",
"Register": "Register",
- "Set an email for account recovery. Use email or phone to optionally be discoverable by existing contacts.": "Set an email for account recovery. Use email or phone to optionally be discoverable by existing contacts.",
- "Set an email for account recovery. Use email to optionally be discoverable by existing contacts.": "Set an email for account recovery. Use email to optionally be discoverable by existing contacts.",
- "Enter your custom homeserver URL What does this mean?": "Enter your custom homeserver URL What does this mean?",
- "Homeserver URL": "Homeserver URL",
- "Enter your custom identity server URL What does this mean?": "Enter your custom identity server URL What does this mean?",
- "Identity Server URL": "Identity Server URL",
- "Other servers": "Other servers",
- "Free": "Free",
- "Join millions for free on the largest public server": "Join millions for free on the largest public server",
- "Premium": "Premium",
- "Premium hosting for organisations Learn more": "Premium hosting for organisations Learn more",
- "Find other public servers or use a custom server": "Find other public servers or use a custom server",
- "Sign in to your Matrix account on %(serverName)s": "Sign in to your Matrix account on %(serverName)s",
- "Sign in to your Matrix account on ": "Sign in to your Matrix account on ",
+ "Add an email to be able to reset your password.": "Add an email to be able to reset your password.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Use email or phone to optionally be discoverable by existing contacts.",
+ "Use email to optionally be discoverable by existing contacts.": "Use email to optionally be discoverable by existing contacts.",
"Sign in with SSO": "Sign in with SSO",
"Couldn't load page": "Couldn't load page",
"You must register to use this functionality": "You must register to use this functionality",
@@ -2485,13 +2500,10 @@
"A new password must be entered.": "A new password must be entered.",
"New passwords must match each other.": "New passwords must match each other.",
"Changing your password will reset any end-to-end encryption keys on all of your sessions, making encrypted chat history unreadable. Set up Key Backup or export your room keys from another session before resetting your password.": "Changing your password will reset any end-to-end encryption keys on all of your sessions, making encrypted chat history unreadable. Set up Key Backup or export your room keys from another session before resetting your password.",
- "Your Matrix account on %(serverName)s": "Your Matrix account on %(serverName)s",
- "Your Matrix account on ": "Your Matrix account on ",
- "No identity server is configured: add one in server settings to reset your password.": "No identity server is configured: add one in server settings to reset your password.",
- "Sign in instead": "Sign in instead",
"New Password": "New Password",
"A verification email will be sent to your inbox to confirm setting your new password.": "A verification email will be sent to your inbox to confirm setting your new password.",
"Send Reset Email": "Send Reset Email",
+ "Sign in instead": "Sign in instead",
"An email has been sent to %(emailAddress)s. Once you've followed the link it contains, click below.": "An email has been sent to %(emailAddress)s. Once you've followed the link it contains, click below.",
"I have verified my email address": "I have verified my email address",
"Your password has been reset.": "Your password has been reset.",
@@ -2513,24 +2525,28 @@
"Please note you are logging into the %(hs)s server, not matrix.org.": "Please note you are logging into the %(hs)s server, not matrix.org.",
"Failed to perform homeserver discovery": "Failed to perform homeserver discovery",
"This homeserver doesn't offer any login flows which are supported by this client.": "This homeserver doesn't offer any login flows which are supported by this client.",
- "Error: Problem communicating with the given homeserver.": "Error: Problem communicating with the given homeserver.",
+ "There was a problem communicating with the homeserver, please try again later.": "There was a problem communicating with the homeserver, please try again later.",
"Can't connect to homeserver via HTTP when an HTTPS URL is in your browser bar. Either use HTTPS or enable unsafe scripts.": "Can't connect to homeserver via HTTP when an HTTPS URL is in your browser bar. Either use HTTPS or enable unsafe scripts.",
"Can't connect to homeserver - please check your connectivity, ensure your homeserver's SSL certificate is trusted, and that a browser extension is not blocking requests.": "Can't connect to homeserver - please check your connectivity, ensure your homeserver's SSL certificate is trusted, and that a browser extension is not blocking requests.",
"Syncing...": "Syncing...",
"Signing In...": "Signing In...",
"If you've joined lots of rooms, this might take a while": "If you've joined lots of rooms, this might take a while",
- "Create account": "Create account",
+ "New? Create account": "New? Create account",
"Unable to query for supported registration methods.": "Unable to query for supported registration methods.",
"Registration has been disabled on this homeserver.": "Registration has been disabled on this homeserver.",
"This server does not support authentication with a phone number.": "This server does not support authentication with a phone number.",
+ "That username already exists, please try another.": "That username already exists, please try another.",
+ "Continue with %(ssoButtons)s": "Continue with %(ssoButtons)s",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s Or %(usernamePassword)s",
+ "Already have an account? Sign in here": "Already have an account? Sign in here",
"Your new account (%(newAccountId)s) is registered, but you're already logged into a different account (%(loggedInUserId)s).": "Your new account (%(newAccountId)s) is registered, but you're already logged into a different account (%(loggedInUserId)s).",
"Continue with previous account": "Continue with previous account",
"Log in to your new account.": "Log in to your new account.",
"You can now close this window or log in to your new account.": "You can now close this window or log in to your new account.",
"Registration Successful": "Registration Successful",
- "Create your Matrix account on %(serverName)s": "Create your Matrix account on %(serverName)s",
- "Create your Matrix account on ": "Create your Matrix account on ",
- "Create your account": "Create your account",
+ "Create account": "Create account",
+ "Host account on": "Host account on",
+ "Decide where your account is hosted": "Decide where your account is hosted",
"Use Recovery Key or Passphrase": "Use Recovery Key or Passphrase",
"Use Recovery Key": "Use Recovery Key",
"Confirm your identity by verifying this login from one of your other sessions, granting it access to encrypted messages.": "Confirm your identity by verifying this login from one of your other sessions, granting it access to encrypted messages.",
diff --git a/src/i18n/strings/et.json b/src/i18n/strings/et.json
index ff95a95ffe..10f4b6f4a0 100644
--- a/src/i18n/strings/et.json
+++ b/src/i18n/strings/et.json
@@ -526,7 +526,7 @@
"%(senderName)s removed the alternative addresses %(addresses)s for this room.|one": "%(senderName)s eemaldas täiendava aadressi %(addresses)s sellelt jututoalt.",
"%(senderName)s changed the alternative addresses for this room.": "%(senderName)s muutis selle jututoa täiendavat aadressi.",
"%(senderName)s changed the main and alternative addresses for this room.": "%(senderName)s muutis selle jututoa põhiaadressi ja täiendavat aadressi.",
- "%(senderName)s changed the addresses for this room.": "%(senderName)s muutis selle jututoa aadressid.",
+ "%(senderName)s changed the addresses for this room.": "%(senderName)s muutis selle jututoa aadresse.",
"Someone": "Keegi",
"(not supported by this browser)": "(ei ole toetatud selles brauseris)",
"(could not connect media)": "(ühendus teise osapoolega ei õnnestunud)",
@@ -1016,7 +1016,7 @@
"Community %(groupId)s not found": "%(groupId)s kogukonda ei leidunud",
"This homeserver does not support communities": "See koduserver ei toeta kogukondade funktsionaalsust",
"Failed to load %(groupId)s": "%(groupId)s kogukonna laadimine ei õnnestunud",
- "Welcome to %(appName)s": "Tere tulemast %(appName)s kasutajaks",
+ "Welcome to %(appName)s": "Tere tulemast suhtlusrakenduse %(appName)s kasutajaks",
"Liberate your communication": "Vabasta oma suhtlus",
"Send a Direct Message": "Saada otsesõnum",
"Are you sure you want to leave the room '%(roomName)s'?": "Kas oled kindel, et soovid lahkuda jututoast '%(roomName)s'?",
@@ -1594,11 +1594,11 @@
"All settings": "Kõik seadistused",
"Feedback": "Tagasiside",
"Use Single Sign On to continue": "Jätkamiseks kasuta ühekordset sisselogimist",
- "Confirm adding this email address by using Single Sign On to prove your identity.": "Kinnita selle e-posti aadressi lisamine kasutades ühekordset sisselogimist oma isiku tuvastamiseks.",
+ "Confirm adding this email address by using Single Sign On to prove your identity.": "Kinnita selle e-posti aadress kasutades oma isiku tuvastamiseks ühekordset sisselogimist (Single Sign On).",
"Single Sign On": "SSO Ühekordne sisselogimine",
"Confirm adding email": "Kinnita e-posti aadressi lisamine",
"Click the button below to confirm adding this email address.": "Klõpsi järgnevat nuppu e-posti aadressi lisamise kinnitamiseks.",
- "Confirm adding this phone number by using Single Sign On to prove your identity.": "Kinnita selle telefoninumbri lisamine kasutades ühekordset sisselogimist oma isiku tuvastamiseks.",
+ "Confirm adding this phone number by using Single Sign On to prove your identity.": "Kinnita selle telefoninumbri lisamine kasutades oma isiku tuvastamiseks ühekordset sisselogimist (Single Sign On).",
"Confirm adding phone number": "Kinnita telefoninumbri lisamine",
"Click the button below to confirm adding this phone number.": "Klõpsi järgnevat nuppu telefoninumbri lisamise kinnitamiseks.",
"Add Phone Number": "Lisa telefoninumber",
@@ -2399,7 +2399,7 @@
"A connection error occurred while trying to contact the server.": "Serveriga ühenduse algatamisel tekkis viga.",
"The server is not configured to indicate what the problem is (CORS).": "Server on seadistatud varjama tegelikke veapõhjuseid (CORS).",
"No files visible in this room": "Selles jututoas pole nähtavaid faile",
- "Attach files from chat or just drag and drop them anywhere in a room.": "Faile saad manueks lisada kas vastava nupu alt vestlusest või sikutades neid jututoa aknasse.",
+ "Attach files from chat or just drag and drop them anywhere in a room.": "Faile saad manuseks lisada kas vastava nupu alt vestlusest või sikutades neid jututoa aknasse.",
"You have no visible notifications in this room.": "Jututoas pole nähtavaid teavitusi.",
"You're all caught up.": "Ei tea... kõik vist on nüüd tehtud.",
"You’re all caught up": "Ei tea... kõik vist on nüüd tehtud",
@@ -2846,5 +2846,131 @@
"Filter rooms and people": "Otsi jututubasid ja inimesi",
"Open the link in the email to continue registration.": "Registreerimisega jätkamiseks vajuta e-kirjas olevat linki.",
"A confirmation email has been sent to %(emailAddress)s": "Saatsime kinnituskirja %(emailAddress)s aadressile",
- "Start a new chat": "Alusta uut vestlust"
+ "Start a new chat": "Alusta uut vestlust",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
Sinu kogukonna lehe HTML'i näidis - see on pealkiri
\n
\n Tutvustamaks uutele liikmetele kogukonda, kasuta seda pikka kirjeldust\n või jaga olulist teavet viidetena\n
\n
\n Pildite lisaminseks võid sa isegi kasutada img-märgendit Matrix'i url'idega \n
\n",
+ "Go to Home View": "Avalehele",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Selleks, et sisu saaks otsingus kasutada, puhverda krüptitud sõnumid kohalikus seadmes turvaliselt. %(rooms)s jututoa andmete salvestamiseks kulub hetkel %(size)s.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Selleks, et sisu saaks otsingus kasutada, puhverda krüptitud sõnumid kohalikus seadmes turvaliselt. %(rooms)s jututoa andmete salvestamiseks kulub hetkel %(size)s.",
+ "This widget would like to:": "See vidin sooviks:",
+ "Approve widget permissions": "Anna vidinale õigused",
+ "Use Ctrl + Enter to send a message": "Sõnumi saatmiseks vajuta Ctrl + Enter",
+ "Decline All": "Keeldu kõigist",
+ "Approve": "Nõustu",
+ "Remain on your screen when viewing another room, when running": "Kui vaatad mõnda teist jututuba, siis jää oma ekraanivaate juurde",
+ "Remain on your screen while running": "Jää oma ekraanivaate juurde",
+ "Send %(eventType)s events as you in this room": "Saada enda nimel %(eventType)s sündmusi siia jututuppa",
+ "with state key %(stateKey)s": "olekuvõtmega %(stateKey)s",
+ "with an empty state key": "tühja olekuvõtmega",
+ "See when anyone posts a sticker to your active room": "Vaata kui keegi on saatnud kleepse aktiivsesse jututuppa",
+ "Send stickers to your active room as you": "Saada enda nimel kleepse hetkel aktiivsesse jututuppa",
+ "See when a sticker is posted in this room": "Vaata kui uus kleeps on siia jututuppa lisatud",
+ "Send stickers to this room as you": "Saada sellesse jututuppa kleepse iseendana",
+ "See when the avatar changes in your active room": "Vaata kui hetkel aktiivse jututoa tunnuspilt muutub",
+ "Change the avatar of your active room": "Muuda oma aktiivse jututoa tunnuspilti",
+ "See when the avatar changes in this room": "Vaata kui selle jututoa tunnuspilt muutub",
+ "Change the avatar of this room": "Muuda selle jututoa tunnuspilti",
+ "See when the name changes in your active room": "Vaata kui hetkel aktiivse jututoa nimi muutub",
+ "Change the name of your active room": "Muuda oma aktiivse jututoa nime",
+ "See when the name changes in this room": "Vaata kui selle jututoa nimi muutub",
+ "Change the name of this room": "Muuda selle jututoa nime",
+ "See when the topic changes in your active room": "Vaata kui hetkel aktiivse jututoa teema muutub",
+ "See when the topic changes in this room": "Vaata kui selle jututoa teema muutub",
+ "Change the topic of your active room": "Muuda oma aktiivse jututoa teemat",
+ "Change the topic of this room": "Muuda selle jututoa teemat",
+ "Change which room you're viewing": "Vaheta vaadatavat jututuba",
+ "Send stickers into your active room": "Saada kleepse hetkel aktiivsesse jututuppa",
+ "Send stickers into this room": "Saada kleepse siia jututuppa",
+ "See text messages posted to this room": "Vaata selle jututoa tekstisõnumeid",
+ "Send text messages as you in your active room": "Saada oma aktiivses jututoas enda nimel tekstisõnumeid",
+ "Send text messages as you in this room": "Saada selles jututoas oma nimel tekstisõnumeid",
+ "See messages posted to your active room": "Vaata sõnumeid oma aktiivses jututoas",
+ "See messages posted to this room": "Vaata selle jututoa sõnumeid",
+ "Send messages as you in your active room": "Saada oma aktiivses jututoas enda nimel sõnumeid",
+ "Send messages as you in this room": "Saada selles jututoas oma nimel sõnumeid",
+ "The %(capability)s capability": "%(capability)s võimekus",
+ "See %(eventType)s events posted to your active room": "Vaata oma aktiivsesse jututuppa saadetud %(eventType)s sündmusi",
+ "Send %(eventType)s events as you in your active room": "Saada oma nimel oma aktiivses jututoas %(eventType)s sündmusi",
+ "See %(eventType)s events posted to this room": "Vaata siia jututuppa saadetud %(eventType)s sündmusi",
+ "Enter phone number": "Sisesta telefoninumber",
+ "Enter email address": "Sisesta e-posti aadress",
+ "Return to call": "Pöördu tagasi kõne juurde",
+ "Fill Screen": "Täida ekraan",
+ "Voice Call": "Häälkõne",
+ "Video Call": "Videokõne",
+ "Use Command + Enter to send a message": "Sõnumi saatmiseks vajuta Command + Enter klahve",
+ "See %(msgtype)s messages posted to your active room": "Näha sinu aktiivsesse jututuppa saadetud %(msgtype)s sõnumeid",
+ "See %(msgtype)s messages posted to this room": "Näha sellesse jututuppa saadetud %(msgtype)s sõnumeid",
+ "Send %(msgtype)s messages as you in your active room": "Saata sinu nimel %(msgtype)s sõnumeid sinu aktiivsesse jututuppa",
+ "Send %(msgtype)s messages as you in this room": "Saata sinu nimel %(msgtype)s sõnumeid siia jututuppa",
+ "See general files posted to your active room": "Näha sinu aktiivsesse jututuppa lisatud muid faile",
+ "See general files posted to this room": "Näha sellesse jututuppa lisatud muid faile",
+ "Send general files as you in your active room": "Saata sinu nimel muid faile sinu aktiivsesse jututuppa",
+ "Send general files as you in this room": "Saata sinu nimel muid faile siia jututuppa",
+ "See videos posted to your active room": "Näha videosid sinu aktiivses jututoas",
+ "See videos posted to this room": "Näha siia jututuppa lisatud videosid",
+ "Send videos as you in your active room": "Saata sinu nimel videosid sinu aktiivsesse jututuppa",
+ "Send videos as you in this room": "Saata sinu nimel videosid siia jututuppa",
+ "See images posted to your active room": "Näha sinu aktiivsesse jututuppa lisatud pilte",
+ "See images posted to this room": "Näha siia jututuppa lisatud pilte",
+ "Send images as you in your active room": "Saata sinu nimel pilte sinu aktiivsesse jututuppa",
+ "Send images as you in this room": "Saata sinu nimel pilte siia jututuppa",
+ "See emotes posted to your active room": "Näha emotesid sinu aktiivses jututoas",
+ "See emotes posted to this room": "Vaata selle jututoa emotesid",
+ "Send emotes as you in your active room": "Saada oma aktiivses jututoas enda nimel emotesid",
+ "Send emotes as you in this room": "Saada selles jututoas oma nimel emotesid",
+ "See text messages posted to your active room": "Vaata tekstisõnumeid oma aktiivses jututoas",
+ "New here? Create an account": "Täitsa uus asi sinu jaoks? Loo omale kasutajakonto",
+ "Got an account? Sign in": "Sul on kasutajakonto olemas? Siis logi sisse",
+ "Render LaTeX maths in messages": "Sõnumites visualiseeri LaTeX-vormingus matemaatikat",
+ "No other application is using the webcam": "Ainsamgi muu rakendus ei kasuta veebikaamerat",
+ "Permission is granted to use the webcam": "Rakendusel õigus veebikaamerat kasutada",
+ "A microphone and webcam are plugged in and set up correctly": "Veebikaamera ja mikrofon oleks ühendatud ja seadistatud",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "Kuna veebikaamerat või mikrofoni kasutada ei saanud, siis kõne ei õnnestunud. Palun kontrolli, et:",
+ "Unable to access webcam / microphone": "Puudub ligipääs veebikaamerale ja mikrofonile",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Kuna mikrofoni kasutada ei saanud, siis kõne ei õnnestunud. Palun kontrolli, et mikrofon oleks ühendatud ja seadistatud.",
+ "Unable to access microphone": "Puudub ligipääs mikrofonile",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Serveri seadistusi muutes võid teise koduserveri aadressi sisestamisel logida sisse muudesse Matrix'i serveritesse. See võimaldab sul vestlusrakenduses Element kasutada olemasolevat kasutajakontot teises koduserveris.",
+ "Continuing without email": "Jätka ilma e-posti aadressi seadistamiseta",
+ "Continue with %(provider)s": "Jätka %(provider)s kasutamist",
+ "Homeserver": "Koduserver",
+ "Server Options": "Serveri seadistused",
+ "Host account on": "Sinu kasutajakontot teenindab",
+ "Decide where your account is hosted": "Vali kes võiks sinu kasutajakontot teenindada",
+ "Already have an account? Sign in here": "Sul juba on kasutajakonto olemas? Logi siin sisse",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s või %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "Jätkamiseks kasuta %(ssoButtons)s teenuseid",
+ "That username already exists, please try another.": "Selline kasutajanimi on juba olemas, palun vali midagi muud.",
+ "New? Create account": "Täitsa uus asi sinu jaoks? Loo omale kasutajakonto",
+ "There was a problem communicating with the homeserver, please try again later.": "Serveriühenduses tekkis viga. Palun proovi mõne aja pärast uuesti.",
+ "Use email to optionally be discoverable by existing contacts.": "Kui soovid, et teised kasutajad saaksid sind leida, siis palun lisa oma e-posti aadress.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Kui soovid, et teised kasutajad saaksid sind leida, siis palun lisa oma e-posti aadress või telefoninumber.",
+ "Add an email to be able to reset your password.": "Selleks et saaksid vajadusel oma salasõna muuta, palun lisa oma e-posti aadress.",
+ "Forgot password?": "Kas unustasid oma salasõna?",
+ "That phone number doesn't look quite right, please check and try again": "See telefoninumber ei tundu õige olema, palun kontrolli ta üle ja proovi uuesti",
+ "About homeservers": "Teave koduserverite kohta",
+ "Learn more": "Loe veel",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Kui sul on oma koduserveri eelistus olemas, siis kasuta seda. Samuti võid soovi korral oma enda koduserveri püsti panna.",
+ "Other homeserver": "Muu koduserver",
+ "We call the places you where you can host your account ‘homeservers’.": "Me nimetame „koduserveriks“ sellist serverit, mis haldab sinu kasutajakontot.",
+ "Sign into your homeserver": "Logi sisse oma koduserverisse",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org on maailma suurim avalik koduserver ja see sobib paljude jaoks.",
+ "Specify a homeserver": "Sisesta koduserver",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Lihtsalt hoiatame, et kui sa ei lisa e-posti aadressi ning unustad oma konto salasõna, siis sa võid püsivalt kaotada ligipääsu oma kontole.",
+ "Reason (optional)": "Põhjus (kui soovid lisada)",
+ "We call the places where you can host your account ‘homeservers’.": "Me nimetame „koduserveriks“ sellist serverit, mis haldab sinu kasutajakontot.",
+ "Invalid URL": "Vigane aadress",
+ "Unable to validate homeserver": "Koduserveri õigsust ei õnnestunud kontrollida",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Kuna veebikaamerat või mikrofoni kasutada ei saanud, siis kõne ei õnnestunud. Palun kontrolli, et:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Kuna mikrofoni kasutada ei saanud, siis kõne ei õnnestunud. Palun kontrolli, et mikrofon oleks ühendatud ja seadistatud.",
+ "sends confetti": "saatis serpentiine",
+ "Sends the given message with confetti": "Lisab sellele sõnumile serpentiine",
+ "Show chat effects": "Näita vestlustes efekte",
+ "Effects": "Vahvad täiendused",
+ "Hold": "Pane ootele",
+ "Resume": "Jätka",
+ "%(peerName)s held the call": "%(peerName)s pani kõne ootele",
+ "You held the call Resume": "Sa panid kõne ootele. Jätka kõnet",
+ "You've reached the maximum number of simultaneous calls.": "Oled jõudnud suurima lubatud samaaegsete kõnede arvuni.",
+ "%(name)s paused": "%(name)s peatas ajutiselt kõne",
+ "Too Many Calls": "Liiga palju kõnesid"
}
diff --git a/src/i18n/strings/fa.json b/src/i18n/strings/fa.json
index eacc029d03..b16700ead6 100644
--- a/src/i18n/strings/fa.json
+++ b/src/i18n/strings/fa.json
@@ -71,7 +71,7 @@
"Collecting logs": "درحال جمعآوری گزارشها",
"Search": "جستجو",
"(HTTP status %(httpStatus)s)": "(HTTP وضعیت %(httpStatus)s)",
- "Failed to forget room %(errCode)s": "فراموش کردن گپگاه %(errCode)s موفقیتآمیز نبود",
+ "Failed to forget room %(errCode)s": "فراموش کردن اتاق با خطا مواجه شد %(errCode)s",
"Wednesday": "چهارشنبه",
"Quote": "گفتآورد",
"Send": "ارسال",
@@ -158,5 +158,69 @@
"Whether or not you're logged in (we don't record your username)": "وارد حساب خود میشوید یا خیر (ما نام کاربری شما را ثبت نمیکنیم)",
"Click the button below to confirm adding this phone number.": "برای تائید اضافهشدن این شماره تلفن، بر روی دکمهی زیر کلیک کنید.",
"Confirm adding this phone number by using Single Sign On to prove your identity.": "برای اثبات هویت خود، اضافهشدن این شماره تلفن را با استفاده از Single Sign On تائید کنید.",
- "Failed to verify email address: make sure you clicked the link in the email": "خطا در تائید آدرس ایمیل: مطمئن شوید که بر روی لینک موجود در ایمیل کلیک کرده اید"
+ "Failed to verify email address: make sure you clicked the link in the email": "خطا در تائید آدرس ایمیل: مطمئن شوید که بر روی لینک موجود در ایمیل کلیک کرده اید",
+ "Forget room": "فراموش کردن اتاق",
+ "Filter room members": "فیلتر کردن اعضای اتاق",
+ "Failure to create room": "ایجاد اتاق با خطا مواجه شد",
+ "Failed to upload profile picture!": "آپلود عکس پروفایل با خطا مواجه شد!",
+ "Failed to unban": "رفع مسدودیت با خطا مواجه شد",
+ "Failed to set display name": "تنظیم نام نمایشی با خطا مواجه شد",
+ "Failed to send request.": "ارسال درخواست با خطا مواجه شد.",
+ "Failed to send email": "ارسال ایمیل با خطا مواجه شد",
+ "Failed to join room": "پیوستن به اتاق انجام نشد",
+ "Failed to ban user": "کاربر مسدود نشد",
+ "Error decrypting attachment": "خطا در رمزگشایی پیوست",
+ "%(senderName)s ended the call.": "%(senderName)s به تماس پایان داد.",
+ "Emoji": "شکلک",
+ "Email address": "آدرس ایمیل",
+ "Email": "ایمیل",
+ "Drop File Here": "پرونده را اینجا رها کنید",
+ "Download %(text)s": "دانلود 2%(text)s",
+ "Disinvite": "پسگرفتن دعوت",
+ "Default": "پیشفرض",
+ "Deops user with given id": "کاربر را با شناسه داده شده را از بین می برد",
+ "Decrypt %(text)s": "رمزگشایی %(text)s",
+ "Deactivate Account": "غیرفعال کردن حساب",
+ "/ddg is not a command": "/ddg یک فرمان نیست",
+ "Current password": "گذرواژه فعلی",
+ "Cryptography": "رمزنگاری",
+ "Create Room": "ایجاد اتاق",
+ "Confirm password": "تأیید گذرواژه",
+ "Commands": "فرمانها",
+ "Command error": "خطای فرمان",
+ "Click here to fix": "برای رفع مشکل اینجا کلیک کنید",
+ "%(senderDisplayName)s removed the room name.": "%(senderDisplayName)s نام اتاق را حذف کرد.",
+ "%(senderName)s changed their profile picture.": "%(senderName)s عکس پروفایل خود را تغییر داد.",
+ "Change Password": "تغییر گذواژه",
+ "Banned users": "کاربران مسدود شده",
+ "Autoplay GIFs and videos": "پخش خودکار GIF و فیلم",
+ "Attachment": "پیوست",
+ "Are you sure you want to reject the invitation?": "آیا مطمئن هستید که می خواهید دعوت را رد کنید؟",
+ "Are you sure you want to leave the room '%(roomName)s'?": "آیا مطمئن هستید که می خواهید از اتاق '2%(roomName)s' خارج شوید؟",
+ "Are you sure?": "مطمئنی؟",
+ "Anyone who knows the room's link, including guests": "هرکسی که لینک اتاق را می داند و کاربران مهمان",
+ "Anyone who knows the room's link, apart from guests": "هرکسی که لینک اتاق را میداند، غیر از کاربران مهمان",
+ "Anyone": "هر کس",
+ "An error has occurred.": "خطایی رخ داده است.",
+ "%(senderName)s answered the call.": "%(senderName)s تماس را پاسخ داد.",
+ "A new password must be entered.": "گذواژه جدید باید وارد شود.",
+ "Authentication": "احراز هویت",
+ "Always show message timestamps": "همیشه مهر زمانهای پیام را نشان بده",
+ "Advanced": "پیشرفته",
+ "Camera": "دوربین",
+ "Microphone": "میکروفون",
+ "Default Device": "دستگاه پیشفرض",
+ "No media permissions": "عدم مجوز رسانه",
+ "No Webcams detected": "هیچ وبکمی شناسایی نشد",
+ "No Microphones detected": "هیچ میکروفونی شناسایی نشد",
+ "Admin": "ادمین",
+ "Add": "افزودن",
+ "Access Token:": "توکن دسترسی:",
+ "Account": "حساب کابری",
+ "Incorrect verification code": "کد فعالسازی اشتباه است",
+ "Incorrect username and/or password.": "نام کاربری و یا گذرواژه اشتباه است.",
+ "I have verified my email address": "ایمیل خود را تأید کردم",
+ "Home": "خانه",
+ "Hangup": "قطع",
+ "For security, this session has been signed out. Please sign in again.": "برای امنیت، این نشست نامعتبر شده است. لطفاً دوباره وارد سیستم شوید."
}
diff --git a/src/i18n/strings/fi.json b/src/i18n/strings/fi.json
index 6c20ed2fae..842ee436b6 100644
--- a/src/i18n/strings/fi.json
+++ b/src/i18n/strings/fi.json
@@ -304,7 +304,7 @@
"Unable to enable Notifications": "Ilmoitusten käyttöönotto epäonnistui",
"Uploading %(filename)s and %(count)s others|other": "Ladataan %(filename)s ja %(count)s muuta",
"Upload Failed": "Lataus epäonnistui",
- "Upload file": "Lataa tiedosto",
+ "Upload file": "Lataa tiedostoja",
"Upload new:": "Lataa uusi:",
"Usage": "Käyttö",
"%(senderDisplayName)s changed the topic to \"%(topic)s\".": "%(senderDisplayName)s vaihtoi aiheeksi \"%(topic)s\".",
@@ -626,7 +626,7 @@
"Send Custom Event": "Lähetä mukautettu tapahtuma",
"Advanced notification settings": "Lisäasetukset ilmoituksille",
"Forget": "Unohda",
- "You cannot delete this image. (%(code)s)": "Et voi poistaa tätä kuvaa. (%(code)s)",
+ "You cannot delete this image. (%(code)s)": "Et voi poistaa tätä kuvaa. (%(code)s)",
"Cancel Sending": "Peruuta lähetys",
"This Room": "Tämä huone",
"Noisy": "Äänekäs",
@@ -702,7 +702,7 @@
"Unable to join network": "Verkkoon liittyminen epäonnistui",
"Sorry, your browser is not able to run %(brand)s.": "Valitettavasti %(brand)s ei toimi selaimessasi.",
"Uploaded on %(date)s by %(user)s": "Ladattu %(date)s käyttäjän %(user)s toimesta",
- "Messages in group chats": "Viestit ryhmäkeskusteluissa",
+ "Messages in group chats": "Viestit ryhmissä",
"Yesterday": "Eilen",
"Error encountered (%(errorDetail)s).": "Virhe: %(errorDetail)s.",
"Low Priority": "Matala prioriteetti",
@@ -765,7 +765,7 @@
"Show join/leave messages (invites/kicks/bans unaffected)": "Näytä liittymisten ja poistumisten viestit (ei vaikuta kutsuihin, huoneesta poistamisiin ja porttikieltoihin)",
"Show developer tools": "Näytä kehitystyökalut",
"Encrypted messages in one-to-one chats": "Salatut viestit kahdenkeskisissä keskusteluissa",
- "Encrypted messages in group chats": "Salatut viestit ryhmäkeskusteluissa",
+ "Encrypted messages in group chats": "Salatut viestit ryhmissä",
"The other party cancelled the verification.": "Toinen osapuoli perui varmennuksen.",
"Verified!": "Varmennettu!",
"You've successfully verified this user.": "Olet varmentanut tämän käyttäjän.",
@@ -798,7 +798,7 @@
"Apple": "Omena",
"Strawberry": "Mansikka",
"Corn": "Maissi",
- "Pizza": "Pizza",
+ "Pizza": "Pitsa",
"Cake": "Kakku",
"Heart": "Sydän",
"Smiley": "Hymiö",
@@ -834,7 +834,7 @@
"Pin": "Nuppineula",
"Call in Progress": "Puhelu meneillään",
"General": "Yleiset",
- "Security & Privacy": "Tietoturva ja -suoja",
+ "Security & Privacy": "Tietoturva ja yksityisyys",
"Roles & Permissions": "Roolit ja oikeudet",
"Room Name": "Huoneen nimi",
"Room Topic": "Huoneen aihe",
@@ -863,7 +863,7 @@
"You don't currently have any stickerpacks enabled": "Sinulla ei ole tarrapaketteja käytössä",
"Stickerpack": "Tarrapaketti",
"Hide Stickers": "Piilota tarrat",
- "Show Stickers": "Näytä tarrat",
+ "Show Stickers": "Tarrat",
"Profile picture": "Profiilikuva",
"Set a new account password...": "Aseta uusi salasana tilille...",
"Set a new status...": "Aseta uusi tila...",
@@ -1135,7 +1135,7 @@
"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. This action is irreversible.": "Tämä tekee tilistäsi lopullisesti käyttökelvottoman. Et voi kirjautua sisään, eikä kukaan voi rekisteröidä samaa käyttäjätunnusta. Tilisi poistuu kaikista huoneista, joihin se on liittynyt, ja tilisi tiedot poistetaan identiteettipalvelimeltasi. Tämä toimenpidettä ei voi kumota.",
"Deactivating your account does not by default cause us to forget messages you have sent. If you would like us to forget your messages, please tick the box below.": "Tilisi poistaminen käytöstä ei oletuksena saa meitä unohtamaan lähettämiäsi viestejä. Jos haluaisit meidän unohtavan viestisi, rastita alla oleva ruutu.",
"Message visibility in Matrix is similar to email. Our forgetting your messages means that messages you have sent will not be shared with any new or unregistered users, but registered users who already have access to these messages will still have access to their copy.": "Viestien näkyvyys Matrixissa on samantapainen kuin sähköpostissa. Vaikka se, että unohdamme viestisi, tarkoittaa, ettei viestejäsi jaeta enää uusille tai rekisteröitymättömille käyttäjille, käyttäjät, jotka ovat jo saaneet viestisi pystyvät lukemaan jatkossakin omaa kopiotaan viesteistäsi.",
- "Please forget all messages I have sent when my account is deactivated (Warning: this will cause future users to see an incomplete view of conversations)": "Unohda kaikki viestit, jotka olen lähettänyt, kun tilini on poistettu käytöstä (b>Varoitus: tästä seuraa, että tulevat käyttäjät näkevät epätäydellisen version keskusteluista)",
+ "Please forget all messages I have sent when my account is deactivated (Warning: this will cause future users to see an incomplete view of conversations)": "Unohda kaikki viestit, jotka olen lähettänyt, kun tilini poistetaan käytöstä (Varoitus: tämä saa tulevat käyttäjät näkemään epätäydellisen version keskusteluista)",
"Verify this user to mark them as trusted. Trusting users gives you extra peace of mind when using end-to-end encrypted messages.": "Varmenna tämä käyttäjä merkitäksesi hänet luotetuksi. Käyttäjiin luottaminen antaa sinulle ylimääräistä mielenrauhaa käyttäessäsi osapuolten välistä salausta.",
"Waiting for partner to confirm...": "Odotetaan, että toinen osapuoli varmistaa...",
"Incoming Verification Request": "Saapuva varmennuspyyntö",
@@ -1631,7 +1631,7 @@
"Discovery options will appear once you have added a phone number above.": "Etsinnän asetukset näkyvät sen jälkeen, kun olet lisännyt puhelinnumeron.",
"Failed to connect to integration manager": "Yhdistäminen integraatioiden lähteeseen epäonnistui",
"Trusted": "Luotettu",
- "Not trusted": "Epäluotettu",
+ "Not trusted": "Ei-luotettu",
"This client does not support end-to-end encryption.": "Tämä asiakasohjelma ei tue osapuolten välistä salausta.",
"Messages in this room are not end-to-end encrypted.": "Tämän huoneen viestit eivät ole salattuja.",
"Messages in this room are end-to-end encrypted.": "Tämän huoneen viestit ovat salattuja.",
@@ -1692,14 +1692,14 @@
"Cross-signing and secret storage are enabled.": "Ristivarmennus ja salavarasto on käytössä.",
"Cross-signing and secret storage are not yet set up.": "Ristivarmennusta ja salavarastoa ei ole vielä otettu käyttöön.",
"Bootstrap cross-signing and secret storage": "Ota käyttöön ristivarmennus ja salavarasto",
- "Cross-signing public keys:": "Ristivarmennuksen julkiset avaimet:",
+ "Cross-signing public keys:": "Ristiinvarmennuksen julkiset avaimet:",
"not found": "ei löydetty",
- "Cross-signing private keys:": "Ristivarmennuksen salaiset avaimet:",
+ "Cross-signing private keys:": "Ristiinvarmennuksen salaiset avaimet:",
"in secret storage": "salavarastossa",
"Secret storage public key:": "Salavaraston julkinen avain:",
"in account data": "tunnuksen tiedoissa",
"not stored": "ei tallennettu",
- "Cross-signing": "Ristivarmennus",
+ "Cross-signing": "Ristiinvarmennus",
"Backup has a valid signature from this user": "Varmuuskopiossa on kelvollinen allekirjoitus tältä käyttäjältä",
"Backup has a invalid signature from this user": "Varmuuskopiossa on epäkelpo allekirjoitus tältä käyttäjältä",
"Backup has a signature from unknown user with ID %(deviceId)s": "Varmuuskopiossa on tuntematon allekirjoitus käyttäjältä, jonka ID on %(deviceId)s",
@@ -1956,7 +1956,7 @@
"Liberate your communication": "Vapauta viestintäsi",
"Send a Direct Message": "Lähetä yksityisviesti",
"Explore Public Rooms": "Selaa julkisia huoneita",
- "Create a Group Chat": "Luo ryhmäkeskustelu",
+ "Create a Group Chat": "Luo ryhmä",
"Super": "Super",
"Cancel replying to a message": "Peruuta viestiin vastaaminen",
"Jump to room search": "Siirry huonehakuun",
@@ -2019,10 +2019,10 @@
"Verify all your sessions to ensure your account & messages are safe": "Varmenna kaikki istuntosi varmistaaksesi, että tunnuksesi ja viestisi ovat turvassa",
"Verify the new login accessing your account: %(name)s": "Varmenna uusi tunnuksellesi sisäänkirjautunut taho: %(name)s",
"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.": "Tällä hetkellä salasanan vaihtaminen nollaa kaikki osapuolten välisen salauksen avaimet kaikissa istunnoissa, tehden salatusta keskusteluhistoriasta lukukelvotonta, ellet ensin vie kaikkia huoneavaimiasi ja tuo niitä salasanan vaihtamisen jäkeen takaisin. Tulevaisuudessa tämä tulee toimimaan paremmin.",
- "Your homeserver does not support cross-signing.": "Kotipalvelimesi ei tue ristivarmennusta.",
- "Your account has a cross-signing identity in secret storage, but it is not yet trusted by this session.": "Tunnuksellasi on ristivarmennuksen identiteetti salavarastossa, mutta tämä istunto ei luota siihen.",
+ "Your homeserver does not support cross-signing.": "Kotipalvelimesi ei tue ristiinvarmennusta.",
+ "Your account has a cross-signing identity in secret storage, but it is not yet trusted by this session.": "Tunnuksellasi on ristiinvarmennuksen identiteetti salaisessa tallennustilassa, mutta tämä istunto ei vielä luota siihen.",
"Reset cross-signing and secret storage": "Nollaa ristivarmennus ja salavarasto",
- "Individually verify each session used by a user to mark it as trusted, not trusting cross-signed devices.": "Varmenna jokainen käyttäjän istunto erikseen, äläkä luota ristivarmennettuihin laitteisiin.",
+ "Individually verify each session used by a user to mark it as trusted, not trusting cross-signed devices.": "Varmenna jokainen istunto erikseen, äläkä luota ristiinvarmennettuihin laitteisiin.",
"Securely cache encrypted messages locally for them to appear in search results.": "Pidä salatut viestit turvallisessa välimuistissa, jotta ne näkyvät hakutuloksissa.",
"%(brand)s is missing some components required for securely caching encrypted messages locally. If you'd like to experiment with this feature, build a custom %(brand)s Desktop with search components added.": "%(brand)sissa ei ole joitain komponentteja, joita tarvitaan viestien turvalliseen välimuistitallennukseen. Jos haluat kokeilla tätä ominaisuutta, käännä mukautettu %(brand)s Desktop, jossa on mukana hakukomponentit.",
"This session is backing up your keys. ": "Tämä istunto varmuuskopioi avaimesi. ",
@@ -2093,9 +2093,9 @@
"Messages in this room are end-to-end encrypted. Learn more & verify this user in their user profile.": "Tämän huoneen viestit ovat salattuja osapuolten välisellä salauksella. Lue lisää ja varmenna tämä käyttäjä hänen profiilistaan.",
"Enter the name of a new server you want to explore.": "Syötä sen uuden palvelimen nimi, jota hauat tutkia.",
"%(networkName)s rooms": "Verkon %(networkName)s huoneet",
- "Destroy cross-signing keys?": "Tuhoa ristivarmennuksen avaimet?",
- "Deleting cross-signing keys is permanent. Anyone you have verified with will see security alerts. You almost certainly don't want to do this, unless you've lost every device you can cross-sign from.": "Ristivarmennuksen avainten tuhoamista ei voi kumota. Jokainen, jonka olet varmentanut, tulee näkemään turvallisuushälytyksiä. Et todennäköisesti halua tehdä tätä, ellet ole hukannut kaikkia laitteitasi, joista pystyt ristivarmentamaan.",
- "Clear cross-signing keys": "Tyhjennä ristivarmennuksen avaimet",
+ "Destroy cross-signing keys?": "Tuhoa ristiinvarmennuksen avaimet?",
+ "Deleting cross-signing keys is permanent. Anyone you have verified with will see security alerts. You almost certainly don't want to do this, unless you've lost every device you can cross-sign from.": "Ristiinvarmennuksen avainten tuhoamista ei voi kumota. Jokainen, jonka olet varmentanut, tulee näkemään turvallisuushälytyksiä. Et todennäköisesti halua tehdä tätä, ellet ole hukannut kaikkia laitteitasi, joista pystyit ristiinvarmentamaan.",
+ "Clear cross-signing keys": "Tyhjennä ristiinvarmennuksen avaimet",
"Enable end-to-end encryption": "Ota osapuolten välinen salaus käyttöön",
"Session key": "Istunnon tunnus",
"Verification Requests": "Varmennuspyynnöt",
@@ -2144,7 +2144,7 @@
"Address (optional)": "Osoite (valinnainen)",
"Light": "Vaalea",
"Dark": "Tumma",
- "Emoji picker": "Emojivalitsin",
+ "Emoji picker": "Emojit",
"No recently visited rooms": "Ei hiljattain vierailtuja huoneita",
"People": "Ihmiset",
"Sort by": "Lajittelutapa",
@@ -2245,7 +2245,7 @@
"Add widgets, bridges & bots": "Lisää sovelmia, siltoja ja botteja",
"Edit widgets, bridges & bots": "Muokkaa sovelmia, siltoja ja botteja",
"Widgets": "Sovelmat",
- "Communities v2 prototypes. Requires compatible homeserver. Highly experimental - use with caution.": "Yhteisöjen v2 prototyypit. Vaatii yhteensopivan kotipalvelimen. Erittäin kokeellinen — käytä varoen.",
+ "Communities v2 prototypes. Requires compatible homeserver. Highly experimental - use with caution.": "Yhteisöjen v2 prototyypit. Vaatii yhteensopivan kotipalvelimen. Erittäin kokeellinen – käytä varoen.",
"Change notification settings": "Muokkaa ilmoitusasetuksia",
"The person who invited you already left the room, or their server is offline.": "Henkilö, joka kutsui sinut, on jo lähtenyt huoneesta, tai hänen palvelin on pois verkosta.",
"The person who invited you already left the room.": "Henkilö, joka kutsui sinut, lähti jo huoneesta.",
@@ -2263,5 +2263,448 @@
"Answered Elsewhere": "Vastattu muualla",
"The call could not be established": "Puhelua ei voitu aloittaa",
"The other party declined the call.": "Toinen osapuoli hylkäsi puhelun.",
- "Call Declined": "Puhelu hylätty"
+ "Call Declined": "Puhelu hylätty",
+ "%(brand)s Android": "%(brand)s Android",
+ "%(brand)s iOS": "%(brand)s iOS",
+ "Starting microphone...": "Käynnistetään mikrofonia...",
+ "Starting camera...": "Käynnistetään kameraa...",
+ "Call connecting...": "Yhdistetään puhelua...",
+ "Calling...": "Soitetaan...",
+ "%(creator)s created this DM.": "%(creator)s loi tämän yksityisviestin.",
+ "You do not have permission to create rooms in this community.": "Sinulla ei ole lupaa luoda huoneita tähän yhteisöön.",
+ "Cannot create rooms in this community": "Tähän yhteisöön ei voi luoda huoneita",
+ "Welcome %(name)s": "Tervetuloa, %(name)s",
+ "Create community": "Luo yhteisö",
+ "No files visible in this room": "Tässä huoneessa ei näy tiedostoja",
+ "Take a picture": "Ota kuva",
+ "Your server isn't responding to some of your requests. Below are some of the most likely reasons.": "Palvelimesi ei vastaa joihinkin pyynnöistäsi. Alla on joitakin todennäköisimpiä syitä.",
+ "Server isn't responding": "Palvelin ei vastaa",
+ "Add comment": "Lisää kommentti",
+ "Update community": "Päivitä yhteisö",
+ "Create a room in %(communityName)s": "Luo huone yhteisöön %(communityName)s",
+ "Your server requires encryption to be enabled in private rooms.": "Palvelimesi edellyttää, että salaus on käytössä yksityisissä huoneissa.",
+ "An image will help people identify your community.": "Kuva auttaa ihmisiä tunnistamaan yhteisösi.",
+ "Add image (optional)": "Lisää kuva (valinnainen)",
+ "Enter name": "Syötä nimi",
+ "What's the name of your community or team?": "Mikä on yhteisösi tai tiimisi nimi?",
+ "Invite people to join %(communityName)s": "Kutsu ihmisiä yhteisöön %(communityName)s",
+ "Send %(count)s invites|one": "Lähetä %(count)s kutsu",
+ "Send %(count)s invites|other": "Lähetä %(count)s kutsua",
+ "Add another email": "Lisää toinen sähköposti",
+ "Click to view edits": "Napsauta nähdäksesi muokkaukset",
+ "Edited at %(date)s": "Muokattu %(date)s",
+ "Role": "Rooli",
+ "Show files": "Näytä tiedostot",
+ "%(count)s people|one": "%(count)s henkilö",
+ "%(count)s people|other": "%(count)s ihmistä",
+ "Forget Room": "Unohda huone",
+ "Show previews of messages": "Näytä viestien esikatselut",
+ "Show rooms with unread messages first": "Näytä ensimmäisenä huoneet, joissa on lukemattomia viestejä",
+ "%(count)s results|one": "%(count)s tulos",
+ "%(count)s results|other": "%(count)s tulosta",
+ "Start a new chat": "Aloita uusi keskustelu",
+ "Can't see what you’re looking for?": "Etkö löydä, mitä etsit?",
+ "This is the start of .": "Tästä alkaa .",
+ "%(displayName)s created this room.": "%(displayName)s loi tämän huoneen.",
+ "You created this room.": "Loit tämän huoneen.",
+ "Add a topic to help people know what it is about.": "Lisää aihe, jotta ihmiset tietävät mistä on kyse.",
+ "Topic: %(topic)s ": "Aihe: %(topic)s ",
+ "Topic: %(topic)s (edit)": "Aihe: %(topic)s (muokkaa)",
+ "This is the beginning of your direct message history with .": "Tästä alkaa yksityisviestihistoriasi käyttäjän kanssa.",
+ "Only the two of you are in this conversation, unless either of you invites anyone to join.": "Vain te kaksi olette tässä keskustelussa, ellei jompi kumpi kutsu muita.",
+ "Remove messages sent by others": "Poista toisten lähettämät viestit",
+ "Privacy": "Tietosuoja",
+ "not ready": "ei valmis",
+ "ready": "valmis",
+ "unexpected type": "odottamaton tyyppi",
+ "Algorithm:": "Algoritmi:",
+ "Failed to save your profile": "Profiilisi tallentaminen ei onnistunut",
+ "Your server isn't responding to some requests.": "Palvelimesi ei vastaa joihinkin pyyntöihin.",
+ "Incoming call": "Saapuva puhelu",
+ "Incoming video call": "Saapuva videopuhelu",
+ "Incoming voice call": "Saapuva äänipuhelu",
+ "Unknown caller": "Tuntematon soittaja",
+ "Enable experimental, compact IRC style layout": "Ota käyttöön kokeellinen, IRC-tyylinen asettelu",
+ "Use Ctrl + Enter to send a message": "Ctrl + Enter lähettää viestin",
+ "Use Command + Enter to send a message": "Komento + Enter lähettää viestin",
+ "%(senderName)s ended the call": "%(senderName)s lopetti puhelun",
+ "You ended the call": "Lopetit puhelun",
+ "New version of %(brand)s is available": "%(brand)s-sovelluksesta on saatavilla uusi versio",
+ "Update %(brand)s": "Päivitä %(brand)s",
+ "Enable desktop notifications": "Ota työpöytäilmoitukset käyttöön",
+ "Takes the call in the current room off hold": "Ottaa nykyisen huoneen puhelun pois pidosta",
+ "Places the call in the current room on hold": "Asettaa nykyisen huoneen puhelun pitoon",
+ "Away": "Poissa",
+ "A confirmation email has been sent to %(emailAddress)s": "Vahvistussähköposti on lähetetty osoitteeseen %(emailAddress)s",
+ "Open the link in the email to continue registration.": "Jatka rekisteröitymistä avaamalla sähköpostissa oleva linkki.",
+ "Enter email address": "Syötä sähköpostiosoite",
+ "Enter phone number": "Syötä puhelinnumero",
+ "Now, let's help you get started": "Autetaanpa sinut alkuun",
+ "delete the address.": "poista osoite.",
+ "Filter rooms and people": "Suodata huoneita ja ihmisiä",
+ "Go to Home View": "Siirry kotinäkymään",
+ "Community and user menu": "Yhteisö- ja käyttäjävalikko",
+ "Decline All": "Kieltäydy kaikista",
+ "Approve": "Hyväksy",
+ "Your area is experiencing difficulties connecting to the internet.": "Alueellasi on ongelmia internet-yhteyksissä.",
+ "The server is offline.": "Palvelin ei ole verkossa.",
+ "Your firewall or anti-virus is blocking the request.": "Palomuurisi tai virustentorjuntaohjelmasi estää pyynnön.",
+ "The server (%(serverName)s) took too long to respond.": "Palvelin (%(serverName)s) ei vastannut ajoissa.",
+ "Feedback sent": "Palaute lähetetty",
+ "There was an error updating your community. The server is unable to process your request.": "Yhteisösi päivittämisessä tapahtui virhe. Palvelin ei voi käsitellä pyyntöäsi.",
+ "You can change this later if needed.": "Voit tarvittaessa vaihtaa tämän myöhemmin.",
+ "There was an error creating your community. The name may be taken or the server is unable to process your request.": "Yhteisösi luomisessa tapahtui virhe. Nimi saattaa olla varattu tai palvelin ei voi käsitellä pyyntöäsi.",
+ "Download logs": "Lataa lokit",
+ "Preparing to download logs": "Valmistellaan lokien lataamista",
+ "About": "Tietoa",
+ "Unpin": "Poista kiinnitys",
+ "Customise your appearance": "Mukauta ulkoasuasi",
+ "Appearance Settings only affect this %(brand)s session.": "Ulkoasuasetukset vaikuttavat vain tähän %(brand)s-istuntoon.",
+ "Set the name of a font installed on your system & %(brand)s will attempt to use it.": "Aseta käyttöjärjestelmääsi asennetun fontin nimi, niin %(brand)s pyrkii käyttämään sitä.",
+ "The operation could not be completed": "Toimintoa ei voitu tehdä loppuun asti",
+ "There are advanced notifications which are not shown here.": "On edistyneitä ilmoituksia, joita ei näytetä tässä.",
+ "Return to call": "Palaa puheluun",
+ "Voice Call": "Äänipuhelu",
+ "Video Call": "Videopuhelu",
+ "Send stickers to your active room as you": "Lähetä aktiiviseen huoneeseesi tarroja itsenäsi",
+ "Send stickers to this room as you": "Lähetä tähän huoneeseen tarroja itsenäsi",
+ "Change the avatar of your active room": "Vaihda aktiivisen huoneesi kuva",
+ "Change the avatar of this room": "Vaihda huoneen kuva",
+ "Change the name of your active room": "Muuta aktiivisen huoneesi nimeä",
+ "Change the name of this room": "Muuta tämän huoneen nimeä",
+ "Change the topic of your active room": "Muuta aktiivisen huoneesi aihetta",
+ "Change the topic of this room": "Muuta huoneen aihetta",
+ "Change which room you're viewing": "Vaihda näytettävää huonetta",
+ "Send stickers into your active room": "Lähetä tarroja aktiiviseen huoneeseesi",
+ "Send stickers into this room": "Lähetä tarroja tähän huoneeseen",
+ "Comoros": "Komorit",
+ "Colombia": "Kolumbia",
+ "Cocos (Keeling) Islands": "Kookossaaret",
+ "Christmas Island": "Joulusaari",
+ "China": "Kiina",
+ "Chile": "Chile",
+ "Chad": "Tšad",
+ "Central African Republic": "Keski-Afrikan tasavalta",
+ "Cayman Islands": "Caymansaaret",
+ "Caribbean Netherlands": "Alankomaiden Karibia",
+ "Cape Verde": "Kap Verde",
+ "Canada": "Kanada",
+ "Cameroon": "Kamerun",
+ "Cambodia": "Kambodža",
+ "Burundi": "Burundi",
+ "Burkina Faso": "Burkina Faso",
+ "Bulgaria": "Bulgaria",
+ "Brunei": "Brunei",
+ "British Virgin Islands": "Brittiläiset Neitsytsaaret",
+ "British Indian Ocean Territory": "Brittiläinen Intian valtameren alue",
+ "Brazil": "Brasilia",
+ "Bouvet Island": "Bouvet’nsaari",
+ "Botswana": "Botswana",
+ "Bosnia": "Bosnia ja Hertsegovina",
+ "Bolivia": "Bolivia",
+ "Bhutan": "Bhutan",
+ "Bermuda": "Bermuda",
+ "Benin": "Benin",
+ "Belize": "Belize",
+ "Belgium": "Belgia",
+ "Belarus": "Valko-Venäjä",
+ "Barbados": "Barbados",
+ "Bangladesh": "Bangladesh",
+ "Bahrain": "Bahrain",
+ "Bahamas": "Bahama",
+ "Azerbaijan": "Azerbaidžan",
+ "Austria": "Itävalta",
+ "Australia": "Australia",
+ "Aruba": "Aruba",
+ "Armenia": "Armenia",
+ "Argentina": "Argentiina",
+ "Antigua & Barbuda": "Antigua ja Barbuda",
+ "Antarctica": "Antarktis",
+ "Anguilla": "Anguilla",
+ "Angola": "Angola",
+ "Andorra": "Andorra",
+ "American Samoa": "Amerikan Samoa",
+ "Algeria": "Algeria",
+ "Albania": "Albania",
+ "Åland Islands": "Ahvenanmaa",
+ "Afghanistan": "Afganistan",
+ "United States": "Yhdysvallat",
+ "United Kingdom": "Iso-Britannia",
+ "No other application is using the webcam": "Mikään muu sovellus ei käytä kameraa",
+ "Permission is granted to use the webcam": "Lupa käyttää kameraa myönnetty",
+ "A microphone and webcam are plugged in and set up correctly": "Mikrofoni ja kamera on kytketty ja asennettu oiken",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Puhelu epäonnistui, koska kameraa tai mikrofonia ei voitu käyttää. Tarkista että:",
+ "Unable to access webcam / microphone": "Kameraa / mikrofonia ei voi käyttää",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Puhelu epäonnistui, koska mikrofonia ei voitu käyttää. Tarkista, että mikrofoni on kytketty ja asennettu oikein.",
+ "Unable to access microphone": "Mikrofonia ei voi käyttää",
+ "El Salvador": "El Salvador",
+ "Egypt": "Egypti",
+ "Ecuador": "Ecuador",
+ "Dominican Republic": "Dominikaaninen tasavalta",
+ "Dominica": "Dominica",
+ "Djibouti": "Djibouti",
+ "Denmark": "Tanska",
+ "Côte d’Ivoire": "Norsunluurannikko",
+ "Czech Republic": "Tšekki",
+ "Cyprus": "Kypros",
+ "Curaçao": "Curaçao",
+ "Cuba": "Kuuba",
+ "Croatia": "Kroatia",
+ "Costa Rica": "Costa Rica",
+ "Cook Islands": "Cookinsaaret",
+ "Congo - Kinshasa": "Kongo-Kinshasa",
+ "Congo - Brazzaville": "Kongo-Brazzavile",
+ "Micronesia": "Mikronesia",
+ "Mexico": "Meksiko",
+ "Mayotte": "Mayotte",
+ "Mauritius": "Mauritius",
+ "Mauritania": "Mauritania",
+ "Martinique": "Martinique",
+ "Marshall Islands": "Marshallinsaaret",
+ "Malta": "Malta",
+ "Mali": "Mali",
+ "Maldives": "Malediivit",
+ "Malaysia": "Malesia",
+ "Malawi": "Malawi",
+ "Madagascar": "Madagaskar",
+ "Macedonia": "Pohjois-Makedonia",
+ "Macau": "Macao",
+ "Luxembourg": "Luxemburg",
+ "Lithuania": "Liettua",
+ "Liechtenstein": "Liechtenstein",
+ "Libya": "Libya",
+ "Liberia": "Liberia",
+ "Lesotho": "Lesotho",
+ "Lebanon": "Libanon",
+ "Latvia": "Latvia",
+ "Laos": "Laos",
+ "Kyrgyzstan": "Kirgisia",
+ "Kuwait": "Kuwait",
+ "Kosovo": "Kosovo",
+ "Kiribati": "Kiribati",
+ "Kenya": "Kenia",
+ "Kazakhstan": "Kazakstan",
+ "Jordan": "Jordania",
+ "Jersey": "Jersey",
+ "Japan": "Japani",
+ "Jamaica": "Jamaika",
+ "Italy": "Italia",
+ "Israel": "Israel",
+ "Isle of Man": "Mansaari",
+ "Ireland": "Irlanti",
+ "Iraq": "Irak",
+ "Iran": "Iran",
+ "Indonesia": "Indonesia",
+ "India": "Intia",
+ "Iceland": "Islanti",
+ "Hungary": "Unkari",
+ "Hong Kong": "Hongkong",
+ "Honduras": "Honduras",
+ "Heard & McDonald Islands": "Heard ja McDonaldinsaaret",
+ "Haiti": "Haiti",
+ "Guyana": "Guyana",
+ "Guinea-Bissau": "Guinea-Bissau",
+ "Guinea": "Guinea",
+ "Guernsey": "Guernsey",
+ "Guatemala": "Guatemala",
+ "Guam": "Guam",
+ "Guadeloupe": "Guadeloupe",
+ "Grenada": "Grenada",
+ "Greenland": "Grönlanti",
+ "Greece": "Kreikka",
+ "Gibraltar": "Gibraltar",
+ "Ghana": "Ghana",
+ "Germany": "Saksa",
+ "Georgia": "Georgia",
+ "Gambia": "Gambia",
+ "Gabon": "Gabon",
+ "French Southern Territories": "Ranskan eteläiset ja antarktiset alueet",
+ "French Polynesia": "Ranskan Polynesia",
+ "French Guiana": "Ranskan Guayana",
+ "France": "Ranska",
+ "Finland": "Suomi",
+ "Fiji": "Fidži",
+ "Faroe Islands": "Färsaaret",
+ "Falkland Islands": "Falklandinsaaret",
+ "Ethiopia": "Etiopia",
+ "Estonia": "Viro",
+ "Eritrea": "Eritrea",
+ "Equatorial Guinea": "Päiväntasaajan Guinea",
+ "You've reached the maximum number of simultaneous calls.": "Saavutit samanaikaisten puheluiden enimmäismäärän.",
+ "Too Many Calls": "Liian monta puhelua",
+ "Moldova": "Moldova",
+ "You’re all caught up": "Olet ajan tasalla",
+ "Room settings": "Huoneen asetukset",
+ "or another cross-signing capable Matrix client": "tai muu ristiinvarmentava Matrix-asiakas",
+ "a device cross-signing signature": "laitteen ristiinvarmennuksen allekirjoitus",
+ "a new cross-signing key signature": "Uusi ristiinvarmennuksen allekirjoitus",
+ "Cross-signing is not set up.": "Ristiinvarmennusta ei ole asennettu.",
+ "Cross-signing is ready for use.": "Ristiinvarmennus on käyttövalmis.",
+ "Back up your encryption keys with your account data in case you lose access to your sessions. Your keys will be secured with a unique Recovery Key.": "Varmuuskopioi tilitietojesi salausavaimet, istuntojen menettämisen varalta. Avaimet varmistetaan ainutlaatuisella Palatusavaimella.",
+ "well formed": "hyvin muotoiltu",
+ "Backup version:": "Varmuuskopiointiversio:",
+ "Backup key stored:": "Varmuuskopioavain tallennettu:",
+ "Backup key cached:": "Välimuistissa oleva varmuuskopioavain:",
+ "Secret storage:": "Salainen tallennus:",
+ "Hey you. You're the best!": "Hei siellä, olet paras!",
+ "Room ID or address of ban list": "Huonetunnus tai -osoite on estolistalla",
+ "Secure Backup": "Turvallinen varmuuskopio",
+ "Add a photo, so people can easily spot your room.": "Lisää kuva, jotta ihmiset voivat helpommin huomata huoneesi.",
+ "Hide Widgets": "Piilota sovelmat",
+ "Show Widgets": "Näytä sovelmat",
+ "Explore community rooms": "Selaa yhteisön huoneita",
+ "Explore public rooms": "Selaa julkisia huoneita",
+ "Custom Tag": "Mukautettu tunniste",
+ "Explore all public rooms": "Selaa julkisia huoneita",
+ "List options": "Lajittele",
+ "Activity": "Aktiivisuus",
+ "A-Z": "A-Ö",
+ "Server Options": "Palvelimen asetukset",
+ "Information": "Tiedot",
+ "Effects": "Tehosteet",
+ "Zimbabwe": "Zimbabwe",
+ "Zambia": "Sambia",
+ "Yemen": "Jemen",
+ "Western Sahara": "Länsi-Sahara",
+ "Wallis & Futuna": "Wallis ja Futuna",
+ "Vietnam": "Vietnam",
+ "Venezuela": "Venezuela",
+ "Vatican City": "Vatikaani",
+ "Vanuatu": "Vanuatu",
+ "Uzbekistan": "Uzbekistan",
+ "Uruguay": "Uruguay",
+ "United Arab Emirates": "Yhdistyneet arabiemiirikunnat",
+ "Ukraine": "Ukraina",
+ "Uganda": "Uganda",
+ "U.S. Virgin Islands": "Yhdysvaltain Neitsytsaaret",
+ "Tuvalu": "Tuvalu",
+ "Turks & Caicos Islands": "Turks- ja Caicossaaret",
+ "Turkmenistan": "Turkmenistan",
+ "Turkey": "Turkki",
+ "Tunisia": "Tunisia",
+ "Trinidad & Tobago": "Trinidad ja Tobago",
+ "Tonga": "Tonga",
+ "Tokelau": "Tokelau",
+ "Togo": "Togo",
+ "Timor-Leste": "Itä-Timor",
+ "Thailand": "Thaimaa",
+ "Tanzania": "Tansania",
+ "Tajikistan": "Tadžikistan",
+ "Taiwan": "Taiwan",
+ "São Tomé & Príncipe": "São Tomé ja Príncipe",
+ "Syria": "Syyria",
+ "Switzerland": "Sveitsi",
+ "Sweden": "Ruotsi",
+ "Swaziland": "Swazimaa",
+ "Svalbard & Jan Mayen": "Huippuvuoret ja Jan Mayen",
+ "Suriname": "Suriname",
+ "Sudan": "Sudan",
+ "St. Vincent & Grenadines": "Saint Vincent ja Grenadiinit",
+ "St. Pierre & Miquelon": "Saint-Pierre ja Miquelon",
+ "St. Martin": "Saint-Martin",
+ "St. Lucia": "Saint Lucia",
+ "St. Kitts & Nevis": "Saint Kitts ja Nevis",
+ "St. Helena": "Saint Helena",
+ "St. Barthélemy": "Saint-Barthélemy",
+ "Sri Lanka": "Sri Lanka",
+ "Spain": "Espanja",
+ "South Sudan": "Etelä-Sudan",
+ "South Korea": "Etelä-Korea",
+ "South Georgia & South Sandwich Islands": "Etelä-Georgia ja Eteläiset Sandwichsaaret",
+ "South Africa": "Etelä-Afrikka",
+ "Somalia": "Somalia",
+ "Solomon Islands": "Salomonsaaret",
+ "Slovenia": "Slovenia",
+ "Slovakia": "Slovakia",
+ "Sint Maarten": "Sint Maarten",
+ "Singapore": "Singapore",
+ "Sierra Leone": "Sierra Leone",
+ "Seychelles": "Seychellit",
+ "Serbia": "Serbia",
+ "Senegal": "Senegal",
+ "Saudi Arabia": "Saudi-Arabia",
+ "San Marino": "San Marino",
+ "Samoa": "Samoa",
+ "Réunion": "Réunion",
+ "Rwanda": "Ruanda",
+ "Russia": "Venäjä",
+ "Romania": "Romania",
+ "Qatar": "Qatar",
+ "Puerto Rico": "Puerto Rico",
+ "Portugal": "Portugali",
+ "Poland": "Puola",
+ "Pitcairn Islands": "Pitcairnsaaret",
+ "Philippines": "Filippiinit",
+ "Peru": "Peru",
+ "Paraguay": "Paraguay",
+ "Papua New Guinea": "Papua-Uusi-Guinea",
+ "Panama": "Panama",
+ "Palestine": "Palestiina",
+ "Palau": "Palau",
+ "Pakistan": "Pakistan",
+ "Oman": "Oman",
+ "Norway": "Norja",
+ "Northern Mariana Islands": "Pohjois-Mariaanit",
+ "North Korea": "Pohjois-Korea",
+ "Norfolk Island": "Norfolkinsaari",
+ "Niue": "Niue",
+ "Nigeria": "Nigeria",
+ "Niger": "Niger",
+ "Nicaragua": "Nicaragua",
+ "New Zealand": "Uusi-Seelanti",
+ "New Caledonia": "Uusi-Kaledonia",
+ "Netherlands": "Alankomaat",
+ "Nepal": "Nepal",
+ "Nauru": "Nauru",
+ "Namibia": "Namibia",
+ "Myanmar": "Myanmar",
+ "Mozambique": "Mosambik",
+ "Morocco": "Marokko",
+ "Montserrat": "Montserrat",
+ "Montenegro": "Montenegro",
+ "Mongolia": "Mongolia",
+ "Monaco": "Monaco",
+ "Room Info": "Huoneen tiedot",
+ "not found in storage": "ei löytynyt muistista",
+ "Sign into your homeserver": "Kirjaudu sisään kotipalvelimellesi",
+ "About homeservers": "Tietoa kotipalvelimista",
+ "Not encrypted": "Ei salattu",
+ "You have no visible notifications in this room.": "Olet nähnyt tämän huoneen kaikki ilmoitukset.",
+ "Session verified": "Istunto vahvistettu",
+ "Verify this login": "Vahvista tämä sisäänkirjautuminen",
+ "User settings": "Käyttäjäasetukset",
+ "Community settings": "Yhteisön asetukset",
+ "Got an account? Sign in": "Sinulla on jo tili? Kirjaudu sisään",
+ "New here? Create an account": "Uusi täällä? Luo tili",
+ "Failed to find the general chat for this community": "Tämän yhteisön yleisen keskustelun löytäminen epäonnistui",
+ "Explore rooms in %(communityName)s": "Tutki %(communityName)s -yhteisön huoneita",
+ "Delete the room address %(alias)s and remove %(name)s from the directory?": "Poistetaanko huoneosoite %(alias)s ja %(name)s hakemisto?",
+ "Self-verification request": "Itsevarmennuspyyntö",
+ "Add a photo so people know it's you.": "Lisää kuva, jotta ihmiset tietävät, että se olet sinä.",
+ "Great, that'll help people know it's you": "Hienoa, tämä auttaa ihmisiä tietämään, että se olet sinä",
+ "Attach files from chat or just drag and drop them anywhere in a room.": "Liitä tiedostot keskustelusta tai vedä ja pudota ne mihin tahansa huoneeseen.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Käytä sähköpostiosoitetta tai puhelinnumeroa, jos haluat olla löydettävissä nykyisille yhteystiedoille.",
+ "Use email to optionally be discoverable by existing contacts.": "Käytä sähköpostiosoitetta, jos haluat olla löydettävissä nykyisille yhteystiedoille.",
+ "Add an email to be able to reset your password.": "Lisää sähköpostiosoite, jotta voit nollata salasanasi.",
+ "Forgot password?": "Unohtuiko salasana?",
+ "That phone number doesn't look quite right, please check and try again": "Tämä puhelinnumero ei näytä oikealta, tarkista se ja yritä uudelleen",
+ "Move right": "Siirry oikealle",
+ "Move left": "Siirry vasemmalle",
+ "Revoke permissions": "Peruuta käyttöoikeudet",
+ "Warning: Your personal data (including encryption keys) is still stored in this session. Clear it if you're finished using this session, or want to sign in to another account.": "Varoitus: Henkilökohtaiset tietosi (mukaan lukien salausavaimet) tallennetaan edelleen tähän istuntoon. Poista ne jos lopetat istunnon käytön, tai haluat kirjautua toiselle tilille.",
+ "You're all caught up.": "Olet ajan tasalla.",
+ "Unable to validate homeserver": "Kotipalvelimen vahvistus epäonnistui",
+ "This version of %(brand)s does not support searching encrypted messages": "Tämä %(brand)s-versio ei tue salattujen viestien hakua",
+ "This version of %(brand)s does not support viewing some encrypted files": "Tämä %(brand)s-versio ei tue joidenkin salattujen tiedostojen katselua",
+ "Use the Desktop app to see all encrypted files": "Voit tarkastella kaikkia salattuja tiedostoja työpöytäsovelluksella",
+ "Use the Desktop app to search encrypted messages": "Käytä salattuja viestejä työpöytäsovelluksella",
+ "Ignored attempt to disable encryption": "Ohitettu yritys poistaa salaus käytöstä",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Tässä olevat viestit on päästä-päähän -salattu. Vahvista %(displayName)s heidät - napauta heidän profiilikuvia.",
+ "Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "Tämän huoneen viestit on päästä-päähän -salattu. Kun ihmisiä liittyy, voit vahvistaa heidät heidän profiilista, napauta vain heidän profiilikuvaa.",
+ "Unpin a widget to view it in this panel": "Irrota sovelma, jotta voit tarkastella sitä tässä paneelissa",
+ "You can only pin up to %(count)s widgets|other": "Voit kiinnittää enintään %(count)s sovelmaa",
+ "Favourited": "Suositut",
+ "Use the + to make a new room or explore existing ones below": "Luo uusi huone tai tutustu alla oleviin + -painikeella",
+ "%(senderDisplayName)s set the server ACLs for this room.": "%(senderDisplayName)s asetti palvelimen pääsyhallintaluetteloon tämän huoneen.",
+ "%(senderDisplayName)s changed the server ACLs for this room.": "%(senderDisplayName)s muutti palvelimen pääsyhallintaluetteloa tälle huoneelle."
}
diff --git a/src/i18n/strings/fr.json b/src/i18n/strings/fr.json
index 29af11cb58..efae1f7b26 100644
--- a/src/i18n/strings/fr.json
+++ b/src/i18n/strings/fr.json
@@ -2503,7 +2503,7 @@
"No files visible in this room": "Aucun fichier visible dans ce salon",
"Enter the location of your Element Matrix Services homeserver. It may use your own domain name or be a subdomain of element.io.": "Entrez l'emplacement de votre serveur d'accueil Element Matrix Services. Cela peut utiliser votre propre nom de domaine ou être un sous-domaine de element.io.",
"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.": "Vous pouvez utiliser l'option de serveur personnalisé pour vous connecter à d'autres serveurs Matrix en spécifiant une URL de serveur d'accueil différente. Cela vous permet d'utiliser %(brand)s avec un compte Matrix existant sur un serveur d'accueil différent.",
- "Away": "Tout droit",
+ "Away": "Absent",
"Move right": "Aller à droite",
"Move left": "Aller à gauche",
"Revoke permissions": "Révoquer les permissions",
diff --git a/src/i18n/strings/gl.json b/src/i18n/strings/gl.json
index 4f9a6f34b4..cc3575dbb3 100644
--- a/src/i18n/strings/gl.json
+++ b/src/i18n/strings/gl.json
@@ -1580,7 +1580,7 @@
"Dark": "Escuro",
"Customise your appearance": "Personaliza o aspecto",
"Appearance Settings only affect this %(brand)s session.": "Os axustes da aparencia só lle afectan a esta sesión %(brand)s.",
- "Key share requests are sent to your other sessions automatically. If you rejected or dismissed the key share request on your other sessions, click here to request the keys for this session again.": "As solicitudes de compartir Chave envíanse ás outras túas sesións abertas. Se rexeitaches ou obviaches a solicitude nas outras sesións, preme aquí para voltar a facer a solicitude.",
+ "Key share requests are sent to your other sessions automatically. If you rejected or dismissed the key share request on your other sessions, click here to request the keys for this session again.": "As solicitudes de compartir Chave envíanse ás outras túas sesións abertas. Se rexeitaches ou obviaches a solicitude nas outras sesións, preme aquí para volver a facer a solicitude.",
"If your other sessions do not have the key for this message you will not be able to decrypt them.": "Se as túas outras sesións non teñen a chave para esta mensaxe non poderás descifrala.",
"Re-request encryption keys from your other sessions.": "Volta a solicitar chaves de cifrado desde as outras sesións.",
"This message cannot be decrypted": "Esta mensaxe non pode descifrarse",
@@ -1713,7 +1713,7 @@
"Remove recent messages": "Eliminar mensaxes recentes",
"%(role)s in %(roomName)s": "%(role)s en %(roomName)s",
"Deactivate user?": "¿Desactivar usuaria?",
- "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?": "Ao desactivar esta usuaria ficará desconectada e non poderá voltar a conectar. Ademáis deixará todas as salas nas que estivese. Esta acción non ten volta, ¿desexas desactivar esta usuaria?",
+ "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?": "Ao desactivar esta usuaria ficará desconectada e non poderá volver a conectar. Ademáis deixará todas as salas nas que estivese. Esta acción non ten volta, ¿desexas desactivar esta usuaria?",
"Deactivate user": "Desactivar usuaria",
"Failed to deactivate user": "Fallo ao desactivar a usuaria",
"This client does not support end-to-end encryption.": "Este cliente non soporta o cifrado extremo-a-extremo.",
@@ -1864,8 +1864,8 @@
"Hide advanced": "Ocultar Avanzado",
"Show advanced": "Mostrar Avanzado",
"Block users on other matrix homeservers from joining this room (This setting cannot be changed later!)": "Evitar que usuarias de outros servidores matrix se unan a esta sala (Este axuste non se pode cambiar máis tarde!)",
- "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 o historial da conversa, debes exportar as chaves da sala antes de desconectarte. Necesitarás voltar á nova versión de %(brand)s para facer esto",
- "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.": "Xa utilizaches unha versión máis nova de %(brand)s nesta sesión. Para usar esta versión novamente con cifrado extremo-a-extremo tes que desconectarte e voltar a conectar.",
+ "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 o historial da conversa, debes exportar as chaves da sala antes de desconectarte. Necesitarás volver á nova versión de %(brand)s para facer esto",
+ "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.": "Xa utilizaches unha versión máis nova de %(brand)s nesta sesión. Para usar esta versión novamente con cifrado extremo-a-extremo tes que desconectarte e volver a conectar.",
"Incompatible Database": "Base de datos non compatible",
"Continue With Encryption Disabled": "Continuar con Cifrado Desactivado",
"Are you sure you want to deactivate your account? This is irreversible.": "¿Tes a certeza de querer desactivar a túa conta? Esto é irreversible.",
@@ -1918,7 +1918,7 @@
"The authenticity of this encrypted message can't be guaranteed on this device.": "A autenticidade desta mensaxe cifrada non está garantida neste dispositivo.",
"Signature upload success": "Subeuse correctamente a sinatura",
"Signature upload failed": "Fallou a subida da sinatura",
- "You've previously used %(brand)s on %(host)s with lazy loading of members enabled. In this version lazy loading is disabled. As the local cache is not compatible between these two settings, %(brand)s needs to resync your account.": "Anteriormente utilizaches %(brand)s en %(host)s con carga preguiceira de membros. Nesta versión a carga preguiceira está desactivada. Como a caché local non é compatible entre as dúas configuracións, %(brand)s precisa voltar a sincronizar a conta.",
+ "You've previously used %(brand)s on %(host)s with lazy loading of members enabled. In this version lazy loading is disabled. As the local cache is not compatible between these two settings, %(brand)s needs to resync your account.": "Anteriormente utilizaches %(brand)s en %(host)s con carga preguiceira de membros. Nesta versión a carga preguiceira está desactivada. Como a caché local non é compatible entre as dúas configuracións, %(brand)s precisa volver a sincronizar a conta.",
"Incompatible local cache": "Caché local incompatible",
"Clear cache and resync": "Baleirar caché e sincronizar",
"%(brand)s now uses 3-5x less memory, by only loading information about other users when needed. Please wait whilst we resynchronise with the server!": "%(brand)s utiliza agora entre 3 e 5 veces menos memoria, cargando só información sobre as usuarias cando é preciso. Agarda mentras se sincroniza co servidor!",
@@ -2079,7 +2079,7 @@
"Premium hosting for organisations Learn more": "Hospedaxe Premium para organizacións Saber máis",
"Find other public servers or use a custom server": "Atopa outros servidores públicos ou usa un servidor personalizado",
"Couldn't load page": "Non se puido cargar a páxina",
- "You are an administrator of this community. You will not be able to rejoin without an invite from another administrator.": "Administras esta comunidade. Non poderás voltar a unirte sen un convite doutra persoa administradora.",
+ "You are an administrator of this community. You will not be able to rejoin without an invite from another administrator.": "Administras esta comunidade. Non poderás volver a unirte sen un convite doutra persoa administradora.",
"Want more than a community? Get your own server": "¿Queres algo máis que unha comunidade? Monta o teu propio servidor",
"This homeserver does not support communities": "Este servidor non soporta comunidades",
"Welcome to %(appName)s": "Benvida a %(appName)s",
@@ -2845,5 +2845,131 @@
"Filter rooms and people": "Fitrar salas e persoas",
"Open the link in the email to continue registration.": "Abre a ligazón que hai no email para continuar co rexistro.",
"A confirmation email has been sent to %(emailAddress)s": "Enviouse un email de confirmación a %(emailAddress)s",
- "Start a new chat": "Comezar nova conversa"
+ "Start a new chat": "Comezar nova conversa",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Conservar na memoria local as mensaxes cifradas de xeito seguro para que aparezan nas buscas, usando %(size)s para gardar mensaxes de %(rooms)s salas.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Conservar na memoria local as mensaxes cifradas de xeito seguro para que aparezan nas buscas, usando %(size)s para gardar mensaxes de %(rooms)s salas.",
+ "Go to Home View": "Ir á Páxina de Inicio",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
HTML para a páxina da túa comunidade
\n
\n Usa a descrición longa para presentar a comunidade ás novas particpantes, ou publicar \nalgunha ligazón importante\n \n
\n
\n Tamén podes engadir imaxes con URLs de Matrix \n
\n",
+ "The %(capability)s capability": "A capacidade de %(capability)s",
+ "Decline All": "Rexeitar todo",
+ "Approve": "Aprobar",
+ "This widget would like to:": "O widget podería querer:",
+ "Approve widget permissions": "Aprovar permisos do widget",
+ "Use Ctrl + Enter to send a message": "Usar Ctrl + Enter para enviar unha mensaxe",
+ "Use Command + Enter to send a message": "Usar Command + Enter para enviar unha mensaxe",
+ "See %(msgtype)s messages posted to your active room": "Ver mensaxes %(msgtype)s publicados na túa sala activa",
+ "See %(msgtype)s messages posted to this room": "Ver mensaxes %(msgtype)s publicados nesta sala",
+ "Send %(msgtype)s messages as you in your active room": "Enviar mensaxes %(msgtype)s no teu nome á túa sala activa",
+ "Send %(msgtype)s messages as you in this room": "Enviar mensaxes %(msgtype)s no teu nome a esta sala",
+ "See general files posted to your active room": "Ver ficheiros publicados na túa sala activa",
+ "See general files posted to this room": "Ver ficheiros publicados nesta sala",
+ "Send general files as you in your active room": "Enviar ficheiros no teu nome á túa sala activa",
+ "Send general files as you in this room": "Enviar ficheiros no teu nome a esta sala",
+ "See videos posted to your active room": "Ver vídeos publicados na túa sala activa",
+ "See videos posted to this room": "Ver vídeos publicados nesta sala",
+ "Send videos as you in your active room": "Enviar vídeos no teu nome á túa sala activa",
+ "Send videos as you in this room": "Enviar vídeos no teu nome a esta sala",
+ "See images posted to your active room": "Ver imaxes publicadas na túa sala activa",
+ "See images posted to this room": "Ver imaxes publicadas nesta sala",
+ "Send images as you in your active room": "Enviar imaxes no teu nome á túa sala activa",
+ "Send images as you in this room": "Enviar imaxes no teu nome a esta sala",
+ "See emotes posted to your active room": "Ver emotes publicados na túa sala activa",
+ "See emotes posted to this room": "Ver emotes publicados nesta sala",
+ "Send emotes as you in your active room": "Enviar emotes no teu nome á túa sala activa",
+ "Send emotes as you in this room": "Enviar emotes no teu nome a esta sala",
+ "See text messages posted to your active room": "Ver mensaxes de texto publicados na túa sala activa",
+ "See text messages posted to this room": "Ver mensaxes de texto publicados nesta sala",
+ "Send text messages as you in your active room": "Enviar mensaxes de texto no teu nome á túa sala activa",
+ "Send text messages as you in this room": "Enviar mensaxes de texto no teu nome a esta sala",
+ "See messages posted to your active room": "Ver as mensaxes publicadas na túa sala activa",
+ "See messages posted to this room": "Ver as mensaxes publicadas nesta sala",
+ "Send messages as you in your active room": "Enviar mensaxes no teu nome na túa sala activa",
+ "Send messages as you in this room": "Enviar mensaxes no teu nome nesta sala",
+ "See %(eventType)s events posted to your active room": "Ver os eventos %(eventType)s publicados na túa sala activa",
+ "Send %(eventType)s events as you in your active room": "Envía no teu nome %(eventType)s eventos á túa sala activa",
+ "See %(eventType)s events posted to this room": "Ver %(eventType)s eventos publicados nesta sala",
+ "Send %(eventType)s events as you in this room": "Envia no teu nome %(eventType)s eventos a esta sala",
+ "with state key %(stateKey)s": "coa chave de estado %(stateKey)s",
+ "with an empty state key": "cunha chave de estado baleiro",
+ "See when anyone posts a sticker to your active room": "Ver cando alguén publica un adhesivo na túa sala activa",
+ "Send stickers to your active room as you": "Enviar no teu nome adhesivos á túa sala activa",
+ "See when a sticker is posted in this room": "Ver cando un adhesivo se publica nesta sala",
+ "Send stickers to this room as you": "Enviar no teu nome adhesivos a esta sala",
+ "See when the avatar changes in your active room": "Ver cando o avatar da túa sala activa cambie",
+ "Change the avatar of your active room": "Cambiar o avatar da túa sala activa",
+ "See when the avatar changes in this room": "Ver cando o avatar desta sala cambie",
+ "Change the avatar of this room": "Cambiar o avatar desta sala",
+ "See when the name changes in your active room": "Ver cando o nome da túa sala activa cambie",
+ "Change the name of your active room": "Cambiar o tema da túa sala activa",
+ "See when the name changes in this room": "Ver cando o nome desta sala cambie",
+ "Change the name of this room": "Cambiar o nome desta sala",
+ "See when the topic changes in your active room": "Ver cando o tema da túa sala activa cambie",
+ "Change the topic of your active room": "Cambiar o tema da túa sala activa",
+ "See when the topic changes in this room": "Ver cando o tema desta sala cambie",
+ "Change the topic of this room": "Cambiar o tema desta sala",
+ "Change which room you're viewing": "Cambiar a sala que estás vendo",
+ "Send stickers into your active room": "Enviar adhesivos á túa sala activa",
+ "Send stickers into this room": "Enviar adhesivos a esta sala",
+ "Remain on your screen while running": "Permanecer na túa pantalla mentras se executa",
+ "Remain on your screen when viewing another room, when running": "Permanecer na túa pantalla cando visualizas outra sala, ó executar",
+ "Enter phone number": "Escribe número de teléfono",
+ "Enter email address": "Escribe enderezo email",
+ "Return to call": "Volver á chamada",
+ "Fill Screen": "Encher a pantalla",
+ "Voice Call": "Chamada de voz",
+ "Video Call": "Chamada de vídeo",
+ "New here? Create an account": "Acabas de coñecernos? Crea unha conta",
+ "Got an account? Sign in": "Tes unha conta? Conéctate",
+ "Render LaTeX maths in messages": "Mostrar fórmulas matemáticas LaTex",
+ "No other application is using the webcam": "Outra aplicación non está usando a cámara",
+ "Permission is granted to use the webcam": "Tes permiso para acceder ó uso da cámara",
+ "A microphone and webcam are plugged in and set up correctly": "O micrófono e a cámara están conectados e correctamente configurados",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "A chamada fallou porque non están accesibles a cámara ou o micrófono. Comproba que:",
+ "Unable to access webcam / microphone": "Non se puido acceder a cámara / micrófono",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "A chamada fallou porque non se puido acceder a un micrófono. Comproba que o micrófono está conectado e correctamente configurado.",
+ "Unable to access microphone": "Non se puido acceder ó micrófono",
+ "Decide where your account is hosted": "Decide onde queres crear a túa conta",
+ "Host account on": "Crea a conta en",
+ "Already have an account? Sign in here": "Xa tes unha conta? Conecta aquí",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s Ou %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "Continúa con %(ssoButtons)s",
+ "That username already exists, please try another.": "Ese nome de usuaria xa existe, proba con outro.",
+ "New? Create account": "Recén cheagada? Crea unha conta",
+ "There was a problem communicating with the homeserver, please try again later.": "Houbo un problema de comunicación co servidor de inicio, inténtao máis tarde.",
+ "Use email to optionally be discoverable by existing contacts.": "Usa o email para ser opcionalmente descubrible para os contactos existentes.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Usa un email ou teléfono para ser (opcionalmente) descubrible polos contactos existentes.",
+ "Add an email to be able to reset your password.": "Engade un email para poder restablecer o contrasinal.",
+ "Forgot password?": "Esqueceches o contrasinal?",
+ "That phone number doesn't look quite right, please check and try again": "Non semella correcto este número, compróbao e inténtao outra vez",
+ "About homeservers": "Acerca dos servidores de inicio",
+ "Learn more": "Saber máis",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Usa o teu servidor de inicio Matrix preferido, ou usa o teu propio.",
+ "Other homeserver": "Outro servidor de inicio",
+ "We call the places you where you can host your account ‘homeservers’.": "Chamámoslle 'Servidores de inicio' ós servidores onde poderías ter a túa conta.",
+ "Sign into your homeserver": "Conecta co teu servidor de inicio",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org é o servidor de inicio máis grande de todos, polo que é lugar común para moitas persoas.",
+ "Specify a homeserver": "Indica un servidor de inicio",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Lembra que se non engades un email e esqueces o contrasinal perderás de xeito permanente o acceso á conta.",
+ "Continuing without email": "Continuando sen email",
+ "Continue with %(provider)s": "Continuar con %(provider)s",
+ "Homeserver": "Servidor de inicio",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Podes usar as opcións do servidor para poder conectarte a outros servidores Matrix indicando o URL dese servidor. Esto permíteche usar Element cunha conta Matrix existente noutro servidor.",
+ "Server Options": "Opcións do servidor",
+ "Reason (optional)": "Razón (optativa)",
+ "We call the places where you can host your account ‘homeservers’.": "Ós lugares onde podes ter unha conta chamámoslle 'servidores de inicio'.",
+ "Invalid URL": "URL non válido",
+ "Unable to validate homeserver": "Non se puido validar o servidor de inicio",
+ "sends confetti": "envía confetti",
+ "Sends the given message with confetti": "Envía a mensaxe con confetti",
+ "Show chat effects": "Mostrar efectos do chat",
+ "Effects": "Efectos",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "A chamada fallou porque non tiñas acceso á cámara ou ó micrófono. Comproba:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "A chamada fallou porque non tiñas acceso ó micrófono. Comproba que o micrófono está conectado e correctamente configurado.",
+ "Hold": "Colgar",
+ "Resume": "Retomar",
+ "%(peerName)s held the call": "%(peerName)s finalizou a chamada",
+ "You held the call Resume": "Colgaches a chamada, Retomar",
+ "You've reached the maximum number of simultaneous calls.": "Acadaches o número máximo de chamadas simultáneas.",
+ "Too Many Calls": "Demasiadas chamadas",
+ "%(name)s paused": "detido por %(name)s"
}
diff --git a/src/i18n/strings/hu.json b/src/i18n/strings/hu.json
index 9c00c8ff6e..2739154c19 100644
--- a/src/i18n/strings/hu.json
+++ b/src/i18n/strings/hu.json
@@ -2663,8 +2663,8 @@
"Bolivia": "Bolívia",
"Bhutan": "Bhután",
"Topic: %(topic)s (edit)": "Téma: %(topic)s (szerkesztés)",
- "This is the beginning of your direct message history with .": "Ez a közvetlen üzeneteinek előzményeinek eleje a következővel: .",
- "Only the two of you are in this conversation, unless either of you invites anyone to join.": "Csak ketten vannak ebben a beszélgetésben, hacsak valamelyikőjük nem hív meg valakit, hogy csatlakozzon.",
+ "This is the beginning of your direct message history with .": "Ez a közvetlen beszélgetés kezdete felhasználóval.",
+ "Only the two of you are in this conversation, unless either of you invites anyone to join.": "Csak önök ketten vannak ebben a beszélgetésben, hacsak valamelyikőjük nem hív meg valakit, hogy csatlakozzon.",
"Call Paused": "Hívás szüneteltetve",
"Takes the call in the current room off hold": "Visszaveszi tartásból a jelenlegi szoba hívását",
"Places the call in the current room on hold": "Tartásba teszi a jelenlegi szoba hívását",
@@ -2832,11 +2832,19 @@
"Gibraltar": "Gibraltár",
"%(creator)s created this DM.": "%(creator)s hozta létre ezt az üzenetet.",
"Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "A szobában lévő üzenetek végpontok között titkosítottak. Miután csatlakoztak a felhasználók, ellenőrizheted őket a profiljukban, amit a profilképükre kattintással nyithatsz meg.",
- "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Az üzenetek végpontok között titkosítottak. Ellenőrizze %(displayName)s személyazonosságát a profilján – koppintson a profilképére.",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Az üzenetek végpontok között titkosítottak. Ellenőrizze %(displayName)s személyazonosságát a profilján – kattintson %(displayName)s profilképére.",
"This is the start of .": "Ez a(z) kezdete.",
"Add a photo, so people can easily spot your room.": "Állíts be egy fényképet, hogy az emberek könnyebben felismerjék a szobát!",
"%(displayName)s created this room.": "%(displayName)s készítette ezt a szobát.",
"You created this room.": "Te készítetted ezt a szobát.",
"Add a topic to help people know what it is about.": "Állítsd be a szoba témáját, hogy az emberek tudják, hogy miről van itt szó.",
- "Topic: %(topic)s ": "Téma: %(topic)s "
+ "Topic: %(topic)s ": "Téma: %(topic)s ",
+ "Send stickers to this room as you": "Ön helyett matricák küldése a szobába",
+ "Change the avatar of this room": "A szoba képének megváltoztatása",
+ "Change the name of this room": "A szoba nevének megváltoztatása",
+ "Change the topic of your active room": "Az aktív szoba témájának megváltoztatása",
+ "Change the topic of this room": "A szoba témájának megváltoztatása",
+ "Change which room you're viewing": "Az ön által nézett szoba megváltoztatása",
+ "Send stickers into your active room": "Matricák küldése az ön aktív szobájába",
+ "Send stickers into this room": "Matricák küldése a szobába"
}
diff --git a/src/i18n/strings/it.json b/src/i18n/strings/it.json
index 4b5b85d415..74255a6d2a 100644
--- a/src/i18n/strings/it.json
+++ b/src/i18n/strings/it.json
@@ -2848,5 +2848,128 @@
"Filter rooms and people": "Filtra stanze e persone",
"Open the link in the email to continue registration.": "Apri il link nell'email per continuare la registrazione.",
"A confirmation email has been sent to %(emailAddress)s": "È stata inviata un'email di conferma a %(emailAddress)s",
- "Start a new chat": "Inizia una nuova chat"
+ "Start a new chat": "Inizia una nuova chat",
+ "Go to Home View": "Vai alla vista home",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
HTML per la pagina della tua comunità
\n
\n Usa la descrizione estesa per introdurre i nuovi membri alla comunità, o distribuisci\n alcuni link importanti\n
\n
\n Puoi anche aggiungere immagini con gli URL Matrix \n
\n",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Salva in cache i messaggi cifrati localmente in modo che appaiano nei risultati di ricerca, usando %(size)s per salvarli da %(rooms)s stanza.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Salva in cache i messaggi cifrati localmente in modo che appaiano nei risultati di ricerca, usando %(size)s per salvarli da %(rooms)s stanze.",
+ "See text messages posted to your active room": "Vedi messaggi di testo inviati alla tua stanza attiva",
+ "See text messages posted to this room": "Vedi messaggi di testo inviati a questa stanza",
+ "Send text messages as you in your active room": "Invia messaggi di testo a tuo nome nella tua stanza attiva",
+ "Send text messages as you in this room": "Invia messaggi di testo a tuo nome in questa stanza",
+ "See messages posted to your active room": "Vedi messaggi inviati alla tua stanza attiva",
+ "See messages posted to this room": "Vedi messaggi inviati a questa stanza",
+ "Send messages as you in your active room": "Invia messaggi a tuo nome nella tua stanza attiva",
+ "Send messages as you in this room": "Invia messaggi a tuo nome in questa stanza",
+ "The %(capability)s capability": "La capacità %(capability)s",
+ "See %(eventType)s events posted to your active room": "Vedi eventi %(eventType)s inviati alla tua stanza attiva",
+ "Send %(eventType)s events as you in your active room": "Invia eventi %(eventType)s a tuo nome nella tua stanza attiva",
+ "See %(eventType)s events posted to this room": "Vedi eventi %(eventType)s inviati a questa stanza",
+ "Send %(eventType)s events as you in this room": "Invia eventi %(eventType)s a tuo nome in questa stanza",
+ "with state key %(stateKey)s": "con la chiave di stato %(stateKey)s",
+ "with an empty state key": "con una chiave di stato vuota",
+ "See when anyone posts a sticker to your active room": "Vedi quando qualcuno invia un adesivo alla tua stanza attiva",
+ "Send stickers to this room as you": "Invia adesivi a questa stanza a tuo nome",
+ "Send stickers to your active room as you": "Invia adesivi alla tua stanza attiva a tuo nome",
+ "See when a sticker is posted in this room": "Vedi quando viene inviato un adesivo in questa stanza",
+ "See when the avatar changes in your active room": "Vedi quando l'avatar cambia nella tua stanza attiva",
+ "Change the avatar of your active room": "Cambia l'avatar della tua stanza attiva",
+ "See when the avatar changes in this room": "Vedi quando l'avatar cambia in questa stanza",
+ "Change the avatar of this room": "Cambia l'avatar di questa stanza",
+ "See when the name changes in your active room": "Vedi quando il nome cambia nella tua stanza attiva",
+ "Change the name of your active room": "Cambia il nome della tua stanza attiva",
+ "See when the name changes in this room": "Vedi quando il nome cambia in questa stanza",
+ "Change the name of this room": "Cambia il nome di questa stanza",
+ "See when the topic changes in your active room": "Vedi quando l'argomento cambia nella tua stanza attiva",
+ "Change the topic of your active room": "Cambia l'argomento della tua stanza attiva",
+ "See when the topic changes in this room": "Vedi quando l'argomento cambia in questa stanza",
+ "Change the topic of this room": "Cambia l'argomento di questa stanza",
+ "Change which room you're viewing": "Cambia quale stanza stai vedendo",
+ "Send stickers into your active room": "Invia adesivi nella tua stanza attiva",
+ "Send stickers into this room": "Invia adesivi in questa stanza",
+ "Remain on your screen while running": "Resta sul tuo schermo mentre in esecuzione",
+ "Remain on your screen when viewing another room, when running": "Resta sul tuo schermo quando vedi un'altra stanza, quando in esecuzione",
+ "No other application is using the webcam": "Nessun'altra applicazione sta usando la webcam",
+ "Permission is granted to use the webcam": "Permesso concesso per usare la webcam",
+ "A microphone and webcam are plugged in and set up correctly": "Un microfono e una webcam siano collegati e configurati correttamente",
+ "Unable to access webcam / microphone": "Impossibile accedere alla webcam / microfono",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Chiamata fallita perchè la webcam o il microfono non sono accessibili. Controlla che:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Chiamata fallita perchè il microfono non è accessibile. Controlla che ci sia un microfono collegato e configurato correttamente.",
+ "Unable to access microphone": "Impossibile accedere al microfono",
+ "Decide where your account is hosted": "Decidi dove ospitare il tuo account",
+ "Host account on": "Ospita account su",
+ "Already have an account? Sign in here": "Hai già un account? Accedi qui",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s o %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "Continua con %(ssoButtons)s",
+ "That username already exists, please try another.": "Quel nome utente esiste già, provane un altro.",
+ "New? Create account": "Prima volta? Crea un account",
+ "There was a problem communicating with the homeserver, please try again later.": "C'è stato un problema nella comunicazione con l'homeserver, riprova più tardi.",
+ "New here? Create an account": "Prima volta qui? Crea un account",
+ "Got an account? Sign in": "Hai un account? Accedi",
+ "Use email to optionally be discoverable by existing contacts.": "Usa l'email per essere facoltativamente trovabile dai contatti esistenti.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Usa l'email o il telefono per essere facoltativamente trovabile dai contatti esistenti.",
+ "Add an email to be able to reset your password.": "Aggiungi un'email per poter reimpostare la password.",
+ "Forgot password?": "Hai dimenticato la password?",
+ "That phone number doesn't look quite right, please check and try again": "Quel numero di telefono non sembra corretto, controlla e riprova",
+ "Enter phone number": "Inserisci numero di telefono",
+ "Enter email address": "Inserisci indirizzo email",
+ "Decline All": "Rifiuta tutti",
+ "Approve widget permissions": "Approva permessi del widget",
+ "Approve": "Approva",
+ "This widget would like to:": "Il widget vorrebbe:",
+ "About homeservers": "Riguardo gli homeserver",
+ "Learn more": "Maggiori informazioni",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Usa il tuo homeserver Matrix preferito se ne hai uno, o ospitane uno tuo.",
+ "Other homeserver": "Altro homeserver",
+ "We call the places where you can host your account ‘homeservers’.": "Chiamiamo \"homeserver\" i posti dove puoi ospitare il tuo account.",
+ "Sign into your homeserver": "Accedi al tuo homeserver",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org è il più grande homeserver pubblico del mondo, quindi è un buon posto per molti.",
+ "Specify a homeserver": "Specifica un homeserver",
+ "Invalid URL": "URL non valido",
+ "Unable to validate homeserver": "Impossibile validare l'homeserver",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Solo un avviso, se non aggiungi un'email e dimentichi la password, potresti perdere permanentemente l'accesso al tuo account.",
+ "Continuing without email": "Continuando senza email",
+ "Reason (optional)": "Motivo (facoltativo)",
+ "Continue with %(provider)s": "Continua con %(provider)s",
+ "Homeserver": "Homeserver",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Puoi usare le opzioni di server personalizzato per accedere ad altri server Matrix specificando un URL homeserver diverso. Ciò ti permette di usare Element con un account Matrix esistente su un homeserver differente.",
+ "Server Options": "Opzioni server",
+ "Return to call": "Torna alla chiamata",
+ "Fill Screen": "Riempi schermo",
+ "Voice Call": "Telefonata",
+ "Video Call": "Videochiamata",
+ "sends confetti": "invia coriandoli",
+ "Sends the given message with confetti": "Invia il messaggio in questione con coriandoli",
+ "Show chat effects": "Mostra effetti chat",
+ "Use Ctrl + Enter to send a message": "Usa Ctrl + Invio per inviare un messaggio",
+ "Use Command + Enter to send a message": "Usa Comando + Invio per inviare un messaggio",
+ "Render LaTeX maths in messages": "Renderizza matematica LaTeX nei messaggi",
+ "See %(msgtype)s messages posted to your active room": "Vedi messaggi %(msgtype)s inviati alla tua stanza attiva",
+ "See %(msgtype)s messages posted to this room": "Vedi messaggi %(msgtype)s inviati a questa stanza",
+ "Send %(msgtype)s messages as you in your active room": "Invia messaggi %(msgtype)s a tuo nome nella tua stanza attiva",
+ "Send %(msgtype)s messages as you in this room": "Invia messaggi %(msgtype)s a tuo nome in questa stanza",
+ "See general files posted to your active room": "Vedi file generici inviati alla tua stanza attiva",
+ "See general files posted to this room": "Vedi file generici inviati a questa stanza",
+ "Send general files as you in your active room": "Invia file generici a tuo nome nella tua stanza attiva",
+ "Send general files as you in this room": "Invia file generici a tuo nome in questa stanza",
+ "See videos posted to your active room": "Vedi video inviati alla tua stanza attiva",
+ "See videos posted to this room": "Vedi video inviati a questa stanza",
+ "Send videos as you in your active room": "Invia video a tuo nome nella tua stanza attiva",
+ "Send videos as you in this room": "Invia video a tuo nome in questa stanza",
+ "See images posted to your active room": "Vedi immagini inviate alla tua stanza attiva",
+ "See images posted to this room": "Vedi immagini inviate a questa stanza",
+ "Send images as you in your active room": "Invia immagini a tuo nome nella tua stanza attiva",
+ "Send images as you in this room": "Invia immagini a tuo nome in questa stanza",
+ "See emotes posted to your active room": "Vedi emoticon inviate alla tua stanza attiva",
+ "See emotes posted to this room": "Vedi emoticon inviate a questa stanza",
+ "Send emotes as you in your active room": "Invia emoticon a tuo nome nella tua stanza attiva",
+ "Send emotes as you in this room": "Invia emoticon a tuo nome in questa stanza",
+ "Effects": "Effetti",
+ "Hold": "Sospendi",
+ "Resume": "Riprendi",
+ "%(name)s paused": "%(name)s ha messo in pausa",
+ "%(peerName)s held the call": "%(peerName)s ha sospeso la chiamata",
+ "You held the call Resume": "Hai sospeso la chiamata Riprendi",
+ "You've reached the maximum number of simultaneous calls.": "Hai raggiungo il numero massimo di chiamate simultanee.",
+ "Too Many Calls": "Troppe chiamate"
}
diff --git a/src/i18n/strings/ja.json b/src/i18n/strings/ja.json
index 4ab5bc0de2..c8c356bdcd 100644
--- a/src/i18n/strings/ja.json
+++ b/src/i18n/strings/ja.json
@@ -497,7 +497,7 @@
"'%(groupId)s' is not a valid community ID": "'%(groupId)s' は有効なコミュニティIDではありません",
"Showing flair for these communities:": "次のコミュニティのバッジを表示:",
"This room is not showing flair for any communities": "この部屋はどんなコミュニティに対してもバッジを表示していません",
- "New community ID (e.g. +foo:%(localDomain)s)": "新しいコミュニティID (例 +foo:%(localDomain)s)",
+ "New community ID (e.g. +foo:%(localDomain)s)": "新しいコミュニティ ID (例: +foo:%(localDomain)s)",
"You have enabled URL previews by default.": "デフォルトで URL プレビューが有効です。",
"You have disabled URL previews by default.": "デフォルトで URL プレビューが無効です。",
"URL previews are enabled by default for participants in this room.": "この部屋の参加者は、デフォルトで URL プレビューが有効です。",
@@ -981,7 +981,7 @@
"Preferences": "環境設定",
"Security & Privacy": "セキュリティとプライバシー",
"Room information": "部屋の情報",
- "Internal room ID:": "内部 部屋ID:",
+ "Internal room ID:": "内部部屋 ID:",
"Room version": "部屋のバージョン",
"Room version:": "部屋のバージョン:",
"Developer options": "開発者オプション",
@@ -1007,7 +1007,7 @@
"Block users on other matrix homeservers from joining this room (This setting cannot be changed later!)": "他の Matrix ホームサーバーからの参加を禁止する (この設定はあとから変更できません!)",
"Room Settings - %(roomName)s": "部屋の設定 - %(roomName)s",
"Explore": "探索",
- "Filter": "フィルター",
+ "Filter": "検索",
"Find a room… (e.g. %(exampleRoom)s)": "部屋を探す… (例: %(exampleRoom)s)",
"If you can't find the room you're looking for, ask for an invite or Create a new room.": "もしお探しの部屋が見つからない場合、招待してもらうか部屋を作成しましょう。",
"Enable room encryption": "部屋の暗号化を有効化",
@@ -1466,5 +1466,27 @@
"Recently Direct Messaged": "最近ダイレクトメッセージで会話したユーザー",
"Invite someone using their name, username (like ) or share this room.": "この部屋に誰かを招待したい場合は、招待したいユーザーの名前、( の様な)ユーザー名、またはメールアドレスを指定するか、この部屋を共有してください。",
"Invite someone using their name, email address, username (like ) or share this room.": "この部屋に誰かを招待したい場合は、招待したいユーザーの名前、メールアドレス、または( の様な)ユーザー名を指定するか、この部屋を共有してください。",
- "Upgrade your encryption": "暗号化をアップグレード"
+ "Upgrade your encryption": "暗号化をアップグレード",
+ "Role": "役割",
+ "Send a reply…": "返信を送信する…",
+ "Send a message…": "メッセージを送信する…",
+ "This is the beginning of your direct message history with .": "ここがあなたと のダイレクトメッセージの履歴の先頭です。",
+ "This is the start of .": "ここが の先頭です。",
+ "Add a topic to help people know what it is about.": "トピックを追加すると参加者がこの部屋の目的を理解しやすくなります。",
+ "Topic: %(topic)s (edit)": "トピック: %(topic)s (編集)",
+ "Topic: %(topic)s ": "トピック: %(topic)s ",
+ "%(displayName)s created this room.": "%(displayName)s がこの部屋を作成しました。",
+ "You created this room.": "あなたがこの部屋を作成しました。",
+ "%(creator)s created this DM.": "%(creator)s がこの DM を作成しました。",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "ここでのメッセージはエンドツーエンド暗号化されています。%(displayName)s のアバターをタップして、プロフィールから検証を行うことができます。",
+ "Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "この部屋内でのメッセージの送受信はエンドツーエンド暗号化されています。参加者のアバターをタップして、プロフィールから検証を行うことができます。",
+ "Use default": "既定の設定を使用",
+ "e.g. my-room": "例: my-room",
+ "Room address": "ルームアドレス",
+ "New published address (e.g. #alias:server)": "新しい公開アドレス (例: #alias:server)",
+ "No other published addresses yet, add one below": "現在、公開アドレスがありません。以下から追加可能です。",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "検索結果を表示させるために、暗号化されたメッセージをローカルに安全にキャッシュしています。現在、%(rooms)s 件の部屋のメッセージの保存に %(size)s を使用中です。",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "検索結果を表示させるために、暗号化されたメッセージをローカルに安全にキャッシュしています。現在、%(rooms)s 件の部屋のメッセージの保存に %(size)s を使用中です。",
+ "Mentions & Keywords": "メンションとキーワード",
+ "Security Key": "セキュリティキー"
}
diff --git a/src/i18n/strings/lt.json b/src/i18n/strings/lt.json
index cdee2c3549..62b34104a6 100644
--- a/src/i18n/strings/lt.json
+++ b/src/i18n/strings/lt.json
@@ -218,7 +218,7 @@
"Room %(roomId)s not visible": "Kambarys %(roomId)s nematomas",
"/ddg is not a command": "/ddg nėra komanda",
"Changes your display nickname": "Pakeičia jūsų rodomą slapyvardį",
- "Invites user with given id to current room": "Pakviečia naudotoją su nurodytu id į esamą kambarį",
+ "Invites user with given id to current room": "Pakviečia vartotoją su nurodytu id į dabartinį kambarį",
"You are now ignoring %(userId)s": "Dabar ignoruojate %(userId)s",
"Opens the Developer Tools dialog": "Atveria Programuotojo Įrankių dialogą",
"Verified key": "Patvirtintas raktas",
@@ -272,7 +272,7 @@
"%(senderName)s uploaded a file": "%(senderName)s įkėlė failą",
"Options": "Parinktys",
"Key request sent.": "Rakto užklausa išsiųsta.",
- "Failed to mute user": "Nepavyko nutildyti naudotoją",
+ "Failed to mute user": "Nepavyko nutildyti vartotojo",
"Are you sure?": "Ar tikrai?",
"Ignore": "Ignoruoti",
"Invite": "Pakviesti",
@@ -422,13 +422,13 @@
"Autoplay GIFs and videos": "Automatiškai paleisti GIF ir vaizdo įrašus",
"This event could not be displayed": "Nepavyko parodyti šio įvykio",
"Kick": "Išmesti",
- "Kick this user?": "Išmesti šį naudotoją?",
+ "Kick this user?": "Išmesti šį vartotoją?",
"Failed to kick": "Nepavyko išmesti",
"Unban": "Atblokuoti",
"Ban": "Užblokuoti",
- "Unban this user?": "Atblokuoti šį naudotoją?",
- "Ban this user?": "Užblokuoti šį naudotoją?",
- "Failed to ban user": "Nepavyko užblokuoti naudotoją",
+ "Unban this user?": "Atblokuoti šį vartotoją?",
+ "Ban this user?": "Užblokuoti šį vartotoją?",
+ "Failed to ban user": "Nepavyko užblokuoti vartotojo",
"Invited": "Pakviestas",
"Filter room members": "Filtruoti kambario dalyvius",
"Server unavailable, overloaded, or something else went wrong.": "Serveris neprieinamas, perkrautas arba nutiko kažkas kito.",
@@ -661,7 +661,7 @@
"Public Chat": "Viešas pokalbis",
"There are no visible files in this room": "Šiame kambaryje nėra matomų failų",
"Add a Room": "Pridėti kambarį",
- "Add a User": "Pridėti naudotoją",
+ "Add a User": "Pridėti Vartotoją",
"Long Description (HTML)": "Ilgasis aprašas (HTML)",
"Description": "Aprašas",
"Community %(groupId)s not found": "Bendruomenė %(groupId)s nerasta",
@@ -1686,7 +1686,7 @@
"%(senderName)s removed the alternative addresses %(addresses)s for this room.|other": "%(senderName)s pašalino alternatyvius šio kambario adresus %(addresses)s.",
"Room settings": "Kambario nustatymai",
"Link to most recent message": "Nuoroda į naujausią žinutę",
- "Invite someone using their name, username (like ) or share this room.": "Pakviesti ką nors naudojant jų vardą, vartotojo vardą (pavyzdžiui ) arba bendrinti šį kambarį.",
+ "Invite someone using their name, username (like ) or share this room.": "Pakviesti ką nors naudojant jų vardą, vartotojo vardą (pvz.: ) arba bendrinti šį kambarį.",
"Share Room Message": "Bendrinti Kambario Žinutę",
"Share Room": "Bendrinti Kambarį",
"Use bots, bridges, widgets and sticker packs": "Naudoti botus, tiltus, valdiklius ir lipdukų pakuotes",
@@ -2010,5 +2010,42 @@
"Verify this session by confirming the following number appears on its screen.": "Patvirtinkite šį seansą, įsitikindami, kad jo ekrane rodomas toliau esantis skaičius.",
"Privacy": "Privatumas",
"Accept all %(invitedRooms)s invites": "Priimti visus %(invitedRooms)s pakvietimus",
- "Bulk options": "Grupinės parinktys"
+ "Bulk options": "Grupinės parinktys",
+ "Confirm Security Phrase": "Patvirtinkite Slaptafrazę",
+ "Set a Security Phrase": "Nustatyti Slaptafrazę",
+ "Use a secret phrase only you know, and optionally save a Security Key to use for backup.": "Naudokite slaptafrazę, kurią žinote tik jūs ir pasirinktinai išsaugokite Apsaugos Raktą, naudoti kaip atsarginę kopiją.",
+ "Enter a Security Phrase": "Įveskite Slaptafrazę",
+ "Security Phrase": "Slaptafrazė",
+ "Enter a security phrase only you know, as it’s used to safeguard your data. To be secure, you shouldn’t re-use your account password.": "Įveskite slaptafrazę, kurią žinote tik jūs, nes ji naudojama jūsų duomenims apsaugoti. Tam, kad būtumėte saugūs, neturėtumėte vėl naudoti savo paskyros slaptažodžio.",
+ "Enter your Security Phrase or to continue.": "Įveskite savo Slaptafrazę arba , kad tęstumėte.",
+ "%(creator)s created this DM.": "%(creator)s sukūrė šį tiesioginio susirašymo kambarį.",
+ "Only the two of you are in this conversation, unless either of you invites anyone to join.": "Šiame pokalbyje esate tik jūs dviese, nebent kuris nors iš jūsų pakvies ką nors prisijungti.",
+ "This is the beginning of your direct message history with .": "Tai yra jūsų tiesioginių žinučių su istorijos pradžia.",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Žinutės čia yra visapusiškai užšifruotos. Patvirtinkite %(displayName)s jų profilyje - paspauskite ant jų pseudoportreto.",
+ "See when the avatar changes in your active room": "Matyti kada jūsų aktyviame kambaryje pasikeičia pseudoportretas",
+ "Change the avatar of your active room": "Pakeisti jūsų aktyvaus kambario pseudoportretą",
+ "See when the avatar changes in this room": "Matyti kada šiame kambaryje pasikeičia pseudoportretas",
+ "Change the avatar of this room": "Pakeisti šio kambario pseudoportretą",
+ "Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "Žinutės šiame kambaryje yra visapusiškai užšifruotos. Kai žmonės prisijungia, jūs galite patvirtinti juos jų profilyje, tiesiog paspauskite ant jų pseudoportreto.",
+ "Mentions & Keywords": "Paminėjimai ir Raktažodžiai",
+ "Can't connect to homeserver - please check your connectivity, ensure your homeserver's SSL certificate is trusted, and that a browser extension is not blocking requests.": "Nepavyksta prisijungti prie serverio - patikrinkite savo ryšį, įsitikinkite, kad jūsų serverio SSL sertifikatas yra patikimas ir, kad naršyklės plėtinys neužblokuoja užklausų.",
+ "Not trusted": "Nepatikimas",
+ "Trusted": "Patikimas",
+ "Role": "Rolė",
+ "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.": "Anksčiau šiame seanse naudojote naujesnę %(brand)s versiją. Norėdami vėl naudoti šią versiją su visapusiu šifravimu, turėsite atsijungti ir prisijungti iš naujo.",
+ "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": "Tam, kad neprarastumėte savo pokalbių istorijos, prieš atsijungdami turite eksportuoti kambario raktus. Norėdami tai padaryti, turėsite grįžti į naujesnę %(brand)s versiją",
+ "New version of %(brand)s is available": "Yra nauja %(brand)s versija",
+ "Failed to update community": "Nepavyko atnaujinti bendruomenės",
+ "Update status": "Atnaujinti statusą",
+ "Update community": "Atnaujinti bendruomenę",
+ "The visibility of '%(roomName)s' in %(groupId)s could not be updated.": "Kambario %(roomName)s matomumas bendruomenėje %(groupId)s negalėjo būti atnaujintas.",
+ "Update %(brand)s": "Atnaujinti %(brand)s",
+ "Someone is using an unknown session": "Kažkas naudoja nežinomą seansą",
+ "PRO TIP: If you start a bug, please submit debug logs to help us track down the problem.": "PRO PATARIMAS: Jei pradėjote klaidos pranešimą, pateikite derinimo žurnalus, kad padėtumėte mums išsiaiškinti problemą.",
+ "Please review and accept the policies of this homeserver:": "Peržiūrėkite ir sutikite su šio serverio politika:",
+ "Please review and accept all of the homeserver's policies": "Peržiūrėkite ir sutikite su visa serverio politika",
+ "Please view existing bugs on Github first. No match? Start a new one.": "Pirmiausia peržiūrėkite Github'e esančius pranešimus apie klaidas. Jokio atitikmens? Pradėkite naują pranešimą.",
+ "This usually only affects how the room is processed on the server. If you're having problems with your %(brand)s, please report a bug.": "Paprastai tai turi įtakos tik kambario apdorojimui serveryje. Jei jūs turite problemų su savo %(brand)s, praneškite apie klaidą.",
+ "Report a bug": "Pranešti apie klaidą",
+ "Invite someone using their name, email address, username (like ) or share this room.": "Pakviesti ką nors, naudojant jų vardą, el. pašto adresą, vartotojo vardą (pvz.: ) arba bendrinti šį kambarį."
}
diff --git a/src/i18n/strings/pl.json b/src/i18n/strings/pl.json
index 8122e93f45..f5171c8472 100644
--- a/src/i18n/strings/pl.json
+++ b/src/i18n/strings/pl.json
@@ -385,7 +385,7 @@
"Error decrypting audio": "Błąd deszyfrowania audio",
"Error decrypting image": "Błąd deszyfrowania obrazu",
"Error decrypting video": "Błąd deszyfrowania wideo",
- "Tried to load a specific point in this room's timeline, but was unable to find it.": "Próbowano załadować konkretny punkt na osi czasu w tym pokoju, ale nie nie można go znaleźć.",
+ "Tried to load a specific point in this room's timeline, but was unable to find it.": "Próbowano załadować konkretny punkt na osi czasu w tym pokoju, ale nie można go znaleźć.",
"The exported file will allow anyone who can read it to decrypt any encrypted messages that you can see, so you should be careful to keep it secure. To help with this, you should enter a passphrase below, which will be used to encrypt the exported data. It will only be possible to import the data by using the same passphrase.": "Wyeksportowany plik pozwoli każdej osobie będącej w stanie go odczytać na deszyfrację jakichkolwiek zaszyfrowanych wiadomości, które możesz zobaczyć, tak więc zalecane jest zachowanie ostrożności. Aby w tym pomóc, powinieneś/aś wpisać hasło poniżej; hasło to będzie użyte do zaszyfrowania wyeksportowanych danych. Późniejsze zaimportowanie tych danych będzie możliwe tylko po uprzednim podaniu owego hasła.",
" (unsupported)": " (niewspierany)",
"Idle": "Bezczynny(-a)",
@@ -1150,7 +1150,7 @@
"Disconnect from the identity server ?": "Odłączyć od serwera tożsamości ?",
"Disconnect": "Odłącz",
"Identity Server (%(server)s)": "Serwer tożsamości (%(server)s)",
- "You are currently using to discover and be discoverable by existing contacts you know. You can change your identity server below.": "Używasz , aby odnajdywać i móc być odnajdywanym przez istniejące kontakty, które znasz. Możesz zmienić serwer tożsamości poniżej.",
+ "You are currently using to discover and be discoverable by existing contacts you know. You can change your identity server below.": "Używasz , aby odnajdywać i móc być odnajdywanym przez istniejące kontakty, które znasz. Możesz zmienić serwer tożsamości poniżej.",
"Identity Server": "Serwer Tożsamości",
"You are not currently using an identity server. To discover and be discoverable by existing contacts you know, add one below.": "Nie używasz serwera tożsamości. Aby odkrywać i być odkrywanym przez istniejące kontakty które znasz, dodaj jeden poniżej.",
"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.": "Odłączenie się od serwera tożsamości oznacza, że inni nie będą mogli Cię odnaleźć ani Ty nie będziesz w stanie zaprosić nikogo za pomocą e-maila czy telefonu.",
@@ -1584,5 +1584,391 @@
"Cancel entering passphrase?": "Anulować wpisywanie hasła?",
"Room name or address": "Nazwa lub adres pokoju",
"This will end the conference for everyone. Continue?": "Czy na pewno chcesz zakończyc połączenie grupowe? To zakończy je dla wszystkich uczestnikow.",
- "End conference": "Zakończ połączenie grupowe"
+ "End conference": "Zakończ połączenie grupowe",
+ "Attach files from chat or just drag and drop them anywhere in a room.": "Załącz pliki w rozmowie lub upuść je w dowolnym miejscu rozmowy.",
+ "Sign in with SSO": "Zaloguj się z SSO",
+ "No files visible in this room": "Brak plików widocznych w tym pokoju",
+ "Document": "Dokument",
+ "Service": "Usługa",
+ "Summary": "Opis",
+ "To continue you need to accept the terms of this service.": "Aby kontynuować, musisz zaakceptować zasady użytkowania.",
+ "Connecting to integration manager...": "Łączenie z zarządcą integracji…",
+ "Add widgets, bridges & bots": "Dodaj widżety, mostki i boty",
+ "Forget this room": "Zapomnij o tym pokoju",
+ "You were kicked from %(roomName)s by %(memberName)s": "Zostałeś(-aś) wyrzucony(-a) z %(roomName)s przez %(memberName)s",
+ "List options": "Ustawienia listy",
+ "Explore all public rooms": "Przeglądaj wszystkie publiczne pokoje",
+ "Explore public rooms": "Przeglądaj publiczne pokoje",
+ "Verification Requests": "Żądania weryfikacji",
+ "View Servers in Room": "Zobacz serwery w pokoju",
+ "Changes to who can read history will only apply to future messages in this room. The visibility of existing history will be unchanged.": "Zmiany tego, kto może przeglądać historię wyszukiwania dotyczą tylko przyszłych wiadomości w pokoju. Widoczność wcześniejszej historii nie zmieni się.",
+ "No other published addresses yet, add one below": "Brak innych opublikowanych adresów, dodaj jakiś poniżej",
+ "Other published addresses:": "Inne opublikowane adresy:",
+ "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.": "Opublikowane adresy mogą być używane, aby każdy mógł dołączyć do Twojego pokoju. Aby opublikować adres, należy wcześniej ustawić lokalny adres.",
+ "Room settings": "Ustawienia pokoju",
+ "Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "Wiadomości w tym pokoju są szyfrowane end-to-end. Jeżeli ludzie dołączą do niego, możesz zweryfikować ich na ich profilu, naciskając na ich awatar.",
+ "Messages in this room are not end-to-end encrypted.": "Wiadomości w tym pokoju nie są szyfrowane end-to-end.",
+ "Show files": "Zobacz pliki",
+ "%(count)s people|one": "%(count)s osoba",
+ "%(count)s people|other": "%(count)s ludzi(e)",
+ "About": "Informacje",
+ "Add a topic to help people know what it is about.": "Dodaj temat, aby poinformować ludzi czego to dotyczy.",
+ "Show info about bridges in room settings": "Pokazuj informacje o mostkach w ustawieniach pokoju",
+ "about a day from now": "około dnia od teraz",
+ "about an hour from now": "około godziny od teraz",
+ "about a minute from now": "około minuty od teraz",
+ "Room Info": "Informacje o pokoju",
+ "Reporting this message will send its unique 'event ID' to the administrator of your homeserver. If messages in this room are encrypted, your homeserver administrator will not be able to read the message text or view any files or images.": "Zgłoszenie tej wiadomości wyśle administratorowi serwera unikatowe „ID wydarzenia”. Jeżeli wiadomości w tym pokoju są szyfrowane, administrator serwera może nie być w stanie przeczytać treści wiadomości, lub zobaczyć plików bądź zdjęć.",
+ "Send report": "Wyślij zgłoszenie",
+ "Report Content to Your Homeserver Administrator": "Zgłoś zawartość do administratora swojego serwera",
+ "Private rooms can be found and joined by invitation only. Public rooms can be found and joined by anyone in this community.": "Prywatne pokoje można odnaleźć i dołączyć do nich tylko przez zaproszenie. Do publicznych pokojów może dołączyć każdy w tej społeczności.",
+ "Private rooms can be found and joined by invitation only. Public rooms can be found and joined by anyone.": "Prywatne pokoje można odnaleźć i dołączyć do nich tylko przez zaproszenie. Do publicznych pokojów każdy może dołączyć.",
+ "You might enable this if the room will only be used for collaborating with internal teams on your homeserver. This cannot be changed later.": "Możesz ustawić tę opcję, jeżeli pokój będzie używany wyłącznie do współpracy wewnętrznych zespołów na Twoim serwerze. To nie może być później zmienione.",
+ "Block anyone not part of %(serverName)s from ever joining this room.": "Zablokuj wszystkich niebędących użytkownikami %(serverName)s w tym pokoju.",
+ "You can’t disable this later. Bridges & most bots won’t work yet.": "Nie możesz wyłączyć tego później. Mostki i większość botów nie będą działać.",
+ "Matrix rooms": "Pokoje Matrix",
+ "Start a conversation with someone using their name or username (like ).": "Rozpocznij konwersację z innymi korzystając z ich nazwy lub nazwy użytkownika (np. ).",
+ "Start a conversation with someone using their name, email address or username (like ).": "Rozpocznij konwersację z innymi korzystając z ich nazwy, adresu e-mail lub nazwy użytkownika (np. ).",
+ "Show %(count)s more|one": "Pokaż %(count)s więcej",
+ "Show %(count)s more|other": "Pokaż %(count)s więcej",
+ "Room options": "Ustawienia pokoju",
+ "Manually verify all remote sessions": "Ręcznie weryfikuj wszystkie zdalne sesje",
+ "Privacy": "Prywatność",
+ "This version of %(brand)s does not support searching encrypted messages": "Ta wersja %(brand)s nie obsługuje wyszukiwania zabezpieczonych wiadomości",
+ "Use the Desktop app to search encrypted messages": "Używaj Aplikacji desktopowej, aby wyszukiwać zaszyfrowane wiadomości",
+ "Message search": "Wyszukiwanie wiadomości",
+ "Enable message search in encrypted rooms": "Włącz wyszukiwanie wiadomości w szyfrowanych pokojach",
+ "New version of %(brand)s is available": "Dostępna jest nowa wersja %(brand)s",
+ "Update %(brand)s": "Aktualizuj %(brand)s",
+ "Set up Secure Backup": "Skonfiguruj bezpieczny backup",
+ "Ok": "OK",
+ "Send anonymous usage data which helps us improve %(brand)s. This will use a cookie.": "Wysyłaj anonimowe dane o wykorzystywaniu, które pomogą nam usprawnić %(brand)s. To będzie korzystać z pliku cookie.",
+ "Help us improve %(brand)s": "Pomóż nam usprawnić %(brand)s",
+ "Unknown App": "Nieznana aplikacja",
+ "Enable desktop notifications": "Włącz powiadomienia na pulpicie",
+ "Don't miss a reply": "Nie przegap odpowiedzi",
+ "A session's public name is visible to people you communicate with": "Publiczna nazwa sesji jest widoczna dla osób z którymi się komunikujesz",
+ "Manage the names of and sign out of your sessions below or verify them in your User Profile.": "Zarządzaj nazwami i unieważnaj sesje poniżej, lub weryfikuj je na swoim profilu.",
+ "Where you’re logged in": "Gdzie jesteś zalogowany(-a)",
+ "Review where you’re logged in": "Przejrzyj, gdzie jesteś zalogowany(-a)",
+ "Show tray icon and minimize window to it on close": "Pokazuj ikonę w zasobniku i minimalizuj okno do zasobnika przy zamknięciu",
+ "Display your community flair in rooms configured to show it.": "Wyświetlaj swój wyróżnik społeczności w pokojach skonfigurowanych, aby go używać.",
+ "System font name": "Nazwa czcionki systemowej",
+ "Set the name of a font installed on your system & %(brand)s will attempt to use it.": "Wybierz nazwę czcionki zainstalowanej w systemie, a %(brand)s spróbuje jej użyć.",
+ "Use a system font": "Użyj czcionki systemowej",
+ "Enable experimental, compact IRC style layout": "Włącz eksperymentalny, kompaktowy układ w stylu IRC",
+ "Use a more compact ‘Modern’ layout": "Użyj bardziej kompaktowego „nowoczesnego” układu",
+ "Use custom size": "Użyj niestandardowego rozmiaru",
+ "Appearance Settings only affect this %(brand)s session.": "Ustawienia wyglądu wpływają tylko na tę sesję %(brand)s.",
+ "Customise your appearance": "Dostosuj wygląd",
+ "Use an Integration Manager to manage bots, widgets, and sticker packs.": "Użyj Zarządcy Integracji aby zarządzać botami, widżetami i pakietami naklejek.",
+ "Use an Integration Manager (%(serverName)s) to manage bots, widgets, and sticker packs.": "Użyj Zarządcy Integracji %(serverName)s aby zarządzać botami, widżetami i pakietami naklejek.",
+ "There are two ways you can provide feedback and help us improve %(brand)s.": "Są dwa sposoby na przekazanie informacji zwrotnych i pomoc w usprawnieniu %(brand)s.",
+ "Feedback sent": "Wysłano informacje zwrotne",
+ "Send feedback": "Wyślij informacje zwrotne",
+ "Feedback": "Informacje zwrotne",
+ "You have no visible notifications in this room.": "Nie masz widocznych powiadomień w tym pokoju.",
+ "%(creator)s created this DM.": "%(creator)s utworzył(a) tę wiadomość bezpośrednią.",
+ "You do not have permission to create rooms in this community.": "Nie masz uprawnień do tworzenia pokojów w tej społeczności.",
+ "Cannot create rooms in this community": "Nie można utworzyć pokojów w tej społeczności",
+ "Liberate your communication": "Uwolnij swoją komunikację",
+ "Welcome to %(appName)s": "Witamy w %(appName)s",
+ "Now, let's help you get started": "Teraz pomożemy Ci zacząć",
+ "Welcome %(name)s": "Witaj, %(name)s",
+ "Israel": "Izrael",
+ "Isle of Man": "Man",
+ "Ireland": "Irlandia",
+ "Iraq": "Irak",
+ "Iran": "Iran",
+ "Indonesia": "Indonezja",
+ "India": "Indie",
+ "Iceland": "Islandia",
+ "Hungary": "Węgry",
+ "Hong Kong": "Hong Kong",
+ "Honduras": "Honduras",
+ "Heard & McDonald Islands": "Wyspy Heard i McDonald",
+ "Haiti": "Haiti",
+ "Guyana": "Gujana",
+ "Guinea-Bissau": "Gwinea Bissau",
+ "Guinea": "Gwinea",
+ "Guernsey": "Guernsey",
+ "Guatemala": "Gwatemala",
+ "Guam": "Guam",
+ "Guadeloupe": "Gwadelupa",
+ "Grenada": "Grenada",
+ "Greenland": "Grenlandia",
+ "Greece": "Grecja",
+ "Gibraltar": "Gibraltar",
+ "Ghana": "Ghana",
+ "Germany": "Niemcy",
+ "Georgia": "Gruzja",
+ "Gambia": "Gambia",
+ "Gabon": "Gabon",
+ "French Southern Territories": "Francuskie Terytoria Południowe i Antarktyczne",
+ "French Polynesia": "Polinezja Francuska",
+ "French Guiana": "Gujana Francuska",
+ "France": "Francja",
+ "Finland": "Finlandia",
+ "Fiji": "Fidżi",
+ "Faroe Islands": "Wyspy Owcze",
+ "Falkland Islands": "Falklandy",
+ "Ethiopia": "Etiopia",
+ "Estonia": "Estonia",
+ "Eritrea": "Erytrea",
+ "Equatorial Guinea": "Gwinea Równikowa",
+ "El Salvador": "Salwador",
+ "Egypt": "Egipt",
+ "Ecuador": "Ekwador",
+ "Dominican Republic": "Dominikana",
+ "Dominica": "Dominika",
+ "Djibouti": "Dżibuti",
+ "Denmark": "Dania",
+ "Côte d’Ivoire": "Wybrzeże Kości Słoniowej",
+ "Czech Republic": "Czechy",
+ "Cyprus": "Cypr",
+ "Curaçao": "Curaçao",
+ "Cuba": "Kuba",
+ "Croatia": "Chorwacja",
+ "Costa Rica": "Kostaryka",
+ "Cook Islands": "Wyspy Cooka",
+ "Congo - Kinshasa": "Kinszasa",
+ "Congo - Brazzaville": "Kongo",
+ "Comoros": "Komory",
+ "Colombia": "Kolumbia",
+ "Cocos (Keeling) Islands": "Wyspy Kokosowe",
+ "Christmas Island": "Wyspa Bożego Narodzenia",
+ "China": "Chiny",
+ "Chile": "Chile",
+ "Chad": "Czad",
+ "Central African Republic": "Republika Środkowoafrykańska",
+ "Cayman Islands": "Kajmany",
+ "Caribbean Netherlands": "Holandia Karaibska",
+ "Cape Verde": "Republika Zielonego Przylądka",
+ "Canada": "Kanada",
+ "Cameroon": "Kamerun",
+ "Cambodia": "Kambodża",
+ "Burundi": "Burundi",
+ "Burkina Faso": "Burkina Faso",
+ "Bulgaria": "Bułgaria",
+ "Brunei": "Brunei",
+ "British Virgin Islands": "Brytyjskie Wyspy Dziewicze",
+ "British Indian Ocean Territory": "Brytyjskie Terytorium Oceanu Indyjskiego",
+ "Brazil": "Brazylia",
+ "Bouvet Island": "Wyspa Bouveta",
+ "Botswana": "Botswana",
+ "Bosnia": "Bośnia",
+ "Bolivia": "Boliwia",
+ "Bhutan": "Bhutan",
+ "Bermuda": "Bermudy",
+ "Benin": "Benin",
+ "Belize": "Belize",
+ "Belgium": "Belgia",
+ "Belarus": "Białoruś",
+ "Barbados": "Barbados",
+ "Bangladesh": "Bangladesz",
+ "Bahrain": "Bahrajn",
+ "Bahamas": "Bahamy",
+ "Azerbaijan": "Azerbejdżan",
+ "Austria": "Austria",
+ "Australia": "Australia",
+ "Aruba": "Aruba",
+ "Armenia": "Armenia",
+ "Argentina": "Argentyna",
+ "Antigua & Barbuda": "Antigua i Barbuda",
+ "Antarctica": "Antarktyda",
+ "Anguilla": "Anguilla",
+ "Angola": "Angola",
+ "Andorra": "Andora",
+ "American Samoa": "Samoa Amerykańskie",
+ "Algeria": "Algeria",
+ "Albania": "Albania",
+ "Åland Islands": "Wyspy Alandzkie",
+ "Afghanistan": "Afganistan",
+ "United States": "Stany Zjednoczone",
+ "United Kingdom": "Wielka Brytania",
+ "Marshall Islands": "Wyspy Marshalla",
+ "Malta": "Malta",
+ "Mali": "Mali",
+ "Maldives": "Malediwy",
+ "Malaysia": "Malezja",
+ "Malawi": "Malawi",
+ "Madagascar": "Madagaskar",
+ "Macedonia": "Macedonia",
+ "Macau": "Makau",
+ "Luxembourg": "Luksemburg",
+ "Lithuania": "Litwa",
+ "Liechtenstein": "Liechtenstein",
+ "Libya": "Libia",
+ "Liberia": "Liberia",
+ "Lesotho": "Lesotho",
+ "Lebanon": "Liban",
+ "Latvia": "Łotwa",
+ "Laos": "Laos",
+ "Kyrgyzstan": "Kirgistan",
+ "Kuwait": "Kuwejt",
+ "Kosovo": "Kosowo",
+ "Kiribati": "Kiribati",
+ "Kenya": "Kenia",
+ "Kazakhstan": "Kazachstan",
+ "Jordan": "Jordania",
+ "Jersey": "Jersey",
+ "User rules": "Zasady użytkownika",
+ "Server rules": "Zasady serwera",
+ "not found": "nie znaleziono",
+ "Decline (%(counter)s)": "Odrzuć (%(counter)s)",
+ "Starting backup...": "Rozpoczynanie kopii zapasowej…",
+ "User Autocomplete": "Autouzupełnianie użytkowników",
+ "Community Autocomplete": "Autouzupełnianie społeczności",
+ "Room Autocomplete": "Autouzupełnianie pokojów",
+ "Notification Autocomplete": "Autouzupełnianie powiadomień",
+ "Emoji Autocomplete": "Autouzupełnianie emoji",
+ "Phone (optional)": "Telefon (opcjonalny)",
+ "Upload Error": "Błąd wysyłania",
+ "GitHub issue": "Błąd na GitHubie",
+ "Close dialog": "Zamknij okno dialogowe",
+ "Show all": "Zobacz wszystko",
+ "Deactivate user": "Dezaktywuj użytkownika",
+ "Deactivate user?": "Dezaktywować użytkownika?",
+ "Revoke invite": "Wygaś zaproszenie",
+ "Code block": "Blok kodu",
+ "Ban users": "Zablokuj użytkowników",
+ "Kick users": "Wyrzuć użytkowników",
+ "Syncing...": "Synchronizacja…",
+ "General failure": "Ogólny błąd",
+ "Removing…": "Usuwanie…",
+ "Premium": "Premium",
+ "Cancelling…": "Anulowanie…",
+ "Algorithm:": "Algorytm:",
+ "Bulk options": "Masowe działania",
+ "Modern": "Współczesny",
+ "Compact": "Kompaktowy",
+ "Approve": "Zatwierdź",
+ "Incompatible Database": "Niekompatybilna baza danych",
+ "Show": "Pokaż",
+ "Information": "Informacje",
+ "Categories": "Kategorie",
+ "Reactions": "Reakcje",
+ "Role": "Rola",
+ "Trusted": "Zaufane",
+ "Accepting…": "Akceptowanie…",
+ "Re-join": "Dołącz ponownie",
+ "Unencrypted": "Nieszyfrowane",
+ "Revoke": "Unieważnij",
+ "Encrypted": "Szyfrowane",
+ "Unsubscribe": "Odsubskrybuj",
+ "None": "Brak",
+ "exists": "istnieje",
+ "Change the topic of this room": "Zmień temat tego pokoju",
+ "Change which room you're viewing": "Zmień pokój który przeglądasz",
+ "Send stickers into your active room": "Wyślij naklejki w swoim aktywnym pokoju",
+ "Send stickers into this room": "Wyślij naklejki w tym pokoju",
+ "Zimbabwe": "Zimbabwe",
+ "Zambia": "Zambia",
+ "Yemen": "Jemen",
+ "Western Sahara": "Sahara Zachodnia",
+ "Wallis & Futuna": "Wallis i Futuna",
+ "Vietnam": "Wietnam",
+ "Venezuela": "Wenezuela",
+ "Vatican City": "Watykan",
+ "Vanuatu": "Vanuatu",
+ "Uzbekistan": "Uzbekistan",
+ "Uruguay": "Urugwaj",
+ "United Arab Emirates": "Zjednoczone Emiraty Arabskie",
+ "Ukraine": "Ukraina",
+ "Uganda": "Uganda",
+ "U.S. Virgin Islands": "Wyspy Dziewicze Stanów Zjednoczonych",
+ "Tuvalu": "Tuvalu",
+ "Turks & Caicos Islands": "Turks i Caicos",
+ "Turkmenistan": "Turkmenistan",
+ "Turkey": "Turcja",
+ "Tunisia": "Tunezja",
+ "Trinidad & Tobago": "Trynidad i Tobago",
+ "Tonga": "Tonga",
+ "Tokelau": "Tokelau",
+ "Togo": "Togo",
+ "Timor-Leste": "Timor Wschodni",
+ "Thailand": "Tajlandia",
+ "Tanzania": "Tanzania",
+ "Tajikistan": "Tadżykistan",
+ "Taiwan": "Tajwan",
+ "São Tomé & Príncipe": "Wyspy Świętego Tomasza i Książęca",
+ "Syria": "Syria",
+ "Switzerland": "Szwajcaria",
+ "Sweden": "Szwecja",
+ "Swaziland": "Eswatini",
+ "Svalbard & Jan Mayen": "Svalbard i Jan Mayen",
+ "Suriname": "Surinam",
+ "Sudan": "Sudan",
+ "St. Vincent & Grenadines": "Saint Vincent i Grenadyny",
+ "St. Pierre & Miquelon": "Saint-Pierre i Miquelon",
+ "St. Martin": "Sint Maarten",
+ "St. Lucia": "Saint Lucia",
+ "St. Kitts & Nevis": "Saint Kitts & Nevis",
+ "St. Helena": "Święta Helena",
+ "St. Barthélemy": "Wspólnota Saint-Barthélemy",
+ "Sri Lanka": "Sri Lanka",
+ "Spain": "Hiszpania",
+ "South Sudan": "Sudan Południowy",
+ "South Korea": "Korea Południowa",
+ "South Georgia & South Sandwich Islands": "Georgia Południowa i Sandwich Południowy",
+ "South Africa": "Republika Południowej Afryki",
+ "Somalia": "Somalia",
+ "Solomon Islands": "Wyspy Salomona",
+ "Slovenia": "Słowenia",
+ "Slovakia": "Słowacja",
+ "Sint Maarten": "Sint Maarten",
+ "Singapore": "Singapur",
+ "Sierra Leone": "Sierra Leone",
+ "Seychelles": "Seszele",
+ "Serbia": "Serbia",
+ "Senegal": "Senegal",
+ "Saudi Arabia": "Arabia Saudyjska",
+ "San Marino": "San Marino",
+ "Samoa": "Samoa",
+ "Réunion": "Reunion",
+ "Rwanda": "Rwanda",
+ "Russia": "Rosja",
+ "Romania": "Rumunia",
+ "Qatar": "Katar",
+ "Puerto Rico": "Portoryko",
+ "Portugal": "Portugalia",
+ "Poland": "Polska",
+ "Pitcairn Islands": "Pitcairn",
+ "Philippines": "Filipiny",
+ "Peru": "Peru",
+ "Paraguay": "Paragwaj",
+ "Papua New Guinea": "Papua Nowa Gwinea",
+ "Panama": "Panama",
+ "Palestine": "Palestyna",
+ "Palau": "Palau",
+ "Pakistan": "Pakistan",
+ "Oman": "Oman",
+ "Norway": "Norwegia",
+ "Northern Mariana Islands": "Mariany Północne",
+ "North Korea": "Korea Północna",
+ "Norfolk Island": "Norfolk",
+ "Niue": "Niue",
+ "Nigeria": "Nigeria",
+ "Niger": "Niger",
+ "Nicaragua": "Nikaragua",
+ "New Zealand": "Nowa Zelandia",
+ "New Caledonia": "Nowa Kaledonia",
+ "Netherlands": "Holandia",
+ "Nepal": "Nepal",
+ "Nauru": "Nauru",
+ "Namibia": "Namibia",
+ "Myanmar": "Mjanma",
+ "Mozambique": "Mozambik",
+ "Morocco": "Maroko",
+ "Montserrat": "Montserrat",
+ "Montenegro": "Czarnogóra",
+ "Mongolia": "Mongolia",
+ "Monaco": "Monako",
+ "Moldova": "Mołdawia",
+ "Micronesia": "Mikronezja",
+ "Mexico": "Meksyk",
+ "Mayotte": "Majotta",
+ "Mauritius": "Mauritius",
+ "Mauritania": "Mauretania",
+ "Martinique": "Martynika"
}
diff --git a/src/i18n/strings/pt_BR.json b/src/i18n/strings/pt_BR.json
index aa4bfc41ca..b456c98aed 100644
--- a/src/i18n/strings/pt_BR.json
+++ b/src/i18n/strings/pt_BR.json
@@ -501,7 +501,7 @@
"Visibility in Room List": "Visibilidade na lista de salas",
"Visible to everyone": "Visível para todos",
"Only visible to community members": "Apenas visível para participantes da comunidade",
- "Filter community rooms": "Filtrar salas da comunidade",
+ "Filter community rooms": "Pesquisar salas da comunidade",
"Something went wrong when trying to get your communities.": "Não foi possível carregar suas comunidades.",
"Display your community flair in rooms configured to show it.": "Mostrar o ícone da sua comunidade nas salas que o permitem.",
"You're not currently a member of any communities.": "No momento, você não é participante de nenhuma comunidade.",
@@ -2772,5 +2772,136 @@
"Uzbekistan": "Uzbequistão",
"Role": "Função",
"Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(count)s rooms.|one": "Armazene localmente com segurança as mensagens criptografadas para que apareçam nos resultados da pesquisa, usando %(size)s para armazenar mensagens de %(count)s sala.",
- "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(count)s rooms.|other": "Armazene localmente com segurança as mensagens criptografadas para que apareçam nos resultados da pesquisa, usando %(size)s para armazenar mensagens de %(count)s salas."
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(count)s rooms.|other": "Armazene localmente com segurança as mensagens criptografadas para que apareçam nos resultados da pesquisa, usando %(size)s para armazenar mensagens de %(count)s salas.",
+ "Filter": "Pesquisar",
+ "Start a new chat": "Iniciar uma nova conversa",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Armazene mensagens criptografadas de forma segura localmente para que apareçam nos resultados das buscas. %(size)s é necessário para armazenar mensagens de %(rooms)s sala.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Armazene mensagens criptografadas de forma segura localmente para que apareçam nos resultados das buscas. %(size)s é necessário para armazenar mensagens de %(rooms)s salas.",
+ "Filter rooms and people": "Pesquisar salas e pessoas",
+ "Open the link in the email to continue registration.": "Abra o link no e-mail para continuar o registro.",
+ "A confirmation email has been sent to %(emailAddress)s": "Um e-mail de confirmação foi enviado para %(emailAddress)s",
+ "Go to Home View": "Ir para a tela inicial",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
HTML para a página da sua comunidade
\n
\n Escreva uma descrição longa para apresentar novos membros à comunidade, ou liste\n alguns links importantes\n
\n
\n Você pode até adicionar fotos com URLs na Matrix \n
\n",
+ "Remain on your screen while running": "Permaneça na tela, quando executar",
+ "Remain on your screen when viewing another room, when running": "Permaneça na tela ao visualizar outra sala, quando executar",
+ "New here? Create an account": "Novo por aqui? Crie uma conta",
+ "Got an account? Sign in": "Já tem uma conta? Login",
+ "Use Command + Enter to send a message": "Usar Command + Enter para enviar uma mensagem",
+ "Enter phone number": "Digite o número de telefone",
+ "Enter email address": "Digite o endereço de e-mail",
+ "Decline All": "Recusar tudo",
+ "Approve": "Autorizar",
+ "This widget would like to:": "Este widget gostaria de:",
+ "Approve widget permissions": "Autorizar as permissões do widget",
+ "Return to call": "Retornar para a chamada",
+ "Fill Screen": "Preencher a tela",
+ "Voice Call": "Chamada de voz",
+ "Video Call": "Chamada de vídeo",
+ "Use Ctrl + Enter to send a message": "Usar Ctrl + Enter para enviar uma mensagem",
+ "Render LaTeX maths in messages": "Renderizar fórmulas matemáticas LaTeX em mensagens",
+ "See %(msgtype)s messages posted to your active room": "Veja mensagens de %(msgtype)s enviadas nesta sala ativa",
+ "See %(msgtype)s messages posted to this room": "Veja mensagens de %(msgtype)s enviadas nesta sala",
+ "Send %(msgtype)s messages as you in your active room": "Enviar mensagens de %(msgtype)s nesta sala ativa",
+ "Send %(msgtype)s messages as you in this room": "Enviar mensagens de %(msgtype)s nesta sala",
+ "See general files posted to your active room": "Veja os arquivos enviados nesta sala ativa",
+ "See general files posted to this room": "Veja os arquivos enviados nesta sala",
+ "Send general files as you in your active room": "Enviar arquivos nesta sala ativa",
+ "Send general files as you in this room": "Enviar arquivos nesta sala",
+ "See videos posted to your active room": "Veja os vídeos enviados nesta sala ativa",
+ "See videos posted to this room": "Veja os vídeos enviados nesta sala",
+ "Send videos as you in your active room": "Enviar vídeos nesta sala ativa",
+ "Send videos as you in this room": "Enviar vídeos nesta sala",
+ "See images posted to your active room": "Veja as fotos enviadas nesta sala ativa",
+ "See images posted to this room": "Veja as fotos enviadas nesta sala",
+ "Send images as you in your active room": "Enviar fotos nesta sala ativa",
+ "Send images as you in this room": "Enviar fotos nesta sala",
+ "See emotes posted to your active room": "Veja emojis enviados nesta sala ativa",
+ "See emotes posted to this room": "Veja emojis enviados nesta sala",
+ "Send emotes as you in your active room": "Enviar emojis nesta sala ativa",
+ "Send emotes as you in this room": "Enviar emojis nesta sala",
+ "See text messages posted to your active room": "Veja as mensagens de texto enviadas nesta sala ativa",
+ "See text messages posted to this room": "Veja as mensagens de texto enviadas nesta sala",
+ "Send text messages as you in your active room": "Enviar mensagens de texto nesta sala ativa",
+ "Send text messages as you in this room": "Enviar mensagens de texto nesta sala",
+ "See messages posted to your active room": "Veja as mensagens enviadas nesta sala ativa",
+ "See messages posted to this room": "Veja as mensagens enviadas nesta sala",
+ "Send messages as you in your active room": "Enviar mensagens nesta sala ativa",
+ "Send messages as you in this room": "Enviar mensagens nesta sala",
+ "The %(capability)s capability": "A permissão %(capability)s",
+ "See %(eventType)s events posted to your active room": "Veja eventos de %(eventType)s enviados nesta sala ativa",
+ "Send %(eventType)s events as you in your active room": "Enviar eventos de %(eventType)s nesta sala ativa",
+ "See %(eventType)s events posted to this room": "Veja eventos de %(eventType)s postados nesta sala",
+ "Send %(eventType)s events as you in this room": "Enviar eventos de %(eventType)s nesta sala",
+ "with state key %(stateKey)s": "com chave de estado %(stateKey)s",
+ "with an empty state key": "com uma chave de estado vazia",
+ "See when anyone posts a sticker to your active room": "Veja quando alguém enviar uma figurinha nesta sala ativa",
+ "Send stickers to your active room as you": "Enviar figurinhas para esta sala ativa",
+ "See when a sticker is posted in this room": "Veja quando uma figurinha for enviada nesta sala",
+ "Send stickers to this room as you": "Enviar figurinhas para esta sala",
+ "See when the avatar changes in your active room": "Veja quando a foto desta sala ativa for alterada",
+ "Change the avatar of your active room": "Alterar a foto desta sala ativa",
+ "See when the avatar changes in this room": "Veja quando a foto desta sala for alterada",
+ "Change the avatar of this room": "Alterar a foto desta sala",
+ "See when the name changes in your active room": "Veja quando o nome desta sala ativa for alterado",
+ "Change the name of your active room": "Alterar o nome desta sala ativa",
+ "See when the name changes in this room": "Veja quando o nome desta sala for alterado",
+ "Change the name of this room": "Alterar o nome desta sala",
+ "See when the topic changes in your active room": "Veja quando a descrição for alterada nesta sala ativa",
+ "Change the topic of your active room": "Alterar a descrição desta sala ativa",
+ "See when the topic changes in this room": "Veja quando a descrição for alterada nesta sala",
+ "Change the topic of this room": "Alterar a descrição desta sala",
+ "Change which room you're viewing": "Alterar a sala que você está vendo",
+ "Send stickers into your active room": "Enviar figurinhas nesta sala ativa",
+ "Send stickers into this room": "Enviar figurinhas nesta sala",
+ "No other application is using the webcam": "Nenhum outro aplicativo está usando a câmera",
+ "Permission is granted to use the webcam": "Permissão concedida para usar a câmera",
+ "A microphone and webcam are plugged in and set up correctly": "Um microfone e uma câmera estão conectados e configurados corretamente",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "A chamada falhou porque não foi possível acessar alguma câmera ou microfone. Verifique se:",
+ "Unable to access webcam / microphone": "Não é possível acessar a câmera/microfone",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "A chamada falhou porque não foi possível acessar algum microfone. Verifique se o microfone está conectado e configurado corretamente.",
+ "Unable to access microphone": "Não é possível acessar o microfone",
+ "New? Create account": "Quer se registrar? Crie uma conta",
+ "Decide where your account is hosted": "Decida onde a sua conta será hospedada",
+ "Host account on": "Hospedar conta em",
+ "Already have an account? Sign in here": "Já tem uma conta? Entre aqui",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s ou %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "Continuar com %(ssoButtons)s",
+ "That username already exists, please try another.": "Este nome de usuário já existe. Por favor, digite outro.",
+ "There was a problem communicating with the homeserver, please try again later.": "Ocorreu um problema de comunicação com o servidor local. Tente novamente mais tarde.",
+ "Use email to optionally be discoverable by existing contacts.": "Seja encontrado por seus contatos a partir de um e-mail.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Seja encontrado por seus contatos a partir de um e-mail ou número de telefone.",
+ "Add an email to be able to reset your password.": "Adicione um e-mail para depois poder redefinir sua senha.",
+ "Forgot password?": "Esqueceu a senha?",
+ "That phone number doesn't look quite right, please check and try again": "Esse número de telefone não é válido, verifique e tente novamente",
+ "About homeservers": "Sobre os servidores locais",
+ "Learn more": "Saiba mais",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Use o seu servidor local Matrix preferido, ou hospede o seu próprio servidor.",
+ "Other homeserver": "Outro servidor local",
+ "We call the places you where you can host your account ‘homeservers’.": "Chamamos de \"servidores locais\" os locais onde você pode hospedar a sua conta.",
+ "Sign into your homeserver": "Faça login em seu servidor local",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org é o maior servidor local público do mundo, então é um bom lugar para muitas pessoas.",
+ "Specify a homeserver": "Digite um servidor local",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Apenas um aviso: se você não adicionar um e-mail e depois esquecer sua senha, poderá perder permanentemente o acesso à sua conta.",
+ "Continuing without email": "Continuar sem e-mail",
+ "Continue with %(provider)s": "Continuar com %(provider)s",
+ "Homeserver": "Servidor local",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Você pode usar as opções personalizadas de servidor para entrar em outros servidores Matrix, especificando um URL de servidor local diferente. Isso permite que você use o Element com uma conta Matrix em um servidor local diferente.",
+ "Server Options": "Opções do servidor",
+ "Reason (optional)": "Motivo (opcional)",
+ "We call the places where you can host your account ‘homeservers’.": "Os locais onde você pode hospedar sua conta são chamados de \"servidores locais\".",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "A chamada falhou porque a câmera ou o microfone não puderam ser acessados. Verifique se:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "A chamada falhou porque não foi possível acessar o microfone. Verifique se o microfone está conectado e configurado corretamente.",
+ "Invalid URL": "URL inválido",
+ "Unable to validate homeserver": "Não foi possível validar o servidor local",
+ "sends confetti": "envia confetes",
+ "Sends the given message with confetti": "Envia a mensagem com confetes",
+ "Show chat effects": "Mostrar efeitos em conversas",
+ "Effects": "Efeitos",
+ "Hold": "Pausar",
+ "Resume": "Retomar",
+ "%(peerName)s held the call": "%(peerName)s pausou a chamada",
+ "You held the call Resume": "Você pausou a chamada Retomar",
+ "You've reached the maximum number of simultaneous calls.": "Você atingiu o número máximo de chamadas simultâneas.",
+ "Too Many Calls": "Muitas chamadas",
+ "%(name)s paused": "%(name)s pausou"
}
diff --git a/src/i18n/strings/ru.json b/src/i18n/strings/ru.json
index 8bf1ee15d3..bb6dbdf308 100644
--- a/src/i18n/strings/ru.json
+++ b/src/i18n/strings/ru.json
@@ -1084,8 +1084,8 @@
"Headphones": "Наушники",
"Folder": "Папка",
"Pin": "Кнопка",
- "Your keys are being backed up (the first backup could take a few minutes).": "Выполняется резервная копия ключей (первый раз это может занять несколько минут).",
- "The file '%(fileName)s' exceeds this homeserver's size limit for uploads": "Размер файла '%(fileName)s' превышает ограничения сервера для загрузки.",
+ "Your keys are being backed up (the first backup could take a few minutes).": "Выполняется резервная копия ключей (первый раз это может занять несколько минут).",
+ "The file '%(fileName)s' exceeds this homeserver's size limit for uploads": "Размер файла '%(fileName)s' превышает ограничения сервера для загрузки",
"Prepends ¯\\_(ツ)_/¯ to a plain-text message": "Добавляет смайл ¯\\_(ツ)_/¯ в начало сообщения",
"Changes your display nickname in the current room only": "Изменяет ваш псевдоним только для текущей комнаты",
"Gets or sets the room topic": "Читает или устанавливает тему комнаты",
@@ -1507,7 +1507,7 @@
"Italics": "Курсив",
"Strikethrough": "Перечёркнутый",
"Code block": "Блок кода",
- "%(count)s unread messages.|other": "%(count)s непрочитанные сообщения.",
+ "%(count)s unread messages.|other": "%(count)s непрочитанных сообщений.",
"Unread mentions.": "Непрочитанные упоминания.",
"Show image": "Показать изображение",
"e.g. my-room": "например, моя-комната",
@@ -1535,7 +1535,7 @@
"This client does not support end-to-end encryption.": "Этот клиент не поддерживает сквозное шифрование.",
"Messages in this room are not end-to-end encrypted.": "Сообщения в этой комнате не шифруются сквозным шифрованием.",
"Please create a new issue on GitHub so that we can investigate this bug.": "Пожалуйста, создайте новую проблему/вопрос на GitHub, чтобы мы могли расследовать эту ошибку.",
- "Use an identity server to invite by email. Use the default (%(defaultIdentityServerName)s) or manage in Settings.": "Используйте идентификационный сервер для приглашения по электронной почте. Используйте значение по умолчанию (%(defaultIdentityServerName)s) или управляйте в Настройках.",
+ "Use an identity server to invite by email. Use the default (%(defaultIdentityServerName)s) or manage in Settings.": "Используйте идентификационный сервер для приглашения по электронной почте. Используйте значение по умолчанию (%(defaultIdentityServerName)s) или управляйте в Настройках.",
"Use an identity server to invite by email. Manage in Settings.": "Используйте идентификационный сервер для приглашения по электронной почте. Управление в Настройки.",
"Block users on other matrix homeservers from joining this room (This setting cannot be changed later!)": "Запретить пользователям других Matrix-Серверов присоединяться к этой комнате (этот параметр нельзя изменить позже!)",
"Reporting this message will send its unique 'event ID' to the administrator of your homeserver. If messages in this room are encrypted, your homeserver administrator will not be able to read the message text or view any files or images.": "Отчет о данном сообщении отправит свой уникальный 'event ID' администратору вашего домашнего сервера. Если сообщения в этой комнате зашифрованы, администратор вашего домашнего сервера не сможет прочитать текст сообщения или просмотреть какие-либо файлы или изображения.",
@@ -2191,8 +2191,8 @@
"There was an error updating the room's alternative addresses. It may not be allowed by the server or a temporary failure occurred.": "Произошла ошибка при обновлении альтернативных адресов комнаты. Это может быть запрещено сервером или произошел временный сбой.",
"There was an error creating that address. It may not be allowed by the server or a temporary failure occurred.": "При создании этого адреса произошла ошибка. Это может быть запрещено сервером или произошел временный сбой.",
"There was an error removing that address. It may no longer exist or a temporary error occurred.": "Произошла ошибка при удалении этого адреса. Возможно, он больше не существует или произошла временная ошибка.",
- "Using this widget may share data with %(widgetDomain)s & your Integration Manager.": "Используя этот виджет, вы можете делиться данными с %(widgetDomain)s и вашим Менеджером Интеграции.",
- "Using this widget may share data with %(widgetDomain)s.": "Используя этот виджет, вы можете делиться данными с %(widgetDomain)s.",
+ "Using this widget may share data with %(widgetDomain)s & your Integration Manager.": "Используя этот виджет, вы можете делиться данными с %(widgetDomain)s и вашим Менеджером Интеграции.",
+ "Using this widget may share data with %(widgetDomain)s.": "Используя этот виджет, вы можете делиться данными с %(widgetDomain)s.",
"Can't find this server or its room list": "Не можем найти этот сервер или его список комнат",
"Deleting cross-signing keys is permanent. Anyone you have verified with will see security alerts. You almost certainly don't want to do this, unless you've lost every device you can cross-sign from.": "Удаление ключей кросс-подписи является мгновенным и необратимым действием. Любой, с кем вы прошли проверку, увидит предупреждения безопасности. Вы почти наверняка не захотите этого делать, если только не потеряете все устройства, с которых можно совершать кросс-подпись.",
"Clearing all data from this session is permanent. Encrypted messages will be lost unless their keys have been backed up.": "Очистка всех данных этой сессии является необратимым действием. Зашифрованные сообщения будут потеряны, если их ключи не были сохранены.",
@@ -2524,7 +2524,7 @@
"Starting microphone...": "Запуск микрофона…",
"🎉 All servers are banned from participating! This room can no longer be used.": "🎉 Все серверы запрещены к участию! Эта комната больше не может быть использована.",
"Remove messages sent by others": "Удалить сообщения, отправленные другими",
- "Offline encrypted messaging using dehydrated devices": "Автономный обмен зашифрованными сообщениями с сохраненными устройствами",
+ "Offline encrypted messaging using dehydrated devices": "Автономный обмен зашифрованными сообщениями с сохранёнными устройствами",
"Move right": "Сдвинуть вправо",
"Move left": "Сдвинуть влево",
"Revoke permissions": "Отозвать разрешения",
@@ -2558,5 +2558,410 @@
"Rate %(brand)s": "Оценить %(brand)s",
"Feedback sent": "Отзыв отправлен",
"%(senderName)s ended the call": "%(senderName)s завершил(а) звонок",
- "You ended the call": "Вы закончили звонок"
+ "You ended the call": "Вы закончили звонок",
+ "Send stickers into this room": "Отправить стикеры в эту комнату",
+ "Use Ctrl + Enter to send a message": "Используйте Ctrl + Enter, чтобы отправить сообщение",
+ "Use Command + Enter to send a message": "Используйте Command + Enter, чтобы отправить сообщение",
+ "Go to Home View": "Перейти на главную страницу",
+ "Filter rooms and people": "Фильтровать комнаты и людей",
+ "Open the link in the email to continue registration.": "Откройте ссылку в письме, чтобы продолжить регистрацию.",
+ "A confirmation email has been sent to %(emailAddress)s": "Письмо с подтверждением отправлено на %(emailAddress)s",
+ "Start a new chat": "Начать новый чат",
+ "Role": "Роль",
+ "Messages in this room are end-to-end encrypted. When people join, you can verify them in their profile, just tap on their avatar.": "Сообщения в этой комнате полностью зашифрованы. Когда люди присоединяются, вы можете проверить их в их профиле, просто нажмите на их аватар.",
+ "This is the start of .": "Это начало .",
+ "Add a photo, so people can easily spot your room.": "Добавьте фото, чтобы люди могли легко заметить вашу комнату.",
+ "%(displayName)s created this room.": "%(displayName)s создал(а) эту комнату.",
+ "You created this room.": "Вы создали эту комнату.",
+ "Add a topic to help people know what it is about.": "Добавьте тему, чтобы люди знали, о чём комната.",
+ "Topic: %(topic)s ": "Тема: %(topic)s ",
+ "Topic: %(topic)s (edit)": "Тема: %(topic)s (изменить)",
+ "This is the beginning of your direct message history with .": "Это начало вашей истории прямых сообщений с .",
+ "Only the two of you are in this conversation, unless either of you invites anyone to join.": "В этом разговоре только вы двое, если только кто-нибудь из вас не пригласит кого-нибудь присоединиться.",
+ "Takes the call in the current room off hold": "Прекратить удержание вызова в текущей комнате",
+ "Places the call in the current room on hold": "Перевести вызов в текущей комнате на удержание",
+ "Now, let's help you get started": "Теперь давайте поможем вам начать",
+ "Invite someone using their name, email address, username (like ) or share this room.": "Пригласите кого-нибудь, используя его имя, адрес электронной почты, имя пользователя (например, ) или поделитесь этой комнатой.",
+ "Start a conversation with someone using their name, email address or username (like ).": "Начните разговор с кем-нибудь, используя его имя, адрес электронной почты или имя пользователя (например, ).",
+ "Invite by email": "Пригласить по электронной почте",
+ "Welcome %(name)s": "Добро пожаловать, %(name)s",
+ "Add a photo so people know it's you.": "Добавьте фото, чтобы люди знали, что это вы.",
+ "Great, that'll help people know it's you": "Отлично, это поможет людям узнать, что это ты",
+ "Use the + to make a new room or explore existing ones below": "Используйте +, чтобы создать новую комнату или изучить существующие ниже",
+ "New version of %(brand)s is available": "Доступна новая версия %(brand)s!",
+ "Update %(brand)s": "Обновление %(brand)s",
+ "Enable desktop notifications": "Включить уведомления на рабочем столе",
+ "Don't miss a reply": "Не пропустите ответ",
+ "No other application is using the webcam": "Никакое другое приложение не использует веб-камеру",
+ "Permission is granted to use the webcam": "Разрешение на использование еб-камеры предоставлено",
+ "A microphone and webcam are plugged in and set up correctly": "Микрофон и веб-камера подключены и правильно настроены",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "Вызов не удался, потому что не удалось получить доступ к веб-камере или микрофону. Проверьте это:",
+ "Unable to access webcam / microphone": "Невозможно получить доступ к веб-камере / микрофону",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Вызов не удался, потому что нет доступа к микрофону. Убедитесь, что микрофон подключен и правильно настроен.",
+ "Unable to access microphone": "Нет доступа к микрофону",
+ "Video Call": "Видеовызов",
+ "Voice Call": "Голосовой вызов",
+ "Fill Screen": "Заполнить экран",
+ "Return to call": "Вернуться к звонку",
+ "Got an account? Sign in": "Есть учётная запись? Войти",
+ "New here? Create an account": "Впервые здесь? Создать учётную запись",
+ "Render LaTeX maths in messages": "Отображать математику LaTeX в сообщениях",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
HTML для страницы вашего сообщества
\n
\n Используйте подробное описание, чтобы представить новых участников сообществу или распространить\n некоторые важные ссылки\n
\n
\n Вы даже можете добавлять изображения с URL-адресами Matrix \n
\n",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Надежно кэшируйте зашифрованные сообщения локально, чтобы они отображались в результатах поиска, используется %(size)s для хранения сообщений из %(rooms)s комнаты.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Надежно кэшируйте зашифрованные сообщения локально, чтобы они отображались в результатах поиска, используется %(size)s для хранения сообщений из %(rooms)s комнат.",
+ "Messages here are end-to-end encrypted. Verify %(displayName)s in their profile - tap on their avatar.": "Сообщения в этом чате полностью зашифрованы. Вы можете проверить профиль %(displayName)s, нажав на аватар.",
+ "Unable to validate homeserver": "Невозможно проверить домашний сервер",
+ "Sign into your homeserver": "Войдите на свой домашний сервер",
+ "with state key %(stateKey)s": "с ключом состояния %(stateKey)s",
+ "%(creator)s created this DM.": "%(creator)s начал этот чат.",
+ "Show chat effects": "Показать эффекты чата",
+ "Host account on": "Ваша учётная запись обслуживается",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s или %(usernamePassword)s",
+ "Continuing without email": "Продолжить без электронной почты",
+ "Continue with %(ssoButtons)s": "Продолжить с %(ssoButtons)s",
+ "New? Create account": "Впервые тут? Создать учётную запись",
+ "Specify a homeserver": "Укажите домашний сервер",
+ "Continue with %(provider)s": "Продолжить с %(provider)s",
+ "Enter phone number": "Введите номер телефона",
+ "Enter email address": "Введите адрес электронной почты",
+ "The %(capability)s capability": "%(capability)s возможности",
+ "sends confetti": "отправить конфетти",
+ "Invalid URL": "Неправильный URL-адрес",
+ "Reason (optional)": "Причина (необязательно)",
+ "Forgot password?": "Забыли пароль?",
+ "About homeservers": "О домашних серверах",
+ "Learn more": "Узнать больше",
+ "Other homeserver": "Другой домашний сервер",
+ "Server Options": "Параметры сервера",
+ "Decline All": "Отклонить все",
+ "Homeserver": "Домашний сервер",
+ "Approve": "Одобрить",
+ "Approve widget permissions": "Одобрить разрешения виджета",
+ "Send stickers into your active room": "Отправить стикеры в активную комнату",
+ "Remain on your screen while running": "Оставаться на экране во время работы",
+ "Remain on your screen when viewing another room, when running": "Оставаться на экране, при отображании другой комнаты, во время работы",
+ "Effects": "Эффекты",
+ "Zimbabwe": "Зимбабве",
+ "Zambia": "Замбия",
+ "Yemen": "Йемен",
+ "Western Sahara": "Западная Сахара",
+ "Wallis & Futuna": "Уоллис и Футуна",
+ "Vietnam": "Вьетнам",
+ "Venezuela": "Венесуэла",
+ "Vatican City": "Ватикан",
+ "Vanuatu": "Вануату",
+ "Uzbekistan": "Узбекистан",
+ "Uruguay": "Уругвай",
+ "United Arab Emirates": "Объединенные Арабские Эмираты",
+ "Ukraine": "Украина",
+ "Uganda": "Уганда",
+ "U.S. Virgin Islands": "Виргинские острова (США)",
+ "Tuvalu": "Тувалу",
+ "Turks & Caicos Islands": "Острова Теркс и Кайкос",
+ "Turkmenistan": "Туркменистан",
+ "Turkey": "Турция",
+ "Tunisia": "Тунис",
+ "Trinidad & Tobago": "Тринидад и Тобаго",
+ "Tonga": "Тонга",
+ "Tokelau": "Токелау",
+ "Togo": "Того",
+ "Timor-Leste": "Тимор-Лешти",
+ "Thailand": "Таиланд",
+ "Tanzania": "Танзания",
+ "Tajikistan": "Таджикистан",
+ "Taiwan": "Тайвань",
+ "São Tomé & Príncipe": "Сан-Томе и Принсипи",
+ "Syria": "Сирия",
+ "Switzerland": "Швейцария",
+ "Sweden": "Швеция",
+ "Swaziland": "Эсватини",
+ "Svalbard & Jan Mayen": "Шпицберген и Ян-Майен",
+ "Suriname": "Суринам",
+ "Sudan": "Судан",
+ "St. Vincent & Grenadines": "Сент-Винсент и Гренадины",
+ "St. Pierre & Miquelon": "Сен-Пьер и Микелон",
+ "St. Martin": "Сен-Мартен",
+ "St. Lucia": "Сент-Люсия",
+ "St. Kitts & Nevis": "Сент-Китс и Невис",
+ "St. Helena": "Остров Святой Елены",
+ "St. Barthélemy": "Сен-Бартелеми",
+ "Sri Lanka": "Шри-Ланка",
+ "Spain": "Испания",
+ "South Sudan": "Южный Судан",
+ "South Korea": "Южная Корея",
+ "South Georgia & South Sandwich Islands": "Южная Георгия и Южные Сандвичевы острова",
+ "South Africa": "Южная Африка",
+ "Somalia": "Сомали",
+ "Solomon Islands": "Соломоновы острова",
+ "Slovenia": "Словения",
+ "Slovakia": "Словакия",
+ "Sint Maarten": "Синт-Мартен",
+ "Singapore": "Сингапур",
+ "Sierra Leone": "Сьерра-Леоне",
+ "Seychelles": "Сейшельские острова",
+ "Serbia": "Сербия",
+ "Senegal": "Сенегал",
+ "Saudi Arabia": "Саудовская Аравия",
+ "San Marino": "Сан-Марино",
+ "Samoa": "Самоа",
+ "Réunion": "Реюньон",
+ "Rwanda": "Руанда",
+ "Russia": "Российская Федерация",
+ "Romania": "Румыния",
+ "Qatar": "Катар",
+ "Puerto Rico": "Пуэрто-Рико",
+ "Portugal": "Португалия",
+ "Poland": "Польша",
+ "Pitcairn Islands": "Питкэрн",
+ "Philippines": "Филиппины",
+ "Peru": "Перу",
+ "Paraguay": "Парагвай",
+ "Papua New Guinea": "Папуа - Новая Гвинея",
+ "Panama": "Панама",
+ "Palestine": "Палестина",
+ "Palau": "Палау",
+ "Pakistan": "Пакистан",
+ "Oman": "Оман",
+ "Norway": "Норвегия",
+ "Northern Mariana Islands": "Северные Марианские острова",
+ "North Korea": "Северная Корея",
+ "Norfolk Island": "Остров Норфолк",
+ "Niue": "Ниуэ",
+ "Nigeria": "Нигерия",
+ "Niger": "Нигер",
+ "Nicaragua": "Никарагуа",
+ "New Zealand": "Новая Зеландия",
+ "New Caledonia": "Новая Каледония",
+ "Netherlands": "Нидерланды",
+ "Nepal": "Непал",
+ "Nauru": "Науру",
+ "Namibia": "Намибия",
+ "Myanmar": "Мьянма",
+ "Mozambique": "Мозамбик",
+ "Morocco": "Марокко",
+ "Montserrat": "Монсеррат",
+ "Montenegro": "Черногория",
+ "Mongolia": "Монголия",
+ "Monaco": "Монако",
+ "Moldova": "Молдова",
+ "Micronesia": "Микронезия",
+ "Mexico": "Мексика",
+ "Mayotte": "Майотта",
+ "Mauritius": "Маврикий",
+ "Mauritania": "Мавритания",
+ "Martinique": "Мартиника",
+ "Marshall Islands": "Маршалловы острова",
+ "Malta": "Мальта",
+ "Mali": "Мали",
+ "Maldives": "Мальдивы",
+ "Malaysia": "Малайзия",
+ "Malawi": "Малави",
+ "Madagascar": "Мадагаскар",
+ "Macedonia": "Северная Македония",
+ "Macau": "Макао",
+ "Luxembourg": "Люксембург",
+ "Lithuania": "Литва",
+ "Liechtenstein": "Лихтенштейн",
+ "Libya": "Ливия",
+ "Liberia": "Либерия",
+ "Lesotho": "Лесото",
+ "Lebanon": "Ливан",
+ "Latvia": "Латвия",
+ "Laos": "Лаосская Народно-Демократическая Республика",
+ "Kyrgyzstan": "Кыргызстан",
+ "Kuwait": "Кувейт",
+ "Kosovo": "Косово",
+ "Kiribati": "Кирибати",
+ "Kenya": "Кения",
+ "Kazakhstan": "Казахстан",
+ "Jordan": "Иордания",
+ "Jersey": "Джерси",
+ "Japan": "Япония",
+ "Jamaica": "Ямайка",
+ "Italy": "Италия",
+ "Israel": "Израиль",
+ "Isle of Man": "Остров Мэн",
+ "Ireland": "Ирландия",
+ "Iraq": "Ирак",
+ "Iran": "Иран",
+ "Indonesia": "Индонезия",
+ "India": "Индия",
+ "Iceland": "Исландия",
+ "Hungary": "Венгрия",
+ "Hong Kong": "Гонконг",
+ "Honduras": "Гондурас",
+ "Heard & McDonald Islands": "Остров Херд и Острова Макдоналд",
+ "Haiti": "Гаити",
+ "Guyana": "Гайана",
+ "Guinea-Bissau": "Гвинея-Бисау",
+ "Guinea": "Гвинея",
+ "Guernsey": "Гернси",
+ "Guatemala": "Гватемала",
+ "Guam": "Гуам",
+ "Guadeloupe": "Гваделупа",
+ "Grenada": "Гренада",
+ "Greenland": "Гренландия",
+ "Greece": "Греция",
+ "Gibraltar": "Гибралтар",
+ "Ghana": "Гана",
+ "Germany": "Германия",
+ "Georgia": "Грузия",
+ "Gambia": "Гамбия",
+ "Gabon": "Габон",
+ "French Southern Territories": "Южные Французские Территории",
+ "French Polynesia": "Французская Полинезия",
+ "French Guiana": "Французская Гвиана",
+ "France": "Франция",
+ "Finland": "Финляндия",
+ "Fiji": "Фиджи",
+ "Faroe Islands": "Фарерские острова",
+ "Falkland Islands": "Фолклендские острова",
+ "Ethiopia": "Эфиопия",
+ "Estonia": "Эстония",
+ "Eritrea": "Еритрея",
+ "Equatorial Guinea": "Экваториальная Гвинея",
+ "El Salvador": "Сальвадор",
+ "Egypt": "Египет",
+ "Ecuador": "Эквадор",
+ "Dominican Republic": "Доминиканская Республика",
+ "Dominica": "Доминика",
+ "Djibouti": "Джибути",
+ "Denmark": "Дания",
+ "Côte d’Ivoire": "Кот-д'Ивуар",
+ "Czech Republic": "Чехия",
+ "Cyprus": "Кипр",
+ "Curaçao": "Кюрасао",
+ "Cuba": "Куба",
+ "Croatia": "Хорватия",
+ "Costa Rica": "Коста-Рика",
+ "Cook Islands": "Острова Кука",
+ "Congo - Kinshasa": "Демократическая Республика Конго",
+ "Congo - Brazzaville": "Конго",
+ "Comoros": "Коморские острова",
+ "Colombia": "Колумбия",
+ "Cocos (Keeling) Islands": "Кокосовые (Килинг) острова",
+ "Christmas Island": "Остров Рождества",
+ "China": "Китай",
+ "Chile": "Чили",
+ "Chad": "Чад",
+ "Central African Republic": "Центрально-Африканская Республика",
+ "Cayman Islands": "Каймановы острова",
+ "Caribbean Netherlands": "Бонайре, Синт-Эстатиус и Саба",
+ "Cape Verde": "Кабо-Верде",
+ "Canada": "Канада",
+ "Cameroon": "Камерун",
+ "Cambodia": "Камбоджа",
+ "Burundi": "Бурунди",
+ "Burkina Faso": "Буркина-Фасо",
+ "Bulgaria": "Болгария",
+ "Brunei": "Бруней",
+ "British Virgin Islands": "Британские Виргинские острова",
+ "British Indian Ocean Territory": "Британская территория Индийского океана",
+ "Brazil": "Бразилия",
+ "Bouvet Island": "Остров Буве",
+ "Botswana": "Ботсвана",
+ "Bosnia": "Босния и Герцеговина",
+ "Bolivia": "Боливия",
+ "Bhutan": "Бутан",
+ "Bermuda": "Бермудские острова",
+ "Benin": "Бенин",
+ "Belize": "Белиз",
+ "Belgium": "Бельгия",
+ "Belarus": "Беларусь",
+ "Barbados": "Барбадос",
+ "Bangladesh": "Бангладеш",
+ "Bahrain": "Бахрейн",
+ "Bahamas": "Багамские острова",
+ "Azerbaijan": "Азербайджан",
+ "Austria": "Австрия",
+ "Australia": "Австралия",
+ "Aruba": "Аруба",
+ "Armenia": "Армения",
+ "Argentina": "Аргентина",
+ "Antigua & Barbuda": "Антигуа и Барбуда",
+ "Antarctica": "Антарктика",
+ "Anguilla": "Ангилья",
+ "Angola": "Ангола",
+ "Andorra": "Андорра",
+ "American Samoa": "Американское Самоа",
+ "Algeria": "Алжир",
+ "Albania": "Албания",
+ "Åland Islands": "Аландские острова",
+ "Afghanistan": "Афганистан",
+ "United States": "Соединенные Штаты Америки",
+ "United Kingdom": "Великобритания",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Вызов не удался, потому что не удалось получить доступ к веб-камере или микрофону. Проверь это:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Вызов не удался из-за отсутствия доступа к микрофону. Убедитесь, что микрофон подключен и правильно настроен.",
+ "See %(msgtype)s messages posted to your active room": "Посмотрите %(msgtype)s сообщения, размещённые в вашей активной комнате",
+ "Send general files as you in your active room": "Отправьте файлы от своего имени в активной комнате",
+ "Change the topic of your active room": "Измените тему вашей активной комнаты",
+ "See when the topic changes in this room": "Посмотрите, изменится ли тема этого чата",
+ "Change the topic of this room": "Измените тему этой комнаты",
+ "Change which room you're viewing": "Измените комнату, которую вы просматриваете",
+ "See %(msgtype)s messages posted to this room": "Посмотрите %(msgtype)s сообщения размещённые в этой комнате",
+ "Send %(msgtype)s messages as you in your active room": "Отправьте %(msgtype)s сообщения от своего имени в вашу активную комнату",
+ "Send %(msgtype)s messages as you in this room": "Отправьте %(msgtype)s сообщения от своего имени в эту комнату",
+ "See general files posted to your active room": "Посмотрите файлы, размещённые в вашей активной комнате",
+ "See general files posted to this room": "Посмотрите файлы, размещённые в этой комнате",
+ "Send general files as you in this room": "Отправьте файлы от своего имени в этой комнате",
+ "See videos posted to your active room": "Посмотрите видео размещённые в вашей активной комнате",
+ "See videos posted to this room": "Посмотрите видео размещённые в этой комнате",
+ "Send videos as you in your active room": "Отправьте видео от своего имени в вашей активной комнате",
+ "Send videos as you in this room": "Отправьте видео от своего имени в этой комнате",
+ "See images posted to this room": "Посмотрите изображения, размещённые в этой комнате",
+ "See emotes posted to your active room": "Посмотрите эмоции, размещённые в вашей активной комнате",
+ "See emotes posted to this room": "Посмотрите эмоции, размещённые в этой комнате",
+ "See text messages posted to your active room": "Посмотрите текстовые сообщения, размещённые в вашей активной комнате",
+ "See text messages posted to this room": "Посмотрите текстовые сообщения, размещённые в этой комнате",
+ "See messages posted to your active room": "Посмотрите сообщения, размещённые в вашей активной комнате",
+ "See messages posted to this room": "Посмотрите сообщения, размещённые в этой комнате",
+ "See %(eventType)s events posted to your active room": "Посмотрите %(eventType)s события, размещённые в вашей активной комнате",
+ "See %(eventType)s events posted to this room": "Посмотрите %(eventType)s события, размещённые в этой комнате",
+ "See when anyone posts a sticker to your active room": "Посмотрите, когда кто-нибудь размещает стикер в вашей активной комнате",
+ "See when a sticker is posted in this room": "Посмотрите, когда в этой комнате размещается стикер",
+ "See images posted to your active room": "Посмотрите изображения, размещённые в вашей активной комнате",
+ "Send images as you in your active room": "Отправьте изображения от своего имени в свою активную комнату",
+ "Send images as you in this room": "Отправьте изображения от своего имени в эту комнату",
+ "Send emotes as you in your active room": "Отправляйте эмоции от своего имени в активную комнату",
+ "Send emotes as you in this room": "Отправляйте эмоции от своего имени в эту комнату",
+ "Send text messages as you in your active room": "Отправляйте текстовые сообщения от своего имени в активную комнату",
+ "Send text messages as you in this room": "Отправляйте текстовые сообщения от своего имени в этой комнате",
+ "Send messages as you in your active room": "Отправляйте сообщения от своего имени в вашу активную комнату",
+ "Send messages as you in this room": "Отправляйте сообщения от своего имени в этой комнате",
+ "Send %(eventType)s events as you in your active room": "Отправляйте %(eventType)s события от своего имени в вашей активной комнате",
+ "Send %(eventType)s events as you in this room": "Отправляйте события %(eventType)s от своего имени в этой комнате",
+ "Send stickers to your active room as you": "Отправьте стикер от своего имени в активную комнату",
+ "Send stickers to this room as you": "Отправьте стикеры от своего имени в эту комнату",
+ "with an empty state key": "с пустым ключом состояния",
+ "See when the avatar changes in your active room": "Посмотрите, когда изменится аватар в вашей активной комнате",
+ "See when the avatar changes in this room": "Посмотрите, когда изменится аватар в этой комнате",
+ "See when the name changes in your active room": "Посмотрите, когда изменится название в вашей активной комнате",
+ "See when the name changes in this room": "Посмотрите, когда изменится название этой комнаты",
+ "Change the avatar of your active room": "Измените аватар вашей активной комнаты",
+ "Change the avatar of this room": "Смените аватар этой комнаты",
+ "Change the name of this room": "Измените название этой комнаты",
+ "Change the name of your active room": "Измените название вашей активной комнаты",
+ "See when the topic changes in your active room": "Посмотрите, изменится ли тема текущего активного чата",
+ "This widget would like to:": "Этому виджету хотелось бы:",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Вы можете использовать настраиваемые параметры сервера для входа на другие серверы Matrix, указав другой URL-адрес домашнего сервера. Это позволяет вам использовать Element с существующей учётной записью Matrix на другом домашнем сервере.",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Предупреждаем: если вы не добавите адрес электронной почты и забудете пароль, вы можете навсегда потерять доступ к своей учётной записи.",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org - крупнейший в мире домашний публичный сервер, который подходит многим.",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Если вы предпочитаете домашний сервер Matrix, используйте его. Вы также можете настроить свой собственный домашний сервер, если хотите.",
+ "That phone number doesn't look quite right, please check and try again": "Этот номер телефона неправильный, проверьте его и повторите попытку",
+ "Add an email to be able to reset your password.": "Чтобы иметь возможность изменить свой пароль в случае необходимости, добавьте свой адрес электронной почты.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Если вы хотите, чтобы другие пользователи могли вас найти, укажите свой адрес электронной почты или номер телефона.",
+ "Use email to optionally be discoverable by existing contacts.": "Если вы хотите, чтобы другие пользователи могли вас найти, укажите свой адрес электронной почты.",
+ "There was a problem communicating with the homeserver, please try again later.": "Возникла проблема при обмене данными с домашним сервером. Повторите попытку позже.",
+ "That username already exists, please try another.": "Это имя пользователя уже существует, попробуйте другое.",
+ "Already have an account? Sign in here": "Уже есть учётная запись? Войдите здесь",
+ "Decide where your account is hosted": "Выберите, кто обслуживает вашу учётную запись",
+ "We call the places where you can host your account ‘homeservers’.": "Мы называем места, где вы можете разместить свою учётную запись, 'домашними серверами'.",
+ "Sends the given message with confetti": "Отправляет данное сообщение с конфетти",
+ "Hold": "Удерживать",
+ "Resume": "Возобновить",
+ "%(peerName)s held the call": "%(peerName)s удерживает звонок",
+ "You held the call Resume": "Вы удерживаете звонок Возобновить",
+ "%(name)s paused": "%(name)s приостановлен",
+ "You've reached the maximum number of simultaneous calls.": "Вы достигли максимального количества одновременных звонков.",
+ "Too Many Calls": "Слишком много звонков"
}
diff --git a/src/i18n/strings/sl.json b/src/i18n/strings/sl.json
index 2024821bb2..0e9bdb3d3e 100644
--- a/src/i18n/strings/sl.json
+++ b/src/i18n/strings/sl.json
@@ -25,5 +25,7 @@
"Call Declined": "Klic zavrnjen",
"Call Failed": "Klic ni uspel",
"Your homeserver's URL": "URL domačega strežnika",
- "End": "Konec"
+ "End": "Konec",
+ "Use default": "Uporabi privzeto",
+ "Change": "Sprememba"
}
diff --git a/src/i18n/strings/sq.json b/src/i18n/strings/sq.json
index 82644d85a3..827df3fb70 100644
--- a/src/i18n/strings/sq.json
+++ b/src/i18n/strings/sq.json
@@ -874,7 +874,7 @@
"Incompatible Database": "Bazë të dhënash e Papërputhshme",
"Continue With Encryption Disabled": "Vazhdo Me Fshehtëzimin të Çaktivizuar",
"Unable to load! Check your network connectivity and try again.": "S’arrihet të ngarkohet! Kontrolloni lidhjen tuaj në rrjet dhe riprovoni.",
- "Forces the current outbound group session in an encrypted room to be discarded": "Forces the current outbound group session in an encrypted room to be discarded",
+ "Forces the current outbound group session in an encrypted room to be discarded": "",
"Delete Backup": "Fshije Kopjeruajtjen",
"Unable to load key backup status": "S’arrihet të ngarkohet gjendje kopjeruajtjeje kyçesh",
"Backup version: ": "Version kopjeruajtjeje: ",
@@ -1351,7 +1351,7 @@
"Ensure you have a stable internet connection, or get in touch with the server admin": "Sigurohuni se keni një lidhje të qëndrueshme internet, ose lidhuni me përgjegjësin e shërbyesit",
"Your %(brand)s is misconfigured": "%(brand)s-i juaj është i keqformësuar",
"Ask your %(brand)s admin to check your config for incorrect or duplicate entries.": "Kërkojini përgjegjësit të %(brand)s-it tuaj të kontrollojë formësimin tuaj për zëra të pasaktë ose të përsëdytur.",
- "Unexpected error resolving identity server configuration": "Gabim i papritur teksa ftillohej formësimi i shërbyesit të identiteteve",
+ "Unexpected error resolving identity server configuration": "Gabim i papritur teksa ftillohej formësimi i shërbyesit të identiteteve",
"Use lowercase letters, numbers, dashes and underscores only": "Përdorni vetëm shkronja të vogla, numra, vija ndarëse dhe nënvija",
"Cannot reach identity server": "S’kapet dot shërbyesi i identiteteve",
"You can register, but some features will be unavailable until the identity server is back online. If you keep seeing this warning, check your configuration or contact a server admin.": "Mund të regjistroheni, por disa veçori do të jenë të papërdorshme, derisa shërbyesi i identiteteve të jetë sërish në linjë. Nëse vazhdoni ta shihni këtë sinjalizim, kontrolloni formësimin tuaj ose lidhuni me një përgjegjës të shërbyesit.",
@@ -1400,7 +1400,7 @@
"Command Help": "Ndihmë Urdhri",
"Identity Server": "Shërbyes Identitetesh",
"Find others by phone or email": "Gjeni të tjerë përmes telefoni ose email-i",
- "Be found by phone or email": "Bëhuni i gjetshëm përmes telefoni ose email-i",
+ "Be found by phone or email": "Bëhuni i gjetshëm përmes telefoni ose email-i",
"Use bots, bridges, widgets and sticker packs": "Përdorni robotë, ura, widget-e dhe paketa ngjitësish",
"Terms of Service": "Kushte Shërbimi",
"Service": "Shërbim",
@@ -1636,7 +1636,7 @@
"%(brand)s URL": "URL %(brand)s-i",
"Room ID": "ID dhome",
"Widget ID": "ID widget-i",
- "Using this widget may share data with %(widgetDomain)s & your Integration Manager.": "Përdorimi i këtij widget-i mund të sjellë ndarje të dhënash me %(widgetDomain)s & Përgjegjësin tuaj të Integrimeve.",
+ "Using this widget may share data with %(widgetDomain)s & your Integration Manager.": "Përdorimi i këtij widget-i mund të sjellë ndarje të dhënash me %(widgetDomain)s & Përgjegjësin tuaj të Integrimeve.",
"Using this widget may share data with %(widgetDomain)s.": "Përdorimi i këtij widget-i mund të sjellë ndarje të dhënash me %(widgetDomain)s.",
"Widget added by": "Widget i shtuar nga",
"This widget may use cookies.": "Ky widget mund të përdorë cookies.",
@@ -2139,7 +2139,7 @@
"Enter a recovery passphrase": "Jepni një frazëkalim rimarrjesh",
"Enter your recovery passphrase a second time to confirm it.": "Për ta ripohuar, jepeni edhe një herë frazëkalimin tuaj të rimarrjeve.",
"Confirm your recovery passphrase": "Ripohoni frazëkalimin tuaj të rimarrjeve",
- "Your recovery key is a safety net - you can use it to restore access to your encrypted messages if you forget your recovery passphrase.": "Kyçi juaj i rimarrjeve është një rrjet sigurie - mund ta përdorni të të rifituar hyrje te mesazhet tuaj të fshehtëzuar, nëse harroni frazëkalimin tuaj të rimarrjeve.",
+ "Your recovery key is a safety net - you can use it to restore access to your encrypted messages if you forget your recovery passphrase.": "Kyçi juaj i rimarrjeve është një rrjet sigurie - mund ta përdorni për të rifituar hyrje te mesazhet tuaj të fshehtëzuar, nëse harroni frazëkalimin tuaj të rimarrjeve.",
"We'll store an encrypted copy of your keys on our server. Secure your backup with a recovery passphrase.": "Do të ruajmë një kopje të fshehtëzuar të kyçeve tuaj në shërbyesin tonë. Siguroni kopjeruajtjen tuaj me një frazëkalim rimarrjesh.",
"Please enter your recovery passphrase a second time to confirm.": "Ju lutemi, jepeni frazëkalimin tuaj të rimarrjeve edhe një herë, për ta ripohuar.",
"Repeat your recovery passphrase...": "Përsëritni frazëkalimin tuaj të rimarrjeve…",
@@ -2768,7 +2768,7 @@
"Honduras": "Honduras",
"Japan": "Japoni",
"American Samoa": "Samoa Amerikane",
- "South Georgia & South Sandwich Islands": "Xhorxhia Jugore dhe Ishujt Snduiç të Jugut",
+ "South Georgia & South Sandwich Islands": "Xhorxhia Jugore dhe Ishujt Sanduiç të Jugut",
"Palestine": "Palestinë",
"Austria": "Austri",
"Suriname": "Surinam",
@@ -2834,5 +2834,134 @@
"Mauritania": "Mauritani",
"Bangladesh": "Bangladesh",
"Falkland Islands": "Ishujt Falkland",
- "Sweden": "Suedi"
+ "Sweden": "Suedi",
+ "Filter rooms and people": "Filtroni dhoma dhe njerëz",
+ "Open the link in the email to continue registration.": "Që të vazhdohet regjistrimi, hapni lidhjen te email-i.",
+ "A confirmation email has been sent to %(emailAddress)s": "Te %(emailAddress)s u dërgua një email ripohimi",
+ "Role": "Rol",
+ "Start a new chat": "Nisni një fjalosje të re",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Ruajini lokalisht në fshehtinë në mënyrë të sigurt mesazhet e fshehtëzuar, që të shfaqen në përfundime kërkimi, duke përdorur %(size)s që të depozitoni mesazhe nga %(rooms)s dhomë.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Ruajini lokalisht në fshehtinë në mënyrë të sigurt mesazhet e fshehtëzuar, që të shfaqen në përfundime kërkimi, duke përdorur %(size)s që të depozitoni mesazhe nga %(rooms)s dhoma.",
+ "See emotes posted to your active room": "Shihni emotikonë postuar në dhomën tuaj aktive",
+ "See emotes posted to this room": "Shihni emotikone postuar në këtë dhomë",
+ "Send emotes as you in your active room": "Dërgoni emotikone si ju në këtë dhomë",
+ "Send emotes as you in this room": "Dërgoni emotikone si ju në këtë dhomë",
+ "See text messages posted to your active room": "Shihni mesazhe tekst postuar në dhomën tuaj aktive",
+ "See text messages posted to this room": "Shihni mesazhe tekst postuar në këtë dhomë",
+ "Send text messages as you in your active room": "Dërgoni mesazhe tekst si ju në dhomën tuaj aktive",
+ "Send text messages as you in this room": "Dërgoni mesazhe tekst si ju në këtë dhomë",
+ "See messages posted to your active room": "Shihni mesazhe të postuar në dhomën tuaj aktive",
+ "See messages posted to this room": "Shihni mesazhe të postuar në këtë dhomë",
+ "Send messages as you in your active room": "Dërgoni mesazhe si ju në dhomën tuaj aktive",
+ "Send messages as you in this room": "Dërgoni mesazhi si ju në këtë dhomë",
+ "The %(capability)s capability": "Aftësia %(capability)s",
+ "See %(eventType)s events posted to your active room": "Shihni akte %(eventType)s postuar në dhomën tuaj aktive",
+ "Send %(eventType)s events as you in your active room": "Shihni akte %(eventType)s si ju në këtë dhomë",
+ "See %(eventType)s events posted to this room": "Shihni akte %(eventType)s postuar në këtë dhomë",
+ "Send %(eventType)s events as you in this room": "Dërgoni akte %(eventType)s në këtë dhomë si ju",
+ "with state key %(stateKey)s": "me kyç gjendjeje %(stateKey)s",
+ "with an empty state key": "me një kyç të zbrazët gjendjeje",
+ "See when anyone posts a sticker to your active room": "Shihni kur dikush poston një ngjitës në dhomën tuaj aktive",
+ "Send stickers to your active room as you": "Dërgoni ngjitës në dhomën tuaj aktive si ju",
+ "See when a sticker is posted in this room": "Shihni kur postohet një ngjitës në këtë dhomë",
+ "Send stickers to this room as you": "Dërgoni ngjitës në këtë dhomë si ju",
+ "See when the avatar changes in your active room": "Shihni kur ndryshon avatari në dhomën tuaj aktive",
+ "Change the avatar of your active room": "Ndryshoni avatarin në dhomën tuaj aktive",
+ "See when the avatar changes in this room": "Shihni kur ndryshon avatari në këtë dhomë",
+ "Change the avatar of this room": "Ndryshoni avatarin e kësaj dhome",
+ "See when the name changes in your active room": "Shihni kur ndryshon emri në dhomën tuaj aktive",
+ "Change the name of your active room": "Ndryshoni emrin e dhomës tuaj aktive",
+ "See when the name changes in this room": "Shihni kur ndryshohet emri në këtë dhomë",
+ "Change the name of this room": "Ndryshoni emrin e kësaj dhome",
+ "See when the topic changes in your active room": "Shihni kur ndryshon tema në dhomën tuaj aktive",
+ "Change the topic of your active room": "Ndryshoni temën në dhomën tuaj aktive",
+ "See when the topic changes in this room": "Shihni kur ndryshohet tema në këtë dhomë",
+ "Change the topic of this room": "Ndryshoni temën e kësaj dhome",
+ "Change which room you're viewing": "Ndryshoni cilën dhomë shihni",
+ "Send stickers into your active room": "Dërgoni ngjitës në dhomën tuaj aktive",
+ "Send stickers into this room": "Dërgoni ngjitës në këtë dhomë",
+ "Go to Home View": "Kaloni te Pamja Kreu",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n
\n You can even add images with Matrix URLs \n
\n": "
HTML për faqen e bashkësisë tuaj
\n
\n Përdoreni përshkrimin e gjatë që t’i prezantoni bashkësisë anëtarë të rinj, ose për t’u dhënë lidhje të rëndësishme\n
\n
\n Mundeni madje të shtoni figura me URL-ra Matrix \n
\n",
+ "Enter phone number": "Jepni numër telefoni",
+ "Enter email address": "Jepni adresë email-i",
+ "Decline All": "Hidhi Krejt Poshtë",
+ "Approve": "Miratojeni",
+ "This widget would like to:": "Ky widget do të donte të:",
+ "Approve widget permissions": "Miratoni leje widget-i",
+ "Use Ctrl + Enter to send a message": "Që të dërgoni një mesazh përdorni tastet Ctrl + Enter",
+ "Use Command + Enter to send a message": "Që të dërgoni një mesazh, përdorni tastet Command + Enter",
+ "See %(msgtype)s messages posted to your active room": "Shihni mesazhe %(msgtype)s postuar në dhomën tuaj aktive",
+ "See %(msgtype)s messages posted to this room": "Shihni mesazhe %(msgtype)s postuar në këtë dhomë",
+ "Send %(msgtype)s messages as you in your active room": "Dërgoni mesazhe %(msgtype)s si ju në dhomën tuaj aktive",
+ "Send %(msgtype)s messages as you in this room": "Dërgoni mesazhe %(msgtype)s si ju në këtë dhomë",
+ "See general files posted to your active room": "Shihni kartela të përgjithshme postuar në dhomën tuaj aktive",
+ "See general files posted to this room": "Shihni kartela të përgjithshme postuar në këtë dhomë",
+ "Send general files as you in your active room": "Dërgoni kartela të përgjithshme si ju në dhomën tuaj aktive",
+ "Send general files as you in this room": "Dërgoni kartela të përgjithshme si ju në këtë dhomë",
+ "See videos posted to your active room": "Shihni video të postuara në dhomën tuaj aktive",
+ "See videos posted to this room": "Shihni video të postuara në këtë dhomë",
+ "Send videos as you in your active room": "Dërgoni video si ju në dhomën tuaj aktive",
+ "Send videos as you in this room": "Dërgoni video si ju në këtë dhomë",
+ "See images posted to your active room": "Shihni figura postuar te dhoma juaj aktive",
+ "See images posted to this room": "Shihni figura postuar në këtë dhomë",
+ "Send images as you in your active room": "Dërgoni figura si ju në dhomën tuaj aktive",
+ "New here? Create an account": "I sapoardhur? Krijoni një llogari",
+ "Got an account? Sign in": "Keni një llogari? Hyni",
+ "Return to call": "Kthehu te thirrja",
+ "Fill Screen": "Mbushe Ekranin",
+ "Voice Call": "Thirrje Zanore",
+ "Video Call": "Thirrje Video",
+ "Render LaTeX maths in messages": "Formo formula LaTeX në mesazhe",
+ "Send images as you in this room": "Dërgoni figura si ju, në këtë dhomë",
+ "No other application is using the webcam": "Kamerën s’po e përdor aplikacion tjetër",
+ "Permission is granted to use the webcam": "Është dhënë leje për përdorimin e kamerës",
+ "A microphone and webcam are plugged in and set up correctly": "Një mikrofon dhe një kamerë janë futur dhe ujdisur si duhet",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "Thirrja dështoi, ngaqë s’u bë dot hyrje në kamerë ose mikrofon. Kontrolloni që:",
+ "Unable to access webcam / microphone": "S’arrihet të përdoret kamerë / mikrofon",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Thirrja dështoi, ngaqë s’u hap dot ndonjë mikrofon. Shihni që të jetë futur një mikrofon dhe ujdiseni saktë.",
+ "Unable to access microphone": "S’arrihet të përdoret mikrofoni",
+ "Decide where your account is hosted": "Vendosni se ku të ruhet llogaria juaj",
+ "Host account on": "Strehoni llogari në",
+ "Already have an account? Sign in here": "Keni tashmë një llogari? Bëni hyrjen këtu",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s Ose %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "Vazhdo me %(ssoButtons)s",
+ "That username already exists, please try another.": "Ka tashmë një emër përdoruesi të tillë, ju lutemi, provoni një tjetër.",
+ "New? Create account": "I ri? Krijoni llogari",
+ "There was a problem communicating with the homeserver, please try again later.": "Pati një problem në komunikimin me shërbyesin Home, ju lutemi, riprovoni më vonë.",
+ "Use email to optionally be discoverable by existing contacts.": "Përdorni email që, nëse doni, të mund t’ju gjejnë kontaktet ekzistues.",
+ "Use email or phone to optionally be discoverable by existing contacts.": "Përdorni email ose telefon që, nëse doni, të mund t’ju gjejnë kontaktet ekzistues.",
+ "Add an email to be able to reset your password.": "Shtoni një email, që të jeni në gjendje të ricaktoni fjalëkalimin tuaj.",
+ "Forgot password?": "Harruat fjalëkalimin?",
+ "That phone number doesn't look quite right, please check and try again": "Ai numër telefoni s’duket i saktë, ju lutemi, rikontrollojeni dhe riprovojeni",
+ "About homeservers": "Mbi shërbyesit Home",
+ "Learn more": "Mësoni më tepër",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "Përdorni shërbyesin tuaj Home të parapëlqyer Matrix, nëse keni një të tillë, ose strehoni një tuajin.",
+ "Other homeserver": "Tjetër shërbyes home",
+ "We call the places you where you can host your account ‘homeservers’.": "Vendet ku mund të strehoni llogarinë tuaj i quajmë “shërbyes Home”.",
+ "Sign into your homeserver": "Bëni hyrjen te shërbyesi juaj Home",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org është shërbyesi Home më i madh në botë, ndaj është një vend i mirë për shumë vetë.",
+ "Specify a homeserver": "Tregoni një shërbyes Home",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "Që mos thoni nuk e dinim, nëse s’shtoni një email dhe harroni fjalëkalimin tuaj, mund të humbi përgjithmonë hyrjen në llogarinë tuaj.",
+ "Continuing without email": "Vazhdim pa email",
+ "Continue with %(provider)s": "Vazhdo me %(provider)s",
+ "Homeserver": "Shërbyes Home",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Mund të përdorni mundësi vetjake shërbyesi që të bëni hyrjen në shërbyes të tjerë Matrix duke dhënë një tjetër URL shërbyesi Home. Kjo ju lejon të përdorni Element-in me një llogari Matrix ekzistuese në një tjetër shërbyes Home.",
+ "Server Options": "Mundësi Shërbyesi",
+ "Hold": "Mbaje",
+ "Resume": "Rimerre",
+ "We call the places where you can host your account ‘homeservers’.": "Vendet ku mund të strehoni llogarinë tuaj i quajmë ‘shërbyes Home’.",
+ "Invalid URL": "URL e pavlefshme",
+ "Unable to validate homeserver": "S’arrihet të vlerësohet shërbyesi Home",
+ "Reason (optional)": "Arsye (opsionale)",
+ "%(name)s paused": "%(name)s pushoi",
+ "%(peerName)s held the call": "%(peerName)s mbajti thirrjen",
+ "You held the call Resume": "E mbajtët thirrjen Rimerreni",
+ "sends confetti": "dërgon bonbone",
+ "Sends the given message with confetti": "E dërgon mesazhin e dhënë me bonbone",
+ "Show chat effects": "Shfaq efekte fjalosjeje",
+ "Effects": "Efekte",
+ "You've reached the maximum number of simultaneous calls.": "Keni mbërritur në numrin maksimum të thirrjeve të njëkohshme.",
+ "Too Many Calls": "Shumë Thirrje",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Thirrja dështoi, ngaqë s’u hy dot kamera ose mikrofoni. Kontrolloni që:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Thirrja dështoi, ngaqë s’u hy dot te mikrofoni. Kontrolloni që të jetë futur një mikrofon dhe të jetë ujdisur saktësisht."
}
diff --git a/src/i18n/strings/sv.json b/src/i18n/strings/sv.json
index f1bf9f08cd..8c053de410 100644
--- a/src/i18n/strings/sv.json
+++ b/src/i18n/strings/sv.json
@@ -87,7 +87,7 @@
"Failed to set display name": "Misslyckades att ange visningsnamn",
"Failed to unban": "Misslyckades att avbanna",
"Failed to verify email address: make sure you clicked the link in the email": "Misslyckades att bekräfta e-postadressen: set till att du klickade på länken i e-postmeddelandet",
- "Favourite": "Favorit",
+ "Favourite": "Favoritmarkera",
"Accept": "Godkänn",
"Access Token:": "Åtkomsttoken:",
"Active call (%(roomName)s)": "Aktiv samtal (%(roomName)s)",
@@ -2506,7 +2506,7 @@
"Brazil": "Brasilien",
"Bouvet Island": "Bouvetön",
"Botswana": "Botswana",
- "Bosnia": "Bosnien",
+ "Bosnia": "Bosnien och Hercegovina",
"Bolivia": "Bolivia",
"Bhutan": "Bhutan",
"Bermuda": "Bermuda",
@@ -2601,7 +2601,7 @@
"Syria": "Syrien",
"Switzerland": "Schweiz",
"Sweden": "Sverige",
- "Swaziland": "Swaziland",
+ "Swaziland": "Eswatini",
"Svalbard & Jan Mayen": "Svalbard och Jan Mayen",
"Suriname": "Surinam",
"Sudan": "Sudan",
@@ -2774,5 +2774,91 @@
"Central African Republic": "Centralafrikanska republiken",
"Cayman Islands": "Caymanöarna",
"Caribbean Netherlands": "Karibiska Nederländerna",
- "Cape Verde": "Kap Verde"
+ "Cape Verde": "Kap Verde",
+ "Change which room you're viewing": "Ändra vilket rum du visar",
+ "Send stickers into your active room": "Skicka in dekaler i ditt aktiva rum",
+ "Send stickers into this room": "Skicka in dekaler i det här rummet",
+ "Remain on your screen while running": "Stanna kvar på skärmen när det körs",
+ "Remain on your screen when viewing another room, when running": "Stanna kvar på skärmen när ett annat rum visas, när det körs",
+ "See when the topic changes in this room": "Se när ämnet ändras i det här rummet",
+ "Change the topic of this room": "Ändra ämnet för det här rummet",
+ "See when the topic changes in your active room": "Se när ämnet ändras i ditt aktiva rum",
+ "Change the topic of your active room": "Ändra ämnet för ditt aktiva rum",
+ "Change the avatar of your active room": "Byta avatar för ditt aktiva rum",
+ "Change the avatar of this room": "Byta avatar för det här rummet",
+ "Change the name of your active room": "Byta namn på ditt aktiva rum",
+ "Change the name of this room": "Byta namn på det här rummet",
+ "See when the avatar changes in your active room": "Se när avataren byts för ditt aktiva rum",
+ "See when the avatar changes in this room": "Se när avataren byts för det här rummet",
+ "See when the name changes in your active room": "Se när namnet på ditt aktiva rum byts",
+ "See when the name changes in this room": "Se när namnet på det här rummet byts",
+ "See text messages posted to your active room": "Se textmeddelanden som skickas i ditt aktiva rum",
+ "See text messages posted to this room": "Se textmeddelanden som skickas i det här rummet",
+ "Send text messages as you in your active room": "Skicka textmeddelanden som dig i ditt aktiva rum",
+ "Send text messages as you in this room": "Skicka textmeddelanden som dig i det här rummet",
+ "See messages posted to your active room": "Se meddelanden som skickas i ditt aktiva rum",
+ "See messages posted to this room": "Se meddelanden som skickas i det här rummet",
+ "Send messages as you in your active room": "Skicka meddelanden som dig i ditt aktiva rum",
+ "Send messages as you in this room": "Skicka meddelanden som dig i det här rummet",
+ "with an empty state key": "med en tom statusnyckel",
+ "The %(capability)s capability": "%(capability)s-kapaciteten",
+ "See %(eventType)s events posted to your active room": "Se %(eventType)s-händelser som skickas i ditt aktiva rum",
+ "Send %(eventType)s events as you in your active room": "Skicka %(eventType)s-händelser som dig i ditt aktiva rum",
+ "See %(eventType)s events posted to this room": "Se %(eventType)s-händelser skickade i det här rummet",
+ "Send %(eventType)s events as you in this room": "Skicka %(eventType)s-händelser som dig i det här rummet",
+ "with state key %(stateKey)s": "med statusnyckel %(stateKey)s",
+ "See when a sticker is posted in this room": "Se när en dekal skickas i det här rummet",
+ "See when anyone posts a sticker to your active room": "Se när någon skickar en dekal till ditt aktiva rum",
+ "Send stickers to your active room as you": "Skicka dekaler till ditt aktiva rum som dig",
+ "Send stickers to this room as you": "Skicka dekaler till det här rummet som dig",
+ "Reason (optional)": "Orsak (valfritt)",
+ "Continue with %(provider)s": "Fortsätt med %(provider)s",
+ "Homeserver": "Hemserver",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "Du kan använda anpassade serveralternativ för att logga in på andra Matrixservrar genom att specificera en annan hemserver-URL. Detta låter dig använda Element med ett existerande Matrix-konto på en annan hemserver.",
+ "Server Options": "Serveralternativ",
+ "Start a new chat": "Starta en ny chatt",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "Cacha på ett säkert sätt krypterade meddelanden lokalt för att de ska visas i sökresultat, och använd %(size)s för att lagra meddelanden från %(rooms)s rum.",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "Cacha på ett säkert sätt krypterade meddelanden lokalt för att de ska visas i sökresultat, och använd %(size)s för att lagra meddelanden från %(rooms)s rum.",
+ "Return to call": "Återgå till samtal",
+ "Fill Screen": "Fyll skärmen",
+ "Voice Call": "Röstsamtal",
+ "Video Call": "Videosamtal",
+ "Use Ctrl + Enter to send a message": "Använd Ctrl + Enter för att skicka ett meddelande",
+ "Use Command + Enter to send a message": "Använd Kommando + Enter för att skicka ett meddelande",
+ "Render LaTeX maths in messages": "Rendera LaTeX-matte i meddelanden",
+ "No other application is using the webcam": "Inget annat program använder webbkameran",
+ "Permission is granted to use the webcam": "Åtkomst till webbkameran har beviljats",
+ "A microphone and webcam are plugged in and set up correctly": "En webbkamera och en mikrofon är inkopplad och korrekt inställd",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "Samtal misslyckades eftersom ingen webbkamera eller mikrofon kunde kommas åt. Kolla att:",
+ "Unable to access webcam / microphone": "Kan inte komma åt webbkamera eller mikrofon",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Samtal misslyckades eftersom ingen mikrofon kunde kommas åt. Kolla att en mikrofon är inkopplad och korrekt inställd.",
+ "Unable to access microphone": "Kan inte komma åt mikrofonen",
+ "See %(msgtype)s messages posted to your active room": "Se %(msgtype)s-meddelanden som skickas i ditt aktiva rum",
+ "See %(msgtype)s messages posted to this room": "Se %(msgtype)s-meddelanden som skickas i det här rummet",
+ "Send %(msgtype)s messages as you in your active room": "Skicka %(msgtype)s-meddelanden som dig i ditt aktiva rum",
+ "Send %(msgtype)s messages as you in this room": "Skicka %(msgtype)s-meddelanden som dig i det här rummet",
+ "See general files posted to your active room": "Se generella filer som skickas i ditt aktiva rum",
+ "See general files posted to this room": "Se generella filer som skickas i det här rummet",
+ "Send general files as you in your active room": "Skicka generella filer som dig i ditt aktiva rum",
+ "Send general files as you in this room": "Skicka generella filer som dig i det här rummet",
+ "See videos posted to your active room": "Se videor som skickas i ditt aktiva rum",
+ "See videos posted to this room": "Se videor som skickas i det här rummet",
+ "Send videos as you in your active room": "Skicka videor som dig i ditt aktiva rum",
+ "Send videos as you in this room": "Skicka videor som dig i det här rummet",
+ "See images posted to your active room": "Se bilder som skickas i ditt aktiva rum",
+ "See images posted to this room": "Se bilder som skickas i det här rummet",
+ "Send images as you in your active room": "Skicka bilder som dig i ditt aktiva rum",
+ "Send images as you in this room": "Skicka bilder som dig i det här rummet",
+ "See emotes posted to your active room": "Se emotes som skickas i ditt aktiva rum",
+ "See emotes posted to this room": "Se emotes som skickas i det här rummet",
+ "Send emotes as you in your active room": "Skicka emotes som dig i ditt aktiva rum",
+ "Send emotes as you in this room": "Skicka emotes som dig i det här rummet",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "En förvarning, om du inte lägger till en e-postadress och glömmer ditt lösenord, så kan du permanent förlora åtkomst till ditt konto.",
+ "Continuing without email": "Fortsätter utan e-post",
+ "sends confetti": "skickar konfetti",
+ "Sends the given message with confetti": "Skickar det givna meddelandet med konfetti",
+ "Show chat effects": "Visa chatteffekter",
+ "Effects": "Effekter",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "Samtal misslyckades eftersom webbkamera eller mikrofon inte kunde kommas åt. Kolla att:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "Samtal misslyckades eftersom att mikrofonen inte kunde kommas åt. Kolla att en mikrofon är inkopplat och korrekt inställd."
}
diff --git a/src/i18n/strings/uk.json b/src/i18n/strings/uk.json
index 4151a3f755..061c0799ac 100644
--- a/src/i18n/strings/uk.json
+++ b/src/i18n/strings/uk.json
@@ -66,14 +66,14 @@
"Can't connect to homeserver - please check your connectivity, ensure your homeserver's SSL certificate is trusted, and that a browser extension is not blocking requests.": "Не вдається підключитись до домашнього серверу - перевірте підключення, переконайтесь, що ваш SSL-сертифікат домашнього сервера є довіреним і що розширення браузера не блокує запити.",
"Cannot add any more widgets": "Неможливо додати більше віджетів",
"Change Password": "Змінити пароль",
- "%(senderName)s changed their profile picture.": "%(senderName)s змінив/ла зображення профілю.",
- "%(senderName)s changed the power level of %(powerLevelDiffText)s.": "%(senderName)s змінив(-ла) рівень повноважень %(powerLevelDiffText)s.",
- "%(senderDisplayName)s changed the room name to %(roomName)s.": "%(senderDisplayName)s змінив/ла назву кімнати на %(roomName)s.",
+ "%(senderName)s changed their profile picture.": "%(senderName)s змінює зображення профілю.",
+ "%(senderName)s changed the power level of %(powerLevelDiffText)s.": "%(senderName)s змінює рівень повноважень %(powerLevelDiffText)s.",
+ "%(senderDisplayName)s changed the room name to %(roomName)s.": "%(senderDisplayName)s змінює назву кімнати на %(roomName)s.",
"%(senderDisplayName)s removed the room name.": "%(senderDisplayName)s видалив ім'я кімнати.",
- "%(senderDisplayName)s changed the topic to \"%(topic)s\".": "%(senderDisplayName)s змінив тему на %(topic)s.",
+ "%(senderDisplayName)s changed the topic to \"%(topic)s\".": "%(senderDisplayName)s змінює тему на %(topic)s.",
"Email": "е-пошта",
"Email address": "Адреса е-пошти",
- "Failed to send email": "Помилка відправки е-почти",
+ "Failed to send email": "Помилка надсилання електронного листа",
"Edit": "Відредагувати",
"Unpin Message": "Відкріпити повідомлення",
"Register": "Зареєструватися",
@@ -124,7 +124,7 @@
"Explore Room State": "Перегляд статуса кімнати",
"Source URL": "Джерельне посилання",
"Messages sent by bot": "Повідомлення, надіслані ботом",
- "Filter results": "Відцідити результати",
+ "Filter results": "Відфільтрувати результати",
"Members": "Учасники",
"No update available.": "Оновлення відсутні.",
"Resend": "Перенадіслати",
@@ -177,7 +177,7 @@
"Can't update user notification settings": "Неможливо оновити налаштування користувацьких сповіщень",
"Notify for all other messages/rooms": "Сповіщати щодо всіх повідомлень/кімнат",
"Unable to look up room ID from server": "Неможливо знайти ID кімнати на сервері",
- "Couldn't find a matching Matrix room": "Неможливо знайти відповідну кімнату",
+ "Couldn't find a matching Matrix room": "Не вдалось знайти відповідну кімнату",
"Invite to this room": "Запросити до цієї кімнати",
"Thursday": "Четвер",
"Search…": "Пошук…",
@@ -326,13 +326,13 @@
"Reason": "Причина",
"%(senderName)s requested a VoIP conference.": "%(senderName)s бажає розпочати дзвінок-конференцію.",
"%(senderName)s invited %(targetName)s.": "%(senderName)s запросив/ла %(targetName)s.",
- "%(oldDisplayName)s changed their display name to %(displayName)s.": "%(oldDisplayName)s змінив(-ла) своє видиме ім'я на %(displayName)s.",
+ "%(oldDisplayName)s changed their display name to %(displayName)s.": "%(oldDisplayName)s змінює своє видиме ім'я на %(displayName)s.",
"%(senderName)s set their display name to %(displayName)s.": "%(senderName)s зазначив(-ла) своє видиме ім'я: %(displayName)s.",
"%(senderName)s removed their display name (%(oldDisplayName)s).": "%(senderName)s видалив(-ла) своє видиме ім'я (%(oldDisplayName)s).",
"%(senderName)s removed their profile picture.": "%(senderName)s вилучав/ла свою світлину профілю.",
- "%(senderName)s set a profile picture.": "%(senderName)s встановив/ла світлину профілю.",
+ "%(senderName)s set a profile picture.": "%(senderName)s встановлює світлину профілю.",
"VoIP conference started.": "Розпочато дзвінок-конференцію.",
- "%(targetName)s joined the room.": "%(targetName)s приєднав/лася до кімнати.",
+ "%(targetName)s joined the room.": "%(targetName)s приєднується до кімнати.",
"VoIP conference finished.": "Дзвінок-конференцію завершено.",
"%(targetName)s rejected the invitation.": "%(targetName)s відкинув/ла запрошення.",
"%(targetName)s left the room.": "%(targetName)s залишив/ла кімнату.",
@@ -347,7 +347,7 @@
"(could not connect media)": "(не можливо під'єднати медіа)",
"(no answer)": "(немає відповіді)",
"(unknown failure: %(reason)s)": "(невідома помилка: %(reason)s)",
- "%(senderName)s ended the call.": "%(senderName)s завершив/ла дзвінок.",
+ "%(senderName)s ended the call.": "%(senderName)s завершує дзвінок.",
"%(senderName)s sent an invitation to %(targetDisplayName)s to join the room.": "%(senderName)s надіслав(-ла) запрошення %(targetDisplayName)s приєднатися до кімнати.",
"Show developer tools": "Показувати розробницькі засоби",
"Default": "Типово",
@@ -357,8 +357,8 @@
"%(senderName)s made future room history visible to anyone.": "%(senderName)s зробив(-ла) майбутню історію кімнати видимою для всіх.",
"%(senderName)s made future room history visible to unknown (%(visibility)s).": "%(senderName)s зробив(-ла) майбутню історію видимою для невідомого значення (%(visibility)s).",
"%(userId)s from %(fromPowerLevel)s to %(toPowerLevel)s": "%(userId)s з %(fromPowerLevel)s до %(toPowerLevel)s",
- "%(senderName)s changed the pinned messages for the room.": "%(senderName)s змінив(-ла) приколоті повідомлення у кімнаті.",
- "%(widgetName)s widget modified by %(senderName)s": "%(senderName)s змінив(-ла) знадіб %(widgetName)s",
+ "%(senderName)s changed the pinned messages for the room.": "%(senderName)s змінює приколоті повідомлення у кімнаті.",
+ "%(widgetName)s widget modified by %(senderName)s": "%(senderName)s змінює знадіб %(widgetName)s",
"%(widgetName)s widget added by %(senderName)s": "%(senderName)s додав(-ла) знадіб %(widgetName)s",
"%(widgetName)s widget removed by %(senderName)s": "%(senderName)s вилучив(-ла) знадіб %(widgetName)s",
"Failure to create room": "Не вдалося створити кімнату",
@@ -658,23 +658,23 @@
"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 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 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 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 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 розпочав(-ла) відеовиклик.",
@@ -780,7 +780,7 @@
"Guest": "Гість",
"There was an error joining the room": "Помилка при вході в кімнату",
"You joined the call": "Ви приєднались до дзвінку",
- "%(senderName)s joined the call": "%(senderName)s приєднався(лась) до дзвінку",
+ "%(senderName)s joined the call": "%(senderName)s приєднується до розмови",
"Call in progress": "Дзвінок у процесі",
"You left the call": "Ви припинили розмову",
"%(senderName)s left the call": "%(senderName)s покинув(ла) дзвінок",
@@ -794,7 +794,7 @@
"%(senderName)s: %(reaction)s": "%(senderName)s: %(reaction)s",
"%(senderName)s: %(stickerName)s": "%(senderName)s: %(stickerName)s",
"Custom user status messages": "користувацький статус",
- "Group & filter rooms by custom tags (refresh to apply changes)": "Групувати та проціджувати кімнати за нетиповими наличками (оновіть щоб застосувати зміни)",
+ "Group & filter rooms by custom tags (refresh to apply changes)": "Групувати та фільтрувати кімнати за нетиповими наличками (оновіть, щоб застосувати зміни)",
"Multiple integration managers": "Декілька менджерів інтеграції",
"Try out new ways to ignore people (experimental)": "Спробуйте нові способи ігнорувати людей (експериментальні)",
"Support adding custom themes": "Підтримка користувацьких тем",
@@ -960,7 +960,7 @@
"Show less": "Згорнути",
"Show more": "Розгорнути",
"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.": "Змінення пароля призведе до скидання всіх ключів наскрізного шифрування та унеможливить читання історії листування, якщо тільки ви не експортуєте ваші ключі кімнати та не імпортуєте їх згодом. Це буде вдосконалено у майбутньому.",
- "Santa": "Санта Клаус",
+ "Santa": "Св. Миколай",
"Gift": "Подарунок",
"Lock": "Замок",
"Cross-signing and secret storage are not yet set up.": "Перехресне підписування та таємне сховище ще не налагоджені.",
@@ -998,7 +998,7 @@
"Disconnect": "Відключити",
"You should:": "Вам варто:",
"Disconnect anyway": "Відключити в будь-якому випадку",
- "Identity Server (%(server)s)": "Сервер ідентифікації (%(server)s)",
+ "Identity Server (%(server)s)": "Сервер ідентифікації (%(server)s)",
"Identity Server": "Сервер ідентифікації",
"Do not use an identity server": "Не використовувати сервер ідентифікації",
"Enter a new identity server": "Введіть новий сервер ідентифікації",
@@ -1017,7 +1017,7 @@
"Unable to revoke sharing for email address": "Не вдалось відкликати оприлюднювання адреси е-пошти",
"Revoke": "Відкликати",
"Unable to revoke sharing for phone number": "Не вдалось відкликати оприлюднювання телефонного номеру",
- "Filter room members": "Відцідити учасників кімнати",
+ "Filter room members": "Відфільтрувати учасників кімнати",
"Voice call": "Голосовий виклик",
"Video call": "Відеовиклик",
"Not now": "Не зараз",
@@ -1101,7 +1101,7 @@
"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": "Поліпшити ваше шифрування",
- "Show a placeholder for removed messages": "Показувати позначку-заповнювач для видалених повідомлень",
+ "Show a placeholder for removed messages": "Показувати замісну позначку замість видалених повідомлень",
"Show join/leave messages (invites/kicks/bans unaffected)": "Показувати повідомлення про приєднання/залишення (не впливає на запрошення/викидання/заборону)",
"Show avatar changes": "Показувати зміни личини",
"Show display name changes": "Показувати зміни видимого імені",
@@ -1121,7 +1121,7 @@
"Are you sure? You will lose your encrypted messages if your keys are not backed up properly.": "Ви впевнені? Ви загубите ваші зашифровані повідомлення якщо копія ключів не була зроблена коректно.",
"Encrypted messages are secured with end-to-end encryption. Only you and the recipient(s) have the keys to read these messages.": "Зашифровані повідомлення захищені наскрізним шифруванням. Лише ви та отримувачі повідомлень мають ключі для їх читання.",
"Display Name": "Видиме ім'я",
- "wait and try again later": "почекайте та спробуйте пізніше",
+ "wait and try again later": "зачекайте та спопробуйте ще раз пізніше",
"Once enabled, encryption for a room cannot be disabled. Messages sent in an encrypted room cannot be seen by the server, only by the participants of the room. Enabling encryption may prevent many bots and bridges from working correctly. Learn more about encryption.": "Якщо ви увімкнете шифрування для кімнати, його неможливо буде вимкнути. Надіслані у зашифровану кімнату повідомлення будуть прочитними тільки для учасників кімнати, натомість для сервера вони будуть непрочитними. Увімкнення шифрування може унеможливити роботу ботів та мостів. Дізнатись більше про шифрування.",
"Encrypted": "Зашифроване",
"This room is end-to-end encrypted": "Ця кімната є наскрізно зашифрованою",
@@ -1182,11 +1182,11 @@
"Emoji": "Емодзі",
"Emoji Autocomplete": "Самодоповнення емодзі",
"%(senderName)s created a ban rule matching %(glob)s for %(reason)s": "%(senderName)s створив(-ла) правило блокування зі збігом з %(glob)s через %(reason)s",
- "%(senderName)s changed a rule that was banning users matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінив(-ла) правило блокування користувачів зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
- "%(senderName)s changed a rule that was banning rooms matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінив(-ла) правило блокування кімнат зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
- "%(senderName)s changed a rule that was banning servers matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінив(-ла) правило блокування серверів зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
- "%(senderName)s updated a ban rule that was matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінив(-ла) правило блокування зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
- "Enable Community Filter Panel": "Увімкнути панель спільнот",
+ "%(senderName)s changed a rule that was banning users matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінює правило блокування користувачів зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
+ "%(senderName)s changed a rule that was banning rooms matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінює правило блокування кімнат зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
+ "%(senderName)s changed a rule that was banning servers matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінює правило блокування серверів зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
+ "%(senderName)s updated a ban rule that was matching %(oldGlob)s to matching %(newGlob)s for %(reason)s": "%(senderName)s змінєю правило блокування зі збігу з %(oldGlob)s на збіг з %(newGlob)s через %(reason)s",
+ "Enable Community Filter Panel": "Увімкнути фільтр панелі спільнот",
"Messages containing my username": "Повідомлення, що містять моє користувацьке ім'я",
"Messages containing @room": "Повідомлення, що містять @room",
"When rooms are upgraded": "Коли кімнати поліпшено",
@@ -1204,8 +1204,8 @@
"You have ignored this user, so their message is hidden. Show anyways.": "Ви ігноруєте цього користувача, тож його повідомлення приховано. Все одно показати.",
"Show all": "Показати все",
"Add an Integration": "Додати інтеграцію",
- "Filter community members": "Відцідити учасників спільноти",
- "Filter community rooms": "Відцідити кімнати спільноти",
+ "Filter community members": "Відфільтрувати учасників спільноти",
+ "Filter community rooms": "Відфільтрувати кімнати спільноти",
"Display your community flair in rooms configured to show it.": "Відбивати ваш спільнотний значок у кімнатах, що налаштовані показувати його.",
"Using this widget may share data with %(widgetDomain)s & your Integration Manager.": "Користування цим знадобом може призвести до поширення ваших даних з %(widgetDomain)s та вашим менеджером інтеграцій.",
"Show advanced": "Показати розширені",
@@ -1219,13 +1219,13 @@
"Did you know: you can use communities to filter your %(brand)s experience!": "Чи знаєте ви, що спільноти можна використовувати для припасування %(brand)s під ваші потреби?",
"Communities": "Спільноти",
"Create a new community": "Створити нову спільноту",
- "Clear filter": "Очистити цідило",
+ "Clear filter": "Очистити фільтр",
"Syncing...": "Синхронізування…",
"Signing In...": "Входження…",
"If you've joined lots of rooms, this might take a while": "Якщо ви приєднались до багатьох кімнат, це може зайняти деякий час",
"Create account": "Створити обліковий запис",
"Failed to fetch avatar URL": "Не вдалось вибрати URL личини",
- "Clear room list filter field": "Очистити поле цідила списку кімнат",
+ "Clear room list filter field": "Очистити поле фільтра списку кімнат",
"Cancel autocomplete": "Скасувати самодоповнення",
"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.": "Журнали зневадження містять дані використання застосунку, включно з вашим користувацьким ім’ям, ідентифікаторами або псевдонімами відвіданих вами кімнат або груп, а також іменами інших користувачів. Вони не містять повідомлень.",
"Confirm your account deactivation by using Single Sign On to prove your identity.": "Підтвердьте знедіяння вашого облікового запису через Single Sign On щоб підтвердити вашу особу.",
@@ -1264,5 +1264,277 @@
"The other party cancelled the verification.": "Друга сторона скасувала звірення.",
"Verified!": "Звірено!",
"You've successfully verified this user.": "Ви успішно звірили цього користувача.",
- "Got It": "Зрозуміло"
+ "Got It": "Зрозуміло",
+ "Comoros": "Коморські Острови",
+ "Colombia": "Колумбія",
+ "Cocos (Keeling) Islands": "Кокосові (Кілінг) Острови",
+ "Christmas Island": "Острів Різдва",
+ "China": "Китай",
+ "Chile": "Чилі",
+ "Chad": "Чад",
+ "Central African Republic": "Центральноафриканська Республіка",
+ "Cayman Islands": "Кайманові Острови",
+ "Caribbean Netherlands": "Бонайре, Сінт-Естатіус і Саба",
+ "Cape Verde": "Кабо-Верде",
+ "Canada": "Канада",
+ "Cameroon": "Камерун",
+ "Cambodia": "Камбоджа",
+ "Burundi": "Бурунді",
+ "Burkina Faso": "Буркіна-Фасо",
+ "Bulgaria": "Болгарія",
+ "Brunei": "Бруней",
+ "British Virgin Islands": "Британські Віргінські Острови",
+ "British Indian Ocean Territory": "Британська територія в Індійському океані",
+ "Brazil": "Бразилія",
+ "Bouvet Island": "Острів Буве",
+ "Botswana": "Ботсвана",
+ "Bosnia": "Боснія і Герцеговина",
+ "Bolivia": "Болівія",
+ "Bhutan": "Бутан",
+ "Bermuda": "Бермудські Острови",
+ "Benin": "Бенін",
+ "Belize": "Беліз",
+ "Belgium": "Бельгія",
+ "Belarus": "Білорусь",
+ "Barbados": "Барбадос",
+ "Bangladesh": "Бангладеш",
+ "Bahrain": "Бахрейн",
+ "Bahamas": "Багамські Острови",
+ "Azerbaijan": "Азербайджан",
+ "Austria": "Австрія",
+ "Australia": "Австралія",
+ "Aruba": "Аруба",
+ "Armenia": "Вірменія",
+ "Argentina": "Аргентина",
+ "Antigua & Barbuda": "Антигуа і Барбуда",
+ "Antarctica": "Антарктика",
+ "Anguilla": "Ангілья",
+ "Angola": "Ангола",
+ "Andorra": "Андорра",
+ "American Samoa": "Американське Самоа",
+ "Algeria": "Алжир",
+ "Albania": "Албанія",
+ "Åland Islands": "Аландські Острови",
+ "Afghanistan": "Афганістан",
+ "United States": "Сполучені Штати Америки",
+ "United Kingdom": "Велика Британія",
+ "The call was answered on another device.": "На дзвінок відповіли на іншому пристрої.",
+ "Answered Elsewhere": "Відповіли деінде",
+ "The call could not be established": "Не вдалося встановити зв'язок",
+ "The other party declined the call.": "Інша сторона відхилила дзвінок.",
+ "Call Declined": "Дзвінок відхилено",
+ "Falkland Islands": "Фолклендські (Мальвінські) Острови",
+ "Ethiopia": "Ефіопія",
+ "Estonia": "Естонія",
+ "Eritrea": "Еритрея",
+ "Equatorial Guinea": "Екваторіальна Гвінея",
+ "El Salvador": "Сальвадор",
+ "Egypt": "Єгипет",
+ "Ecuador": "Еквадор",
+ "Dominican Republic": "Домініканська Республіка",
+ "Dominica": "Домініка",
+ "Djibouti": "Джибуті",
+ "Denmark": "Данія",
+ "Côte d’Ivoire": "Кот-Д'Івуар",
+ "Czech Republic": "Чехія",
+ "Cyprus": "Кіпр",
+ "Curaçao": "Кюрасао",
+ "Cuba": "Куба",
+ "Croatia": "Хорватія",
+ "Costa Rica": "Коста-Рика",
+ "Cook Islands": "Острови Кука",
+ "Congo - Kinshasa": "Демократична Республіка Конго",
+ "Congo - Brazzaville": "Конго",
+ "%(senderDisplayName)s changed the server ACLs for this room.": "%(senderDisplayName)s змінює серверні права доступу для цієї кімнати.",
+ "%(senderDisplayName)s set the server ACLs for this room.": "%(senderDisplayName)s встановлює серверні права доступу для цієї кімнати.",
+ "Takes the call in the current room off hold": "Зніміть дзвінок у поточній кімнаті з утримання",
+ "Places the call in the current room on hold": "Переведіть дзвінок у поточній кімнаті на утримання",
+ "Zimbabwe": "Зімбабве",
+ "Zambia": "Замбія",
+ "Yemen": "Ємен",
+ "Western Sahara": "Західна Сахара",
+ "Wallis & Futuna": "Волліс і Футуна",
+ "Vietnam": "В'єтнам",
+ "Venezuela": "Венесуела",
+ "Vatican City": "Ватикан",
+ "Vanuatu": "Вануату",
+ "Uzbekistan": "Узбекистан",
+ "Uruguay": "Уругвай",
+ "United Arab Emirates": "Об'єднані Арабські Емірати",
+ "Ukraine": "Україна",
+ "Uganda": "Уганда",
+ "U.S. Virgin Islands": "Віргінські Острови (США)",
+ "Tuvalu": "Тувалу",
+ "Turks & Caicos Islands": "Острови Теркс і Кайкос",
+ "Turkmenistan": "Туркменистан",
+ "Turkey": "Туреччина",
+ "Tunisia": "Туніс",
+ "Trinidad & Tobago": "Тринідад і Тобаго",
+ "Tonga": "Тонга",
+ "Tokelau": "Токелау",
+ "Togo": "Того",
+ "Timor-Leste": "Тимор-Лешті",
+ "Thailand": "Таїланд",
+ "Tanzania": "Танзанія",
+ "Tajikistan": "Таджикистан",
+ "Taiwan": "Тайвань",
+ "São Tomé & Príncipe": "Сан-Томе і Принсіпі",
+ "Syria": "Сирія",
+ "Switzerland": "Швейцарія",
+ "Sweden": "Швеція",
+ "Swaziland": "Есватіні",
+ "Svalbard & Jan Mayen": "Свальбард і Ян-Маєн",
+ "Suriname": "Суринам",
+ "Sudan": "Судан",
+ "St. Vincent & Grenadines": "Сент-Вінсент і Гренадини",
+ "St. Pierre & Miquelon": "Сен-П'єр і Мікелон",
+ "St. Martin": "Сен-Мартен",
+ "St. Lucia": "Сент-Люсія",
+ "St. Kitts & Nevis": "Сент-Кітс і Невіс",
+ "St. Helena": "Острів Святої Єлени",
+ "St. Barthélemy": "Сен-Бартелемі",
+ "Sri Lanka": "Шрі-Ланка",
+ "Spain": "Іспанія",
+ "South Sudan": "Південний Судан",
+ "South Korea": "Південна Корея",
+ "South Georgia & South Sandwich Islands": "Південна Джорджія та Південні Сандвічеві Острови",
+ "South Africa": "Південна Африка",
+ "Somalia": "Сомалі",
+ "Solomon Islands": "Соломонові Острови",
+ "Slovenia": "Словенія",
+ "Slovakia": "Словаччина",
+ "Sint Maarten": "Сінт-Мартен",
+ "Singapore": "Сингапур",
+ "Sierra Leone": "Сьєрра-Леоне",
+ "Seychelles": "Сейшельські Острови",
+ "Serbia": "Сербія",
+ "Senegal": "Сенегал",
+ "Saudi Arabia": "Саудівська Аравія",
+ "San Marino": "Сан-Марино",
+ "Samoa": "Самоа",
+ "Réunion": "Реюньйон",
+ "Rwanda": "Руанда",
+ "Russia": "Російська Федерація",
+ "Romania": "Румунія",
+ "Qatar": "Катар",
+ "Puerto Rico": "Пуерто-Рико",
+ "Portugal": "Португалія",
+ "Poland": "Польща",
+ "Pitcairn Islands": "Піткерн",
+ "Philippines": "Філіппіни",
+ "Peru": "Перу",
+ "Paraguay": "Парагвай",
+ "Papua New Guinea": "Папуа-Нова Гвінея",
+ "Panama": "Панама",
+ "Palestine": "Палестина",
+ "Palau": "Палау",
+ "Pakistan": "Пакистан",
+ "Oman": "Оман",
+ "Norway": "Норвегія",
+ "Northern Mariana Islands": "Північні Маріанські Острови",
+ "North Korea": "Північна Корея",
+ "Norfolk Island": "Острів Норфолк",
+ "Niue": "Ніуе",
+ "Nigeria": "Нігерія",
+ "Niger": "Нігер",
+ "Nicaragua": "Нікарагуа",
+ "New Zealand": "Нова Зеландія",
+ "New Caledonia": "Нова Каледонія",
+ "Netherlands": "Нідерланди",
+ "Nepal": "Непал",
+ "Nauru": "Науру",
+ "Namibia": "Намібія",
+ "Myanmar": "М'янма",
+ "Mozambique": "Мозамбік",
+ "Morocco": "Марокко",
+ "Montserrat": "Монтсеррат",
+ "Montenegro": "Чорногорія",
+ "Mongolia": "Монголія",
+ "Monaco": "Монако",
+ "Moldova": "Молдова",
+ "Micronesia": "Мікронезія",
+ "Mexico": "Мексика",
+ "Mayotte": "Майотта",
+ "Mauritius": "Маврикій",
+ "Mauritania": "Мавританія",
+ "Martinique": "Мартиніка",
+ "Marshall Islands": "Маршаллові Острови",
+ "Malta": "Мальта",
+ "Mali": "Малі",
+ "Maldives": "Мальдіви",
+ "Malaysia": "Малайзія",
+ "Malawi": "Малаві",
+ "Madagascar": "Мадагаскар",
+ "Macedonia": "Північна Македонія",
+ "Macau": "Макао",
+ "Luxembourg": "Люксембург",
+ "Lithuania": "Литва",
+ "Liechtenstein": "Ліхтенштейн",
+ "Libya": "Лівія",
+ "Liberia": "Ліберія",
+ "Lesotho": "Лесото",
+ "Lebanon": "Ліван",
+ "Latvia": "Латвія",
+ "Laos": "Лаоська Народно-Демократична Республіка",
+ "Kyrgyzstan": "Киргизстан",
+ "Kuwait": "Кувейт",
+ "Kosovo": "Косово",
+ "Kiribati": "Кірибаті",
+ "Kenya": "Кенія",
+ "Kazakhstan": "Казахстан",
+ "Jordan": "Йорданія",
+ "Jersey": "Джерсі",
+ "Japan": "Японія",
+ "Jamaica": "Ямайка",
+ "Italy": "Італія",
+ "Israel": "Ізраїль",
+ "Isle of Man": "Острів Мен",
+ "Ireland": "Ірландія",
+ "Iraq": "Ірак",
+ "Iran": "Іран",
+ "Indonesia": "Індонезія",
+ "India": "Індія",
+ "Iceland": "Ісландія",
+ "Hungary": "Угорщина",
+ "Hong Kong": "Гонконг",
+ "Honduras": "Гондурас",
+ "Heard & McDonald Islands": "Острів Герд і Острови Макдоналд",
+ "Haiti": "Гаїті",
+ "Guyana": "Гаяна",
+ "Guinea-Bissau": "Гвінея-Бісау",
+ "Guinea": "Гвінея",
+ "Guernsey": "Гернсі",
+ "Guatemala": "Гватемала",
+ "Guam": "Гуам",
+ "Guadeloupe": "Гваделупа",
+ "Grenada": "Гренада",
+ "Greenland": "Гренландія",
+ "Greece": "Греція",
+ "Gibraltar": "Гібралтар",
+ "Ghana": "Гана",
+ "Germany": "Німеччина",
+ "Georgia": "Грузія",
+ "Gambia": "Гамбія",
+ "Gabon": "Габон",
+ "French Southern Territories": "Французькі Південні Території",
+ "French Polynesia": "Французька Полінезія",
+ "French Guiana": "Французька Гвіана",
+ "France": "Франція",
+ "Finland": "Фінляндія",
+ "Fiji": "Фіджі",
+ "Faroe Islands": "Фарерські Острови",
+ "Unable to access microphone": "Неможливо доступитись до мікрофона",
+ "Filter rooms and people": "Відфільтрувати кімнати та людей",
+ "Find a room… (e.g. %(exampleRoom)s)": "Знайти кімнату… (напр. %(exampleRoom)s)",
+ "Find a room…": "Знайти кімнату…",
+ "Can't find this server or its room list": "Не вдалось знайти цей сервер у переліку кімнат",
+ "If you can't find the room you're looking for, ask for an invite or Create a new room.": "Якщо ви не можете знайти потрібну кімнату, попросіть запрошення або Створіть нову кімнату власноруч.",
+ "Cannot reach homeserver": "Не вдається зв'язатися з домашнім сервером",
+ "Ensure you have a stable internet connection, or get in touch with the server admin": "Переконайтеся, що у ваше з'єднання з Інтернетом стабільне або зв’яжіться з системним адміністратором",
+ "User %(user_id)s may or may not exist": "Користувач %(user_id)s можливо існує, а можливо й ні",
+ "No need for symbols, digits, or uppercase letters": "Цифри або великі букви не вимагаються",
+ "Capitalization doesn't help very much": "Великі букви не дуже допомагають",
+ "You're all caught up.": "Все готово.",
+ "Hey you. You're the best!": "Гей, ти, так, ти. Ти найкращий!",
+ "You’re all caught up": "Все готово"
}
diff --git a/src/i18n/strings/zh_Hans.json b/src/i18n/strings/zh_Hans.json
index 809be89383..672b1befd1 100644
--- a/src/i18n/strings/zh_Hans.json
+++ b/src/i18n/strings/zh_Hans.json
@@ -2398,5 +2398,16 @@
"Privacy": "隐私",
"Explore community rooms": "探索社区聊天室",
"%(count)s results|one": "%(count)s 个结果",
- "Room Info": "聊天室信息"
+ "Room Info": "聊天室信息",
+ "No other application is using the webcam": "没有其他应用程序正在使用摄像头",
+ "Permission is granted to use the webcam": "授予使用网络摄像头的权限",
+ "A microphone and webcam are plugged in and set up correctly": "麦克风和摄像头已插入并正确设置",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "通话失败,因为无法访问摄像头或麦克风。 检查:",
+ "Unable to access webcam / microphone": "无法访问摄像头/麦克风",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "呼叫失败,因为无法访问任何麦克风。 检查是否已插入麦克风并正确设置。",
+ "Unable to access microphone": "无法使用麦克风",
+ "The call was answered on another device.": "在另一台设备上应答了该通话。",
+ "The call could not be established": "无法建立通话",
+ "The other party declined the call.": "对方拒绝了通话。",
+ "Call Declined": "通话被拒绝"
}
diff --git a/src/i18n/strings/zh_Hant.json b/src/i18n/strings/zh_Hant.json
index d02ac268bd..c820d9b1cf 100644
--- a/src/i18n/strings/zh_Hant.json
+++ b/src/i18n/strings/zh_Hant.json
@@ -2844,5 +2844,135 @@
"Places the call in the current room on hold": "在目前的聊天室撥打通話並等候接聽",
"Role": "角色",
"Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(count)s rooms.|one": "使用 %(size)s 儲存來自 %(count)s 個聊天室的訊息,在本機安全地快取已加密的訊息以讓它們可以在搜尋結果中出現。",
- "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(count)s rooms.|other": "使用 %(size)s 儲存來自 %(count)s 個聊天室的訊息,在本機安全地快取已加密的訊息以讓它們可以在搜尋結果中出現。"
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(count)s rooms.|other": "使用 %(size)s 儲存來自 %(count)s 個聊天室的訊息,在本機安全地快取已加密的訊息以讓它們可以在搜尋結果中出現。",
+ "Go to Home View": "轉到主視窗",
+ "Filter rooms and people": "過濾聊天室與人們",
+ "Open the link in the email to continue registration.": "開啟電子郵件中的連結以繼續註冊。",
+ "A confirmation email has been sent to %(emailAddress)s": "確認電子郵件已寄送至 %(emailAddress)s",
+ "Start a new chat": "開始新聊天",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|one": "使用 %(size)s 來儲存來自 %(rooms)s 個聊天室的訊息,在本機安全地快取已加密的訊息以使其出現在搜尋結果中。",
+ "Securely cache encrypted messages locally for them to appear in search results, using %(size)s to store messages from %(rooms)s rooms.|other": "使用 %(size)s 來儲存來自 %(rooms)s 個聊天室的訊息,在本機安全地快取已加密的訊息以使其出現在搜尋結果中。",
+ "
HTML for your community's page
\n
\n Use the long description to introduce new members to the community, or distribute\n some important links\n
\n",
+ "Decline All": "全部拒絕",
+ "Approve": "批准",
+ "This widget would like to:": "這個小工具想要:",
+ "Approve widget permissions": "批准小工具權限",
+ "Use Ctrl + Enter to send a message": "使用 Ctrl + Enter 來傳送訊息",
+ "Use Command + Enter to send a message": "使用 Command + Enter 來傳送訊息",
+ "See %(msgtype)s messages posted to your active room": "檢視發佈到您的活躍聊天室的 %(msgtype)s 訊息",
+ "See %(msgtype)s messages posted to this room": "檢視發佈到此聊天室的 %(msgtype)s 訊息",
+ "Send %(msgtype)s messages as you in your active room": "在您的活躍聊天室中以您的身份傳送 %(msgtype)s 訊息",
+ "Send %(msgtype)s messages as you in this room": "在此聊天室中以您的身份傳送 %(msgtype)s 訊息",
+ "See general files posted to your active room": "檢視在您的活躍聊天室中發佈的一般檔案",
+ "See general files posted to this room": "檢視在此聊天室中發佈的一般檔案",
+ "Send general files as you in your active room": "在您的活躍聊天室中以您的身份傳送一般檔案",
+ "Send general files as you in this room": "在此聊天室中以您的身份傳送一般檔案",
+ "See videos posted to your active room": "檢視發佈到您的活躍聊天室的影片",
+ "See videos posted to this room": "檢視發佈到此聊天室的影片",
+ "Send videos as you in your active room": "在您的活躍聊天室中以您的身份傳送影片",
+ "Send videos as you in this room": "在此聊天室中以您的身份傳送影片",
+ "See images posted to your active room": "檢視發佈到您的活躍聊天室的圖片",
+ "See images posted to this room": "檢視發佈到此聊天室的圖片",
+ "Send images as you in your active room": "在您活躍的聊天室以您的身份傳送圖片",
+ "Send images as you in this room": "在此聊天室以您的身份傳送圖片",
+ "See emotes posted to your active room": "檢視發佈到您的活躍聊天室的表情符號",
+ "See emotes posted to this room": "檢視發佈到此聊天室的表情符號",
+ "Send emotes as you in your active room": "在您的活躍聊天室中以您的身份傳送表情符號",
+ "Send emotes as you in this room": "在此聊天室中以您的身份傳送表情符號",
+ "See text messages posted to your active room": "檢視發佈到您的活躍聊天室的文字訊息",
+ "See text messages posted to this room": "檢視發佈到此聊天室的文字訊息",
+ "Send text messages as you in your active room": "在您的活躍聊天室以您的身份傳送文字訊息",
+ "Send text messages as you in this room": "在此聊天室以您的身份傳送文字訊息",
+ "See messages posted to your active room": "檢視發佈到您的活躍聊天室的訊息",
+ "See messages posted to this room": "檢視發佈到此聊天室的訊息",
+ "Send messages as you in your active room": "在您的活躍聊天室以您的身份傳送訊息",
+ "Send messages as you in this room": "在此聊天室以您的身份傳送訊息",
+ "The %(capability)s capability": "%(capability)s 能力",
+ "See %(eventType)s events posted to your active room": "檢視發佈到您的活躍聊天室的 %(eventType)s 活動",
+ "Send %(eventType)s events as you in your active room": "以您的身份在您的活躍聊天是傳送 %(eventType)s 活動",
+ "See %(eventType)s events posted to this room": "檢視發佈到此聊天室的 %(eventType)s 活動",
+ "Send %(eventType)s events as you in this room": "以您的身份在此聊天室傳送 %(eventType)s 活動",
+ "with state key %(stateKey)s": "帶有狀態金鑰 %(stateKey)s",
+ "with an empty state key": "帶有空的狀態金鑰",
+ "See when anyone posts a sticker to your active room": "檢視何時有人將貼圖貼到您活躍的聊天室",
+ "Send stickers to your active room as you": "以您的身份傳送貼圖到您活躍的聊天室",
+ "See when a sticker is posted in this room": "檢視貼圖在此聊天室中何時貼出",
+ "Send stickers to this room as you": "以您的身份傳送貼圖到此聊天室",
+ "See when the avatar changes in your active room": "檢視您活躍聊天是的大頭照何時變更",
+ "Change the avatar of your active room": "變更您活躍聊天是的大頭照",
+ "See when the avatar changes in this room": "檢視此聊天是的大頭照何時變更",
+ "Change the avatar of this room": "變更此聊天室的大頭照",
+ "See when the name changes in your active room": "檢視您活躍聊天室的名稱何時變更",
+ "Change the name of your active room": "變更您活躍聊天室的名稱",
+ "See when the name changes in this room": "檢視此聊天是的名稱何時變更",
+ "Change the name of this room": "變更此聊天室的名稱",
+ "See when the topic changes in your active room": "檢視您活躍的聊天是的主題何時變更",
+ "Change the topic of your active room": "變更您活躍聊天是的主題",
+ "See when the topic changes in this room": "檢視此聊天是的主題何時變更",
+ "Change the topic of this room": "變更此聊天室的主題",
+ "Change which room you're viewing": "變更您正在檢視的聊天室",
+ "Send stickers into your active room": "傳送貼圖到您活躍的聊天室",
+ "Send stickers into this room": "傳送貼圖到此聊天室",
+ "Remain on your screen while running": "在執行時保留在您的畫面上",
+ "Remain on your screen when viewing another room, when running": "在執行與檢視其他聊天室時仍保留在您的畫面上",
+ "Enter phone number": "輸入電話號碼",
+ "Enter email address": "輸入電子郵件地址",
+ "Return to call": "回到通話",
+ "Fill Screen": "全螢幕",
+ "Voice Call": "音訊通話",
+ "Video Call": "視訊通話",
+ "New here? Create an account": "新手?建立帳號",
+ "Got an account? Sign in": "有帳號了嗎?登入",
+ "Render LaTeX maths in messages": "在訊息中彩現 LaTeX 數學",
+ "No other application is using the webcam": "無其他應用程式正在使用網路攝影機",
+ "Permission is granted to use the webcam": "授予使用網路攝影機的權限",
+ "A microphone and webcam are plugged in and set up correctly": "麥克風與網路攝影機已插入並正確設定",
+ "Call failed because no webcam or microphone could not be accessed. Check that:": "因為無法存取網路攝影機或麥克風,所以通話失敗。請檢查:",
+ "Unable to access webcam / microphone": "無法存取網路攝影機/麥克風",
+ "Call failed because no microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "因為無法存取麥克風,所以通話失敗。請檢查是否已插入麥克風並正確設定。",
+ "Unable to access microphone": "無法存取麥克風",
+ "Decide where your account is hosted": "決定託管帳號的位置",
+ "Host account on": "帳號託管於",
+ "Already have an account? Sign in here": "已有帳號?在此登入",
+ "%(ssoButtons)s Or %(usernamePassword)s": "%(ssoButtons)s 或 %(usernamePassword)s",
+ "Continue with %(ssoButtons)s": "使用 %(ssoButtons)s 繼續",
+ "That username already exists, please try another.": "使用者名稱已存在,請試試其他的。",
+ "New? Create account": "新人?建立帳號",
+ "There was a problem communicating with the homeserver, please try again later.": "與家伺服器通訊時出現問題,請再試一次。",
+ "Use email to optionally be discoverable by existing contacts.": "使用電子郵件以選擇性地被既有的聯絡人探索。",
+ "Use email or phone to optionally be discoverable by existing contacts.": "使用電子郵件或電話以選擇性地被既有的聯絡人探索。",
+ "Add an email to be able to reset your password.": "新增電子郵件以重設您的密碼。",
+ "Forgot password?": "忘記密碼?",
+ "That phone number doesn't look quite right, please check and try again": "電話號碼看起來不太對,請檢查並再試一次",
+ "About homeservers": "關於家伺服器",
+ "Learn more": "取得更多資訊",
+ "Use your preferred Matrix homeserver if you have one, or host your own.": "如果您有的話,可以使用您偏好的 Matrix 家伺服器,或是自己架一個。",
+ "Other homeserver": "其他家伺服器",
+ "We call the places you where you can host your account ‘homeservers’.": "我們將您可以託管您的帳號的地方稱為「家伺服器」。",
+ "Sign into your homeserver": "登入您的家伺服器",
+ "Matrix.org is the biggest public homeserver in the world, so it’s a good place for many.": "Matrix.org 是世界上最大的公開伺服器,因此對許多人來說是個好地方。",
+ "Specify a homeserver": "指定家伺服器",
+ "Just a heads up, if you don't add an email and forget your password, you could permanently lose access to your account.": "請注意,如果您不新增電子郵件且忘記密碼,您將永遠失去對您帳號的存取權。",
+ "Continuing without email": "不用電子郵件繼續",
+ "Continue with %(provider)s": "使用 %(provider)s 繼續",
+ "Homeserver": "家伺服器",
+ "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use Element with an existing Matrix account on a different homeserver.": "您可以透過指定不同的家伺服器 URL 使用自訂伺服器選項來登入其他 Matrix 伺服器。這讓您可以使用在不同家伺服器上的既有 Matrix 帳號。",
+ "Server Options": "伺服器選項",
+ "Reason (optional)": "理由(選擇性)",
+ "We call the places where you can host your account ‘homeservers’.": "我們將可以託管您的帳號的地方稱為「家伺服器」。",
+ "Invalid URL": "無效的 URL",
+ "Unable to validate homeserver": "無法驗證家伺服器",
+ "sends confetti": "傳送五彩碎紙",
+ "Sends the given message with confetti": "使用五彩碎紙傳送訊息",
+ "Show chat effects": "顯示聊天特效",
+ "Effects": "影響",
+ "Call failed because webcam or microphone could not be accessed. Check that:": "通話失敗,因為無法存取網路攝影機或麥克風。請檢查:",
+ "Call failed because microphone could not be accessed. Check that a microphone is plugged in and set up correctly.": "通話失敗,因為無法存取麥克風。請檢查是否已插入麥克風並正確設定。",
+ "Hold": "保留",
+ "Resume": "繼續",
+ "%(peerName)s held the call": "%(peerName)s 保留通話",
+ "You held the call Resume": "您已保留通話 繼續",
+ "%(name)s paused": "%(name)s 已暫停",
+ "You've reached the maximum number of simultaneous calls.": "您已達到同時通話的最大數量。",
+ "Too Many Calls": "太多通話"
}
diff --git a/src/settings/Settings.ts b/src/settings/Settings.ts
index 31e133be72..b239b809fe 100644
--- a/src/settings/Settings.ts
+++ b/src/settings/Settings.ts
@@ -564,13 +564,6 @@ export const SETTINGS: {[setting: string]: ISetting} = {
// This is a tri-state value, where `null` means "prompt the user".
default: null,
},
- "sendReadReceipts": {
- supportedLevels: LEVELS_ROOM_SETTINGS,
- displayName: _td(
- "Send read receipts for messages (requires compatible homeserver to disable)",
- ),
- default: true,
- },
"showImages": {
supportedLevels: LEVELS_ACCOUNT_SETTINGS,
displayName: _td("Show previews/thumbnails for images"),
@@ -634,6 +627,11 @@ export const SETTINGS: {[setting: string]: ISetting} = {
displayName: _td("Enable experimental, compact IRC style layout"),
default: false,
},
+ "showChatEffects": {
+ supportedLevels: LEVELS_ACCOUNT_SETTINGS,
+ displayName: _td("Show chat effects"),
+ default: true,
+ },
"Widgets.pinned": {
supportedLevels: LEVELS_ROOM_OR_ACCOUNT,
default: {},
diff --git a/src/settings/SettingsStore.ts b/src/settings/SettingsStore.ts
index 7c05e4b500..82f169f498 100644
--- a/src/settings/SettingsStore.ts
+++ b/src/settings/SettingsStore.ts
@@ -42,7 +42,7 @@ for (const key of Object.keys(SETTINGS)) {
if (SETTINGS[key].invertedSettingName) {
// Invert now so that the rest of the system will invert it back
// to what was intended.
- invertedDefaultSettings[key] = !SETTINGS[key].default;
+ invertedDefaultSettings[SETTINGS[key].invertedSettingName] = !SETTINGS[key].default;
}
}
diff --git a/src/stores/WidgetStore.ts b/src/stores/WidgetStore.ts
index a8040f57de..c9cf0a1c70 100644
--- a/src/stores/WidgetStore.ts
+++ b/src/stores/WidgetStore.ts
@@ -29,6 +29,8 @@ import WidgetUtils from "../utils/WidgetUtils";
import {SettingLevel} from "../settings/SettingLevel";
import {WidgetType} from "../widgets/WidgetType";
import {UPDATE_EVENT} from "./AsyncStore";
+import { MatrixClientPeg } from "../MatrixClientPeg";
+import { arrayDiff, arrayHasDiff, arrayUnion } from "../utils/arrays";
interface IState {}
@@ -39,20 +41,26 @@ export interface IApp extends IWidget {
avatar_url: string; // MSC2765 https://github.com/matrix-org/matrix-doc/pull/2765
}
+type PinnedWidgets = Record;
+
interface IRoomWidgets {
widgets: IApp[];
- pinned: Record;
+ pinned: PinnedWidgets;
}
export const MAX_PINNED = 3;
+function widgetUid(app: IApp): string {
+ return `${app.roomId ?? MatrixClientPeg.get().getUserId()}::${app.id}`;
+}
+
// TODO consolidate WidgetEchoStore into this
// TODO consolidate ActiveWidgetStore into this
export default class WidgetStore extends AsyncStoreWithClient {
private static internalInstance = new WidgetStore();
- private widgetMap = new Map();
- private roomMap = new Map();
+ private widgetMap = new Map(); // Key is widget Unique ID (UID)
+ private roomMap = new Map(); // Key is room ID
private constructor() {
super(defaultDispatcher, {});
@@ -126,13 +134,21 @@ export default class WidgetStore extends AsyncStoreWithClient {
// first clean out old widgets from the map which originate from this room
// otherwise we are out of sync with the rest of the app with stale widget events during removal
Array.from(this.widgetMap.values()).forEach(app => {
- if (app.roomId === room.roomId) {
- this.widgetMap.delete(app.id);
- }
+ if (app.roomId !== room.roomId) return; // skip - wrong room
+ this.widgetMap.delete(widgetUid(app));
});
this.generateApps(room).forEach(app => {
- this.widgetMap.set(app.id, app);
+ // Sanity check for https://github.com/vector-im/element-web/issues/15705
+ const existingApp = this.widgetMap.get(widgetUid(app));
+ if (existingApp) {
+ console.warn(
+ `Possible widget ID conflict for ${app.id} - wants to store in room ${app.roomId} ` +
+ `but is currently stored as ${existingApp.roomId} - letting the want win`,
+ );
+ }
+
+ this.widgetMap.set(widgetUid(app), app);
roomInfo.widgets.push(app);
});
this.emit(room.roomId);
@@ -146,61 +162,68 @@ export default class WidgetStore extends AsyncStoreWithClient {
this.emit(UPDATE_EVENT);
};
- public getRoomId = (widgetId: string) => {
- const app = this.widgetMap.get(widgetId);
- if (!app) return null;
- return app.roomId;
- }
-
public getRoom = (roomId: string) => {
return this.roomMap.get(roomId);
};
private onPinnedWidgetsChange = (settingName: string, roomId: string) => {
this.initRoom(roomId);
- this.getRoom(roomId).pinned = SettingsStore.getValue(settingName, roomId);
+
+ const pinned: PinnedWidgets = SettingsStore.getValue(settingName, roomId);
+
+ // Sanity check for https://github.com/vector-im/element-web/issues/15705
+ const roomInfo = this.getRoom(roomId);
+ const remappedPinned: PinnedWidgets = {};
+ for (const widgetId of Object.keys(pinned)) {
+ const isPinned = pinned[widgetId];
+ if (!roomInfo.widgets?.some(w => w.id === widgetId)) {
+ console.warn(`Skipping pinned widget update for ${widgetId} in ${roomId} -- wrong room`);
+ } else {
+ remappedPinned[widgetId] = isPinned;
+ }
+ }
+ roomInfo.pinned = remappedPinned;
+
this.emit(roomId);
this.emit(UPDATE_EVENT);
};
- public isPinned(widgetId: string) {
- const roomId = this.getRoomId(widgetId);
+ public isPinned(roomId: string, widgetId: string) {
return !!this.getPinnedApps(roomId).find(w => w.id === widgetId);
}
- public canPin(widgetId: string) {
- const roomId = this.getRoomId(widgetId);
+ // dev note: we don't need the widgetId on this function, but the contract makes more sense
+ // when we require it.
+ public canPin(roomId: string, widgetId: string) {
return this.getPinnedApps(roomId).length < MAX_PINNED;
}
- public pinWidget(widgetId: string) {
- const roomId = this.getRoomId(widgetId);
+ public pinWidget(roomId: string, widgetId: string) {
const roomInfo = this.getRoom(roomId);
if (!roomInfo) return;
// When pinning, first confirm all the widgets (Jitsi) which were autopinned so that the order is correct
const autoPinned = this.getPinnedApps(roomId).filter(app => !roomInfo.pinned[app.id]);
autoPinned.forEach(app => {
- this.setPinned(app.id, true);
+ this.setPinned(roomId, app.id, true);
});
- this.setPinned(widgetId, true);
+ this.setPinned(roomId, widgetId, true);
// Show the apps drawer upon the user pinning a widget
- if (RoomViewStore.getRoomId() === this.getRoomId(widgetId)) {
+ if (RoomViewStore.getRoomId() === roomId) {
defaultDispatcher.dispatch({
action: "appsDrawer",
show: true,
- })
+ });
}
}
- public unpinWidget(widgetId: string) {
- this.setPinned(widgetId, false);
+ public unpinWidget(roomId: string, widgetId: string) {
+ this.setPinned(roomId, widgetId, false);
}
- private setPinned(widgetId: string, value: boolean) {
- const roomId = this.getRoomId(widgetId);
+ private setPinned(roomId: string, widgetId: string, value: boolean) {
const roomInfo = this.getRoom(roomId);
if (!roomInfo) return;
if (roomInfo.pinned[widgetId] === false && value) {
@@ -211,7 +234,7 @@ export default class WidgetStore extends AsyncStoreWithClient {
// Clean up the pinned record
Object.keys(roomInfo).forEach(wId => {
- if (!roomInfo.widgets.some(w => w.id === wId)) {
+ if (!roomInfo.widgets.some(w => w.id === wId) || !roomInfo.pinned[wId]) {
delete roomInfo.pinned[wId];
}
});
@@ -221,9 +244,8 @@ export default class WidgetStore extends AsyncStoreWithClient {
this.emit(UPDATE_EVENT);
}
- public movePinnedWidget(widgetId: string, delta: 1 | -1) {
+ public movePinnedWidget(roomId: string, widgetId: string, delta: 1 | -1) {
// TODO simplify this by changing the storage medium of pinned to an array once the Jitsi default-on goes away
- const roomId = this.getRoomId(widgetId);
const roomInfo = this.getRoom(roomId);
if (!roomInfo || roomInfo.pinned[widgetId] === false) return;
@@ -264,12 +286,33 @@ export default class WidgetStore extends AsyncStoreWithClient {
});
const order = Object.keys(roomInfo.pinned).filter(k => roomInfo.pinned[k]);
- let apps = order.map(wId => this.widgetMap.get(wId)).filter(Boolean);
- apps = apps.slice(0, priorityWidget ? MAX_PINNED - 1 : MAX_PINNED);
+ const apps = order
+ .map(wId => Array.from(this.widgetMap.values())
+ .find(w2 => w2.roomId === roomId && w2.id === wId))
+ .filter(Boolean)
+ .slice(0, priorityWidget ? MAX_PINNED - 1 : MAX_PINNED);
if (priorityWidget) {
apps.push(priorityWidget);
}
+ // Sanity check for https://github.com/vector-im/element-web/issues/15705
+ // We union the app IDs the above generated with the roomInfo's known widgets to
+ // get a list of IDs which both exist. We then diff that against the generated app
+ // IDs above to ensure that all of the app IDs are captured by the union with the
+ // room - if we grabbed a widget that wasn't part of the roomInfo's list, it wouldn't
+ // be in the union and thus result in a diff.
+ const appIds = apps.map(a => widgetUid(a));
+ const roomAppIds = roomInfo.widgets.map(a => widgetUid(a));
+ const roomAppIdsUnion = arrayUnion(appIds, roomAppIds);
+ const missingSomeApps = arrayHasDiff(roomAppIdsUnion, appIds);
+ if (missingSomeApps) {
+ const diff = arrayDiff(roomAppIdsUnion, appIds);
+ console.warn(
+ `${roomId} appears to have a conflict for which widgets belong to it. ` +
+ `Widget UIDs are: `, [...diff.added, ...diff.removed],
+ );
+ }
+
return apps;
}
diff --git a/src/stores/room-list/RoomListStore.ts b/src/stores/room-list/RoomListStore.ts
index 0f3138fe9e..b2fe630760 100644
--- a/src/stores/room-list/RoomListStore.ts
+++ b/src/stores/room-list/RoomListStore.ts
@@ -34,6 +34,7 @@ import { MarkedExecution } from "../../utils/MarkedExecution";
import { AsyncStoreWithClient } from "../AsyncStoreWithClient";
import { NameFilterCondition } from "./filters/NameFilterCondition";
import { RoomNotificationStateStore } from "../notifications/RoomNotificationStateStore";
+import { VisibilityProvider } from "./filters/VisibilityProvider";
interface IState {
tagsEnabled?: boolean;
@@ -401,6 +402,10 @@ export class RoomListStoreClass extends AsyncStoreWithClient {
}
private async handleRoomUpdate(room: Room, cause: RoomUpdateCause): Promise {
+ if (!VisibilityProvider.instance.isRoomVisible(room)) {
+ return; // don't do anything on rooms that aren't visible
+ }
+
const shouldUpdate = await this.algorithm.handleRoomUpdate(room, cause);
if (shouldUpdate) {
if (SettingsStore.getValue("advancedRoomListLogging")) {
@@ -544,7 +549,8 @@ export class RoomListStoreClass extends AsyncStoreWithClient {
public async regenerateAllLists({trigger = true}) {
console.warn("Regenerating all room lists");
- const rooms = this.matrixClient.getVisibleRooms();
+ const rooms = this.matrixClient.getVisibleRooms()
+ .filter(r => VisibilityProvider.instance.isRoomVisible(r));
const customTags = new Set();
if (this.state.tagsEnabled) {
for (const room of rooms) {
diff --git a/src/stores/room-list/algorithms/Algorithm.ts b/src/stores/room-list/algorithms/Algorithm.ts
index 439141edb4..25059aabe7 100644
--- a/src/stores/room-list/algorithms/Algorithm.ts
+++ b/src/stores/room-list/algorithms/Algorithm.ts
@@ -34,6 +34,7 @@ import { EffectiveMembership, getEffectiveMembership, splitRoomsByMembership } f
import { OrderingAlgorithm } from "./list-ordering/OrderingAlgorithm";
import { getListAlgorithmInstance } from "./list-ordering";
import SettingsStore from "../../../settings/SettingsStore";
+import { VisibilityProvider } from "../filters/VisibilityProvider";
/**
* Fired when the Algorithm has determined a list has been updated.
@@ -188,6 +189,10 @@ export class Algorithm extends EventEmitter {
// Note throughout: We need async so we can wait for handleRoomUpdate() to do its thing,
// otherwise we risk duplicating rooms.
+ if (val && !VisibilityProvider.instance.isRoomVisible(val)) {
+ val = null; // the room isn't visible - lie to the rest of this function
+ }
+
// Set the last sticky room to indicate that we're in a change. The code throughout the
// class can safely handle a null room, so this should be safe to do as a backup.
this._lastStickyRoom = this._stickyRoom || {};
diff --git a/src/stores/room-list/filters/VisibilityProvider.ts b/src/stores/room-list/filters/VisibilityProvider.ts
new file mode 100644
index 0000000000..553dd33ce0
--- /dev/null
+++ b/src/stores/room-list/filters/VisibilityProvider.ts
@@ -0,0 +1,54 @@
+/*
+ * 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 {Room} from "matrix-js-sdk/src/models/room";
+import { RoomListCustomisations } from "../../../customisations/RoomList";
+
+export class VisibilityProvider {
+ private static internalInstance: VisibilityProvider;
+
+ private constructor() {
+ }
+
+ public static get instance(): VisibilityProvider {
+ if (!VisibilityProvider.internalInstance) {
+ VisibilityProvider.internalInstance = new VisibilityProvider();
+ }
+ return VisibilityProvider.internalInstance;
+ }
+
+ public isRoomVisible(room: Room): boolean {
+ /* eslint-disable prefer-const */
+ let isVisible = true; // Returned at the end of this function
+ let forced = false; // When true, this function won't bother calling the customisation points
+ /* eslint-enable prefer-const */
+
+ // ------
+ // TODO: The `if` statements to control visibility of custom room types
+ // would go here. The remainder of this function assumes that the statements
+ // will be here.
+ //
+ // When removing this comment block, please remove the lint disable lines in the area.
+ // ------
+
+ const isVisibleFn = RoomListCustomisations.isRoomVisible;
+ if (!forced && isVisibleFn) {
+ isVisible = isVisibleFn(room);
+ }
+
+ return isVisible;
+ }
+}
diff --git a/src/stores/widgets/StopGapWidgetDriver.ts b/src/stores/widgets/StopGapWidgetDriver.ts
index 60988040d3..2d2d1fcbdb 100644
--- a/src/stores/widgets/StopGapWidgetDriver.ts
+++ b/src/stores/widgets/StopGapWidgetDriver.ts
@@ -40,6 +40,9 @@ import { WidgetPermissionCustomisations } from "../../customisations/WidgetPermi
import { OIDCState, WidgetPermissionStore } from "./WidgetPermissionStore";
import { WidgetType } from "../../widgets/WidgetType";
import { EventType } from "matrix-js-sdk/src/@types/event";
+import { CHAT_EFFECTS } from "../../effects";
+import { containsEmoji } from "../../effects/utils";
+import dis from "../../dispatcher/dispatcher";
// TODO: Purge this from the universe
@@ -123,6 +126,14 @@ export class StopGapWidgetDriver extends WidgetDriver {
} else {
// message event
r = await client.sendEvent(roomId, eventType, content);
+
+ if (eventType === EventType.RoomMessage) {
+ CHAT_EFFECTS.forEach((effect) => {
+ if (containsEmoji(content, effect.emojis)) {
+ dis.dispatch({action: `effects.${effect.command}`});
+ }
+ });
+ }
}
return {roomId, eventId: r.event_id};
diff --git a/src/utils/AutoDiscoveryUtils.js b/src/utils/AutoDiscoveryUtils.js
index 94aa0c3544..18b6451d3e 100644
--- a/src/utils/AutoDiscoveryUtils.js
+++ b/src/utils/AutoDiscoveryUtils.js
@@ -34,6 +34,8 @@ export class ValidatedServerConfig {
isUrl: string;
isDefault: boolean;
+ // when the server config is based on static URLs the hsName is not resolvable and things may wish to use hsUrl
+ isNameResolvable: boolean;
warning: string;
}
@@ -161,7 +163,7 @@ export default class AutoDiscoveryUtils {
const url = new URL(homeserverUrl);
const serverName = url.hostname;
- return AutoDiscoveryUtils.buildValidatedConfigFromDiscovery(serverName, result, syntaxOnly);
+ return AutoDiscoveryUtils.buildValidatedConfigFromDiscovery(serverName, result, syntaxOnly, true);
}
/**
@@ -179,12 +181,12 @@ export default class AutoDiscoveryUtils {
* input.
* @param {string} serverName The domain name the AutoDiscovery result is for.
* @param {*} discoveryResult The AutoDiscovery result.
- * @param {boolean} syntaxOnly If true, errors relating to liveliness of the servers will
- * not be raised.
+ * @param {boolean} syntaxOnly If true, errors relating to liveliness of the servers will not be raised.
+ * @param {boolean} isSynthetic If true, then the discoveryResult was synthesised locally.
* @returns {Promise} Resolves to the validated configuration.
*/
static buildValidatedConfigFromDiscovery(
- serverName: string, discoveryResult, syntaxOnly=false): ValidatedServerConfig {
+ serverName: string, discoveryResult, syntaxOnly=false, isSynthetic=false): ValidatedServerConfig {
if (!discoveryResult || !discoveryResult["m.homeserver"]) {
// This shouldn't happen without major misconfiguration, so we'll log a bit of information
// in the log so we can find this bit of codee but otherwise tell teh user "it broke".
@@ -252,6 +254,7 @@ export default class AutoDiscoveryUtils {
isUrl: preferredIdentityUrl,
isDefault: false,
warning: hsResult.error,
+ isNameResolvable: !isSynthetic,
});
}
}
diff --git a/src/utils/StorageManager.js b/src/utils/StorageManager.js
index e29b6d9b0e..c90281bacf 100644
--- a/src/utils/StorageManager.js
+++ b/src/utils/StorageManager.js
@@ -190,3 +190,79 @@ export function trackStores(client) {
export function setCryptoInitialised(cryptoInited) {
localStorage.setItem("mx_crypto_initialised", cryptoInited);
}
+
+/* Simple wrapper functions around IndexedDB.
+ */
+
+let idb = null;
+
+async function idbInit(): Promise {
+ if (!indexedDB) {
+ throw new Error("IndexedDB not available");
+ }
+ idb = await new Promise((resolve, reject) => {
+ const request = indexedDB.open("matrix-react-sdk", 1);
+ request.onerror = reject;
+ request.onsuccess = (event) => { resolve(request.result); };
+ request.onupgradeneeded = (event) => {
+ const db = request.result;
+ db.createObjectStore("pickleKey");
+ db.createObjectStore("account");
+ };
+ });
+}
+
+export async function idbLoad(
+ table: string,
+ key: string | string[],
+): Promise {
+ if (!idb) {
+ await idbInit();
+ }
+ return new Promise((resolve, reject) => {
+ const txn = idb.transaction([table], "readonly");
+ txn.onerror = reject;
+
+ const objectStore = txn.objectStore(table);
+ const request = objectStore.get(key);
+ request.onerror = reject;
+ request.onsuccess = (event) => { resolve(request.result); };
+ });
+}
+
+export async function idbSave(
+ table: string,
+ key: string | string[],
+ data: any,
+): Promise {
+ if (!idb) {
+ await idbInit();
+ }
+ return new Promise((resolve, reject) => {
+ const txn = idb.transaction([table], "readwrite");
+ txn.onerror = reject;
+
+ const objectStore = txn.objectStore(table);
+ const request = objectStore.put(data, key);
+ request.onerror = reject;
+ request.onsuccess = (event) => { resolve(); };
+ });
+}
+
+export async function idbDelete(
+ table: string,
+ key: string | string[],
+): Promise {
+ if (!idb) {
+ await idbInit();
+ }
+ return new Promise((resolve, reject) => {
+ const txn = idb.transaction([table], "readwrite");
+ txn.onerror = reject;
+
+ const objectStore = txn.objectStore(table);
+ const request = objectStore.delete(key);
+ request.onerror = reject;
+ request.onsuccess = (event) => { resolve(); };
+ });
+}
diff --git a/src/utils/membership.ts b/src/utils/membership.ts
index 696bd57880..80f04dfe76 100644
--- a/src/utils/membership.ts
+++ b/src/utils/membership.ts
@@ -140,6 +140,6 @@ export async function leaveRoomBehaviour(roomId: string) {
}
if (RoomViewStore.getRoomId() === roomId) {
- dis.dispatch({action: 'view_next_room'});
+ dis.dispatch({action: 'view_home_page'});
}
}
diff --git a/src/utils/permalinks/Permalinks.js b/src/utils/permalinks/Permalinks.js
index 0f54bcce05..39c5776852 100644
--- a/src/utils/permalinks/Permalinks.js
+++ b/src/utils/permalinks/Permalinks.js
@@ -129,6 +129,17 @@ export class RoomPermalinkCreator {
return getPermalinkConstructor().forEvent(this._roomId, eventId, this._serverCandidates);
}
+ forShareableRoom() {
+ if (this._room) {
+ // Prefer to use canonical alias for permalink if possible
+ const alias = this._room.getCanonicalAlias();
+ if (alias) {
+ return getPermalinkConstructor().forRoom(alias, this._serverCandidates);
+ }
+ }
+ return getPermalinkConstructor().forRoom(this._roomId, this._serverCandidates);
+ }
+
forRoom() {
return getPermalinkConstructor().forRoom(this._roomId, this._serverCandidates);
}
diff --git a/test/components/structures/auth/Login-test.js b/test/components/structures/auth/Login-test.js
index 7ca210ff93..9b73137936 100644
--- a/test/components/structures/auth/Login-test.js
+++ b/test/components/structures/auth/Login-test.js
@@ -52,7 +52,7 @@ describe('Login', function() {
// Set non-empty flows & matrixClient to get past the loading spinner
root.setState({
- currentFlow: "m.login.password",
+ flows: [{ type: "m.login.password" }],
});
const form = ReactTestUtils.findRenderedComponentWithType(
@@ -61,10 +61,7 @@ describe('Login', function() {
);
expect(form).toBeTruthy();
- const changeServerLink = ReactTestUtils.findRenderedDOMComponentWithClass(
- root,
- 'mx_AuthBody_editServerDetails',
- );
+ const changeServerLink = ReactTestUtils.findRenderedDOMComponentWithClass(root, 'mx_ServerPicker_change');
expect(changeServerLink).toBeTruthy();
});
@@ -77,7 +74,7 @@ describe('Login', function() {
// Set non-empty flows & matrixClient to get past the loading spinner
root.setState({
- currentFlow: "m.login.password",
+ flows: [{ type: "m.login.password" }],
});
const form = ReactTestUtils.findRenderedComponentWithType(
@@ -86,10 +83,70 @@ describe('Login', function() {
);
expect(form).toBeTruthy();
- const changeServerLinks = ReactTestUtils.scryRenderedDOMComponentsWithClass(
- root,
- 'mx_AuthBody_editServerDetails',
- );
+ const changeServerLinks = ReactTestUtils.scryRenderedDOMComponentsWithClass(root, 'mx_ServerPicker_change');
expect(changeServerLinks).toHaveLength(0);
});
+
+ it("should show SSO button if that flow is available", () => {
+ jest.spyOn(SdkConfig, "get").mockReturnValue({
+ disable_custom_urls: true,
+ });
+
+ const root = render();
+
+ // Set non-empty flows & matrixClient to get past the loading spinner
+ root.setState({
+ flows: [{ type: "m.login.sso" }],
+ });
+
+ const ssoButton = ReactTestUtils.findRenderedDOMComponentWithClass(root, "mx_SSOButton");
+ expect(ssoButton).toBeTruthy();
+ });
+
+ it("should show both SSO button and username+password if both are available", () => {
+ jest.spyOn(SdkConfig, "get").mockReturnValue({
+ disable_custom_urls: true,
+ });
+
+ const root = render();
+
+ // Set non-empty flows & matrixClient to get past the loading spinner
+ root.setState({
+ flows: [{ type: "m.login.password" }, { type: "m.login.sso" }],
+ });
+
+ const form = ReactTestUtils.findRenderedComponentWithType(root, sdk.getComponent('auth.PasswordLogin'));
+ expect(form).toBeTruthy();
+
+ const ssoButton = ReactTestUtils.findRenderedDOMComponentWithClass(root, "mx_SSOButton");
+ expect(ssoButton).toBeTruthy();
+ });
+
+ it("should show multiple SSO buttons if multiple identity_providers are available", () => {
+ jest.spyOn(SdkConfig, "get").mockReturnValue({
+ disable_custom_urls: true,
+ });
+
+ const root = render();
+
+ // Set non-empty flows & matrixClient to get past the loading spinner
+ root.setState({
+ flows: [{
+ "type": "m.login.sso",
+ "org.matrix.msc2858.identity_providers": [{
+ id: "a",
+ name: "Provider 1",
+ }, {
+ id: "b",
+ name: "Provider 2",
+ }, {
+ id: "c",
+ name: "Provider 3",
+ }],
+ }],
+ });
+
+ const ssoButtons = ReactTestUtils.scryRenderedDOMComponentsWithClass(root, "mx_SSOButton");
+ expect(ssoButtons.length).toBe(3);
+ });
});
diff --git a/test/components/structures/auth/Registration-test.js b/test/components/structures/auth/Registration-test.js
index bf26763a79..3e8e887329 100644
--- a/test/components/structures/auth/Registration-test.js
+++ b/test/components/structures/auth/Registration-test.js
@@ -48,12 +48,9 @@ describe('Registration', function() {
/>, parentDiv);
}
- it('should show server type selector', function() {
+ it('should show server picker', function() {
const root = render();
- const selector = ReactTestUtils.findRenderedComponentWithType(
- root,
- sdk.getComponent('auth.ServerTypeSelector'),
- );
+ const selector = ReactTestUtils.findRenderedDOMComponentWithClass(root, "mx_ServerPicker");
expect(selector).toBeTruthy();
});
@@ -79,4 +76,27 @@ describe('Registration', function() {
);
expect(form).toBeTruthy();
});
+
+ it("should show SSO options if those are available", () => {
+ jest.spyOn(SdkConfig, "get").mockReturnValue({
+ disable_custom_urls: true,
+ });
+
+ const root = render();
+
+ // Set non-empty flows & matrixClient to get past the loading spinner
+ root.setState({
+ flows: [{
+ stages: [],
+ }],
+ ssoFlow: {
+ type: "m.login.sso",
+ },
+ matrixClient: {},
+ busy: false,
+ });
+
+ const ssoButton = ReactTestUtils.findRenderedDOMComponentWithClass(root, "mx_SSOButton");
+ expect(ssoButton).toBeTruthy();
+ });
});
diff --git a/test/components/views/rooms/SendMessageComposer-test.js b/test/components/views/rooms/SendMessageComposer-test.js
index 83a9388609..6eeac7ceea 100644
--- a/test/components/views/rooms/SendMessageComposer-test.js
+++ b/test/components/views/rooms/SendMessageComposer-test.js
@@ -18,8 +18,10 @@ import Adapter from "enzyme-adapter-react-16";
import { configure, mount } from "enzyme";
import React from "react";
import {act} from "react-dom/test-utils";
-
-import SendMessageComposer, {createMessageContent} from "../../../../src/components/views/rooms/SendMessageComposer";
+import SendMessageComposer, {
+ createMessageContent,
+ isQuickReaction,
+} from "../../../../src/components/views/rooms/SendMessageComposer";
import MatrixClientContext from "../../../../src/contexts/MatrixClientContext";
import EditorModel from "../../../../src/editor/model";
import {createPartCreator, createRenderer} from "../../../editor/mock";
@@ -227,6 +229,42 @@ describe('', () => {
});
});
});
+
+ describe("isQuickReaction", () => {
+ it("correctly detects quick reaction", () => {
+ const model = new EditorModel([], createPartCreator(), createRenderer());
+ model.update("+😊", "insertText", {offset: 3, atNodeEnd: true});
+
+ const isReaction = isQuickReaction(model);
+
+ expect(isReaction).toBeTruthy();
+ });
+
+ it("correctly detects quick reaction with space", () => {
+ const model = new EditorModel([], createPartCreator(), createRenderer());
+ model.update("+ 😊", "insertText", {offset: 4, atNodeEnd: true});
+
+ const isReaction = isQuickReaction(model);
+
+ expect(isReaction).toBeTruthy();
+ });
+
+ it("correctly rejects quick reaction with extra text", () => {
+ const model = new EditorModel([], createPartCreator(), createRenderer());
+ const model2 = new EditorModel([], createPartCreator(), createRenderer());
+ const model3 = new EditorModel([], createPartCreator(), createRenderer());
+ const model4 = new EditorModel([], createPartCreator(), createRenderer());
+ model.update("+😊hello", "insertText", {offset: 8, atNodeEnd: true});
+ model2.update(" +😊", "insertText", {offset: 4, atNodeEnd: true});
+ model3.update("+ 😊😊", "insertText", {offset: 6, atNodeEnd: true});
+ model4.update("+smiley", "insertText", {offset: 7, atNodeEnd: true});
+
+ expect(isQuickReaction(model)).toBeFalsy();
+ expect(isQuickReaction(model2)).toBeFalsy();
+ expect(isQuickReaction(model3)).toBeFalsy();
+ expect(isQuickReaction(model4)).toBeFalsy();
+ });
+ });
});
diff --git a/test/end-to-end-tests/.gitignore b/test/end-to-end-tests/.gitignore
index afca1ddcb3..61f9012393 100644
--- a/test/end-to-end-tests/.gitignore
+++ b/test/end-to-end-tests/.gitignore
@@ -1,3 +1,3 @@
node_modules
*.png
-riot/env
+element/env
diff --git a/test/end-to-end-tests/README.md b/test/end-to-end-tests/README.md
index 8794ef6c9b..b173fb86c2 100644
--- a/test/end-to-end-tests/README.md
+++ b/test/end-to-end-tests/README.md
@@ -5,9 +5,9 @@ This directory contains tests for matrix-react-sdk. The tests fire up a headless
## Setup
Run `./install.sh`. This will:
- - install Synapse, fetches the master branch at the moment. If anything fails here, please refer to the Synapse README to see if you're missing one of the prerequisites.
- - install Riot, this fetches the master branch at the moment.
- - install dependencies (will download copy of chrome)
+ - install Synapse, fetches the develop branch at the moment. If anything fails here, please refer to the Synapse README to see if you're missing one of the prerequisites.
+ - install Element Web, this fetches the develop branch at the moment.
+ - install dependencies (will download copy of Chrome)
## Running the tests
@@ -15,7 +15,7 @@ Run tests with `./run.sh`.
### Debug tests locally.
-`./run.sh` will run the tests against the Riot copy present in `riot/riot-web` served by a static Python HTTP server. You can symlink your `riot-web` develop copy here but that doesn't work well with Webpack recompiling. You can run the test runner directly and specify parameters to get more insight into a failure or run the tests against your local Webpack server.
+`./run.sh` will run the tests against the Element copy present in `element/element-web` served by a static Python HTTP server. You can symlink your `element-web` develop copy here but that doesn't work well with Webpack recompiling. You can run the test runner directly and specify parameters to get more insight into a failure or run the tests against your local Webpack server.
```
./synapse/stop.sh && \
@@ -26,8 +26,7 @@ It's important to always stop and start Synapse before each run of the tests to
start.js accepts these parameters (and more, see `node start.js --help`) that can help running the tests locally:
- - `--riot-url ` don't use the Riot copy and static server provided by the tests, but use a running server like the Webpack watch server to run the tests against. Make sure to have the following local config:
- - `welcomeUserId` disabled as the tests assume there is no riot-bot currently.
+ - `--app-url ` don't use the Element Web copy and static server provided by the tests, but use a running server like the Webpack watch server to run the tests against.
- `--slow-mo` type at a human speed, useful with `--windowed`.
- `--throttle-cpu ` throttle cpu in the browser by the given factor. Useful to reproduce failures because of insufficient timeouts happening on the slower CI server.
- `--windowed` run the tests in an actual browser window Try to limit interacting with the windows while the tests are running. Hovering over the window tends to fail the tests, dragging the title bar should be fine though.
diff --git a/test/end-to-end-tests/Windows.md b/test/end-to-end-tests/Windows.md
index 39b06a9a62..f6ea87d0af 100644
--- a/test/end-to-end-tests/Windows.md
+++ b/test/end-to-end-tests/Windows.md
@@ -5,14 +5,14 @@ and start following these steps to get going:
1. Navigate to your working directory (`cd /mnt/c/users/travisr/whatever/matrix-react-sdk` for example).
2. Run `sudo apt-get install unzip python3 virtualenv dos2unix`
-3. Run `dos2unix ./test/end-to-end-tests/*.sh ./test/end-to-end-tests/synapse/*.sh ./test/end-to-end-tests/riot/*.sh`
+3. Run `dos2unix ./test/end-to-end-tests/*.sh ./test/end-to-end-tests/synapse/*.sh ./test/end-to-end-tests/element/*.sh`
4. Install NodeJS for ubuntu:
```bash
curl -sL https://deb.nodesource.com/setup_12.x | sudo -E bash -
sudo apt-get update
sudo apt-get install nodejs
```
-5. Start Riot on Windows through `yarn start`
+5. Start Element on Windows through `yarn start`
6. While that builds... Run:
```bash
sudo apt-get install x11-apps
@@ -25,11 +25,11 @@ and start following these steps to get going:
cd ./test/end-to-end-tests
./synapse/install.sh
./install.sh
- ./run.sh --riot-url http://localhost:8080 --no-sandbox
+ ./run.sh --app-url http://localhost:8080 --no-sandbox
```
Note that using `yarn test:e2e` probably won't work for you. You might also have to use the config.json from the
-`riot/config-template` directory in order to actually succeed at the tests.
+`element/config-template` directory in order to actually succeed at the tests.
Also note that you'll have to use `--no-sandbox` otherwise Chrome will complain that there's no sandbox available. You
could probably fix this with enough effort, or you could run a headless Chrome in the WSL container without a sandbox.
diff --git a/test/end-to-end-tests/element/.gitignore b/test/end-to-end-tests/element/.gitignore
new file mode 100644
index 0000000000..57fac2072f
--- /dev/null
+++ b/test/end-to-end-tests/element/.gitignore
@@ -0,0 +1,2 @@
+element-web
+element.pid
diff --git a/test/end-to-end-tests/riot/config-template/config.json b/test/end-to-end-tests/element/config-template/config.json
similarity index 92%
rename from test/end-to-end-tests/riot/config-template/config.json
rename to test/end-to-end-tests/element/config-template/config.json
index b647d0bec8..b90fefc2cb 100644
--- a/test/end-to-end-tests/riot/config-template/config.json
+++ b/test/end-to-end-tests/element/config-template/config.json
@@ -8,7 +8,7 @@
"brand": "Element",
"integrations_ui_url": "https://scalar.vector.im/",
"integrations_rest_url": "https://scalar.vector.im/api",
- "bug_report_endpoint_url": "https://riot.im/bugreports/submit",
+ "bug_report_endpoint_url": "https://element.io/bugreports/submit",
"showLabsSettings": true,
"default_federate": true,
"welcomePageUrl": "home.html",
diff --git a/test/end-to-end-tests/riot/install-webserver.sh b/test/end-to-end-tests/element/install-webserver.sh
similarity index 100%
rename from test/end-to-end-tests/riot/install-webserver.sh
rename to test/end-to-end-tests/element/install-webserver.sh
diff --git a/test/end-to-end-tests/element/install.sh b/test/end-to-end-tests/element/install.sh
new file mode 100755
index 0000000000..e38f795df1
--- /dev/null
+++ b/test/end-to-end-tests/element/install.sh
@@ -0,0 +1,16 @@
+#!/bin/bash
+set -e
+ELEMENT_BRANCH=develop
+
+if [ -d $BASE_DIR/element-web ]; then
+ echo "Element is already installed"
+ exit
+fi
+
+curl -L https://github.com/vector-im/element-web/archive/${ELEMENT_BRANCH}.zip --output element.zip
+unzip -q element.zip
+rm element.zip
+mv element-web-${ELEMENT_BRANCH} element-web
+cd element-web
+yarn install
+yarn run build
diff --git a/test/end-to-end-tests/riot/start.sh b/test/end-to-end-tests/element/start.sh
similarity index 87%
rename from test/end-to-end-tests/riot/start.sh
rename to test/end-to-end-tests/element/start.sh
index be226ed257..b344f91a19 100755
--- a/test/end-to-end-tests/riot/start.sh
+++ b/test/end-to-end-tests/element/start.sh
@@ -3,7 +3,7 @@ set -e
PORT=5000
BASE_DIR=$(cd $(dirname $0) && pwd)
-PIDFILE=$BASE_DIR/riot.pid
+PIDFILE=$BASE_DIR/element.pid
CONFIG_BACKUP=config.e2etests_backup.json
if [ -f $PIDFILE ]; then
@@ -11,8 +11,8 @@ if [ -f $PIDFILE ]; then
fi
cd $BASE_DIR/
-echo -n "starting riot on http://localhost:$PORT ... "
-pushd riot-web/webapp/ > /dev/null
+echo -n "Starting Element on http://localhost:$PORT ... "
+pushd element-web/webapp/ > /dev/null
# backup config file before we copy template
if [ -f config.json ]; then
@@ -34,7 +34,7 @@ LOGFILE=$(mktemp)
# NOT expected SIGTERM (128 + 15)
# from stop.sh?
if [ $RESULT -ne 143 ]; then
- echo "failed"
+ echo "Failed"
cat $LOGFILE
rm $PIDFILE 2> /dev/null
fi
@@ -49,6 +49,6 @@ sleep 0.5 &
wait -n; RESULT=$?
# return exit code of first child to exit
if [ $RESULT -eq 0 ]; then
- echo "running"
+ echo "Running"
fi
exit $RESULT
diff --git a/test/end-to-end-tests/riot/stop.sh b/test/end-to-end-tests/element/stop.sh
similarity index 79%
rename from test/end-to-end-tests/riot/stop.sh
rename to test/end-to-end-tests/element/stop.sh
index eb99fa11cc..e39f0077eb 100755
--- a/test/end-to-end-tests/riot/stop.sh
+++ b/test/end-to-end-tests/element/stop.sh
@@ -2,19 +2,19 @@
set -e
BASE_DIR=$(cd $(dirname $0) && pwd)
-PIDFILE=riot.pid
+PIDFILE=element.pid
CONFIG_BACKUP=config.e2etests_backup.json
cd $BASE_DIR
if [ -f $PIDFILE ]; then
- echo "stopping riot server ..."
+ echo "Stopping Element server ..."
PID=$(cat $PIDFILE)
rm $PIDFILE
kill $PID
# revert config file
- cd riot-web/webapp
+ cd element-web/webapp
rm config.json
if [ -f $CONFIG_BACKUP ]; then
mv $CONFIG_BACKUP config.json
diff --git a/test/end-to-end-tests/has_custom_riot.js b/test/end-to-end-tests/has-custom-app.js
similarity index 83%
rename from test/end-to-end-tests/has_custom_riot.js
rename to test/end-to-end-tests/has-custom-app.js
index 95f32d8ad0..00184088fd 100644
--- a/test/end-to-end-tests/has_custom_riot.js
+++ b/test/end-to-end-tests/has-custom-app.js
@@ -15,10 +15,10 @@ limitations under the License.
*/
// used from run.sh as getopts doesn't support long parameters
-const idx = process.argv.indexOf("--riot-url");
-let hasRiotUrl = false;
+const idx = process.argv.indexOf("--app-url");
+let hasAppUrl = false;
if (idx !== -1) {
const value = process.argv[idx + 1];
- hasRiotUrl = !!value;
+ hasAppUrl = !!value;
}
-process.stdout.write(hasRiotUrl ? "1" : "0" );
+process.stdout.write(hasAppUrl ? "1" : "0" );
diff --git a/test/end-to-end-tests/riot/.gitignore b/test/end-to-end-tests/riot/.gitignore
deleted file mode 100644
index 0f07d8e498..0000000000
--- a/test/end-to-end-tests/riot/.gitignore
+++ /dev/null
@@ -1,2 +0,0 @@
-riot-web
-riot.pid
\ No newline at end of file
diff --git a/test/end-to-end-tests/riot/install.sh b/test/end-to-end-tests/riot/install.sh
deleted file mode 100755
index f66ab3224e..0000000000
--- a/test/end-to-end-tests/riot/install.sh
+++ /dev/null
@@ -1,16 +0,0 @@
-#!/bin/bash
-set -e
-RIOT_BRANCH=develop
-
-if [ -d $BASE_DIR/riot-web ]; then
- echo "riot is already installed"
- exit
-fi
-
-curl -L https://github.com/vector-im/riot-web/archive/${RIOT_BRANCH}.zip --output riot.zip
-unzip -q riot.zip
-rm riot.zip
-mv riot-web-${RIOT_BRANCH} riot-web
-cd riot-web
-yarn install
-yarn run build
diff --git a/test/end-to-end-tests/run.sh b/test/end-to-end-tests/run.sh
index b9d589eed9..4421bddc34 100755
--- a/test/end-to-end-tests/run.sh
+++ b/test/end-to-end-tests/run.sh
@@ -9,16 +9,16 @@ echo "Please first run $BASE_DIR/install.sh"
exit 1
fi
-has_custom_riot=$(node has_custom_riot.js $@)
+has_custom_app=$(node has-custom-app.js $@)
-if [ ! -d "riot/riot-web" ] && [ $has_custom_riot -ne "1" ]; then
- echo "Please provide an instance of riot to test against by passing --riot-url or running $BASE_DIR/riot/install.sh"
+if [ ! -d "element/element-web" ] && [ $has_custom_app -ne "1" ]; then
+ echo "Please provide an instance of Element to test against by passing --element-url or running $BASE_DIR/element/install.sh"
exit 1
fi
stop_servers() {
- if [ $has_custom_riot -ne "1" ]; then
- ./riot/stop.sh
+ if [ $has_custom_app -ne "1" ]; then
+ ./element/stop.sh
fi
./synapse/stop.sh
}
@@ -32,8 +32,8 @@ handle_error() {
trap 'handle_error' ERR
./synapse/start.sh
-if [ $has_custom_riot -ne "1" ]; then
- ./riot/start.sh
+if [ $has_custom_app -ne "1" ]; then
+ ./element/start.sh
fi
node start.js $@
stop_servers
diff --git a/test/end-to-end-tests/src/session.js b/test/end-to-end-tests/src/session.js
index 907ee2fb8e..433baa5e48 100644
--- a/test/end-to-end-tests/src/session.js
+++ b/test/end-to-end-tests/src/session.js
@@ -22,12 +22,12 @@ const {delay} = require('./util');
const DEFAULT_TIMEOUT = 20000;
-module.exports = class RiotSession {
- constructor(browser, page, username, riotserver, hsUrl) {
+module.exports = class ElementSession {
+ constructor(browser, page, username, elementServer, hsUrl) {
this.browser = browser;
this.page = page;
this.hsUrl = hsUrl;
- this.riotserver = riotserver;
+ this.elementServer = elementServer;
this.username = username;
this.consoleLog = new LogBuffer(page, "console", (msg) => `${msg.text()}\n`);
this.networkLog = new LogBuffer(page, "requestfinished", async (req) => {
@@ -38,7 +38,7 @@ module.exports = class RiotSession {
this.log = new Logger(this.username);
}
- static async create(username, puppeteerOptions, riotserver, hsUrl, throttleCpuFactor = 1) {
+ static async create(username, puppeteerOptions, elementServer, hsUrl, throttleCpuFactor = 1) {
const browser = await puppeteer.launch(puppeteerOptions);
const page = await browser.newPage();
await page.setViewport({
@@ -50,7 +50,7 @@ module.exports = class RiotSession {
console.log("throttling cpu by a factor of", throttleCpuFactor);
await client.send('Emulation.setCPUThrottlingRate', { rate: throttleCpuFactor });
}
- return new RiotSession(browser, page, username, riotserver, hsUrl);
+ return new ElementSession(browser, page, username, elementServer, hsUrl);
}
async tryGetInnertext(selector) {
@@ -194,7 +194,7 @@ module.exports = class RiotSession {
}
url(path) {
- return this.riotserver + path;
+ return this.elementServer + path;
}
delay(ms) {
diff --git a/test/end-to-end-tests/src/usecases/signup.js b/test/end-to-end-tests/src/usecases/signup.js
index ef8a259091..804cee9599 100644
--- a/test/end-to-end-tests/src/usecases/signup.js
+++ b/test/end-to-end-tests/src/usecases/signup.js
@@ -22,24 +22,12 @@ module.exports = async function signup(session, username, password, homeserver)
await session.goto(session.url('/#/register'));
// change the homeserver by clicking the advanced section
if (homeserver) {
- const advancedButton = await session.query('.mx_ServerTypeSelector_type_Advanced');
- await advancedButton.click();
+ const changeButton = await session.query('.mx_ServerPicker_change');
+ await changeButton.click();
- // depending on what HS is configured as the default, the advanced registration
- // goes the HS/IS entry directly (for matrix.org) or takes you to the user/pass entry (not matrix.org).
- // To work with both, we look for the "Change" link in the user/pass entry but don't fail when we can't find it
- // As this link should be visible immediately, and to not slow down the case where it isn't present,
- // pick a lower timeout of 5000ms
- try {
- const changeHsField = await session.query('.mx_AuthBody_editServerDetails', 5000);
- if (changeHsField) {
- await changeHsField.click();
- }
- } catch (err) {}
-
- const hsInputField = await session.query('#mx_ServerConfig_hsUrl');
+ const hsInputField = await session.query('.mx_ServerPickerDialog_otherHomeserver');
await session.replaceInputText(hsInputField, homeserver);
- const nextButton = await session.query('.mx_Login_submit');
+ const nextButton = await session.query('.mx_ServerPickerDialog_continue');
// accept homeserver
await nextButton.click();
}
@@ -68,7 +56,7 @@ module.exports = async function signup(session, username, password, homeserver)
await registerButton.click();
//confirm dialog saying you cant log back in without e-mail
- const continueButton = await session.query('.mx_QuestionDialog button.mx_Dialog_primary');
+ const continueButton = await session.query('.mx_RegistrationEmailPromptDialog button.mx_Dialog_primary');
await continueButton.click();
//find the privacy policy checkbox and check it
diff --git a/test/end-to-end-tests/start.js b/test/end-to-end-tests/start.js
index 6c80608903..234d60da9f 100644
--- a/test/end-to-end-tests/start.js
+++ b/test/end-to-end-tests/start.js
@@ -14,7 +14,7 @@ See the License for the specific language governing permissions and
limitations under the License.
*/
-const RiotSession = require('./src/session');
+const ElementSession = require('./src/session');
const scenario = require('./src/scenario');
const RestSessionCreator = require('./src/rest/creator');
const fs = require("fs");
@@ -22,7 +22,7 @@ const fs = require("fs");
const program = require('commander');
program
.option('--no-logs', "don't output logs, document html on error", false)
- .option('--riot-url [url]', "riot url to test", "http://localhost:5000")
+ .option('--app-url [url]', "url to test", "http://localhost:5000")
.option('--windowed', "dont run tests headless", false)
.option('--slow-mo', "type at a human speed", false)
.option('--dev-tools', "open chrome devtools in browser window", false)
@@ -57,7 +57,7 @@ async function runTests() {
);
async function createSession(username) {
- const session = await RiotSession.create(username, options, program.riotUrl, hsUrl, program.throttleCpu);
+ const session = await ElementSession.create(username, options, program.appUrl, hsUrl, program.throttleCpu);
sessions.push(session);
return session;
}
diff --git a/test/end-to-end-tests/synapse/config-templates/consent/homeserver.yaml b/test/end-to-end-tests/synapse/config-templates/consent/homeserver.yaml
index 536c017b9e..deb750666f 100644
--- a/test/end-to-end-tests/synapse/config-templates/consent/homeserver.yaml
+++ b/test/end-to-end-tests/synapse/config-templates/consent/homeserver.yaml
@@ -875,8 +875,8 @@ password_config:
# Enable sending emails for notification events
-# Defining a custom URL for Riot is only needed if email notifications
-# should contain links to a self-hosted installation of Riot; when set
+# Defining a custom URL for Element is only needed if email notifications
+# should contain links to a self-hosted installation of Element; when set
# the "app_name" setting is ignored.
#
# If your SMTP server requires authentication, the optional smtp_user &
@@ -897,7 +897,7 @@ email:
notif_template_html: notif_mail.html
notif_template_text: notif_mail.txt
notif_for_new_users: True
- riot_base_url: "http://localhost/riot"
+ client_base_url: "http://localhost/element"
#password_providers:
diff --git a/test/utils/permalinks/Permalinks-test.js b/test/utils/permalinks/Permalinks-test.js
index 72cd66cb69..0bd4466d97 100644
--- a/test/utils/permalinks/Permalinks-test.js
+++ b/test/utils/permalinks/Permalinks-test.js
@@ -34,6 +34,7 @@ function mockRoom(roomId, members, serverACL) {
return {
roomId,
+ getCanonicalAlias: () => roomId,
getJoinedMembers: () => members,
getMember: (userId) => members.find(m => m.userId === userId),
currentState: {
diff --git a/yarn.lock b/yarn.lock
index c06494d319..1a6a0b0fb3 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -1256,10 +1256,10 @@
dependencies:
regenerator-runtime "^0.13.4"
-"@babel/runtime@^7.11.2":
- version "7.11.2"
- resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.11.2.tgz#f549c13c754cc40b87644b9fa9f09a6a95fe0736"
- integrity sha512-TeWkU52so0mPtDcaCTxNBI/IHiz0pZgr8VEFqXFtZWpYD08ZB6FaSwVAS8MKRQAP3bYKiVjwysOJgMFY28o6Tw==
+"@babel/runtime@^7.12.5":
+ version "7.12.5"
+ resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.12.5.tgz#410e7e487441e1b360c29be715d870d9b985882e"
+ integrity sha512-plcc+hbExy3McchJCEQG3knOsuh3HH+Prx1P6cLIkET/0dLuQDEnrT+s27Axgc9bqfsmNUNHfscgMUdBpC9xfg==
dependencies:
regenerator-runtime "^0.13.4"
@@ -4842,9 +4842,9 @@ has@^1.0.1, has@^1.0.3:
function-bind "^1.1.1"
highlight.js@^10.1.2:
- version "10.1.2"
- resolved "https://registry.yarnpkg.com/highlight.js/-/highlight.js-10.1.2.tgz#c20db951ba1c22c055010648dfffd7b2a968e00c"
- integrity sha512-Q39v/Mn5mfBlMff9r+zzA+gWxRsCRKwEMvYTiisLr/XUiFI/4puWt0Ojdko3R3JCNWGdOWaA5g/Yxqa23kC5AA==
+ version "10.4.1"
+ resolved "https://registry.yarnpkg.com/highlight.js/-/highlight.js-10.4.1.tgz#d48fbcf4a9971c4361b3f95f302747afe19dbad0"
+ integrity sha512-yR5lWvNz7c85OhVAEAeFhVCc/GV4C30Fjzc/rCP0aCWzc1UUOPUk55dK/qdwTZHBvMZo+eZ2jpk62ndX/xMFlg==
hoist-non-react-statics@^3.3.0:
version "3.3.2"
@@ -6390,10 +6390,10 @@ log-symbols@^2.0.0, log-symbols@^2.2.0:
dependencies:
chalk "^2.0.1"
-loglevel@^1.7.0:
- version "1.7.0"
- resolved "https://registry.yarnpkg.com/loglevel/-/loglevel-1.7.0.tgz#728166855a740d59d38db01cf46f042caa041bb0"
- integrity sha512-i2sY04nal5jDcagM3FMfG++T69GEEM8CYuOfeOIvmXzOIcwE9a/CJPR0MFM97pYMj/u10lzz7/zd7+qwhrBTqQ==
+loglevel@^1.7.1:
+ version "1.7.1"
+ resolved "https://registry.yarnpkg.com/loglevel/-/loglevel-1.7.1.tgz#005fde2f5e6e47068f935ff28573e125ef72f197"
+ integrity sha512-Hesni4s5UkWkwCGJMQGAh71PaLUmKFM60dHvq0zi/vDhhrzuk+4GgNbTXJ12YYQJn6ZKBDNIjYcuQGKudvqrIw==
lolex@^5.0.0, lolex@^5.1.2:
version "5.1.2"
@@ -6513,15 +6513,15 @@ mathml-tag-names@^2.0.1:
integrity sha512-APMBEanjybaPzUrfqU0IMU5I0AswKMH7k8OTLs0vvV4KZpExkTkY87nR/zpbuTPj+gARop7aGUbl11pnDfW6xg==
"matrix-js-sdk@github:matrix-org/matrix-js-sdk#develop":
- version "9.2.0"
- resolved "https://codeload.github.com/matrix-org/matrix-js-sdk/tar.gz/6661bde6088e6e43f31198e8532432e162aef33c"
+ version "9.3.0"
+ resolved "https://codeload.github.com/matrix-org/matrix-js-sdk/tar.gz/ff6612f9d0aa1a7c08b65a0b41c5ab997506016f"
dependencies:
- "@babel/runtime" "^7.11.2"
+ "@babel/runtime" "^7.12.5"
another-json "^0.2.0"
browser-request "^0.3.3"
bs58 "^4.0.1"
content-type "^1.0.4"
- loglevel "^1.7.0"
+ loglevel "^1.7.1"
qs "^6.9.4"
request "^2.88.2"
unhomoglyph "^1.0.6"
@@ -6994,6 +6994,10 @@ object.values@^1.1.1:
function-bind "^1.1.1"
has "^1.0.3"
+"olm@https://packages.matrix.org/npm/olm/olm-3.2.1.tgz":
+ version "3.2.1"
+ resolved "https://packages.matrix.org/npm/olm/olm-3.2.1.tgz#d623d76f99c3518dde68be8c86618d68bc7b004a"
+
once@^1.3.0, once@^1.3.1, once@^1.4.0:
version "1.4.0"
resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1"