mirror of
https://github.com/element-hq/element-web.git
synced 2024-12-15 13:52:38 +03:00
Merge branch 'develop' into luke/fix-render-1-1-avatars-when-others-leave
This commit is contained in:
commit
121b776e8a
378 changed files with 67241 additions and 11932 deletions
BIN
.DS_Store
vendored
BIN
.DS_Store
vendored
Binary file not shown.
2
.babelrc
2
.babelrc
|
@ -1,4 +1,4 @@
|
||||||
{
|
{
|
||||||
"presets": ["react", "es2015", "es2016"],
|
"presets": ["react", "es2015", "es2016"],
|
||||||
"plugins": ["transform-class-properties", "transform-object-rest-spread", "transform-async-to-generator", "transform-runtime", "add-module-exports"]
|
"plugins": ["transform-class-properties", "transform-object-rest-spread", "transform-async-to-bluebird", "transform-runtime", "add-module-exports"]
|
||||||
}
|
}
|
||||||
|
|
23
.editorconfig
Normal file
23
.editorconfig
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
# Copyright 2017 Aviral Dasgupta
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
charset=utf-8
|
||||||
|
end_of_line = lf
|
||||||
|
insert_final_newline = true
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 4
|
||||||
|
trim_trailing_whitespace = true
|
1
.eslintignore
Normal file
1
.eslintignore
Normal file
|
@ -0,0 +1 @@
|
||||||
|
src/component-index.js
|
109
.eslintignore.errorfiles
Normal file
109
.eslintignore.errorfiles
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
# autogenerated file: run scripts/generate-eslint-error-ignore-file to update.
|
||||||
|
|
||||||
|
src/autocomplete/AutocompleteProvider.js
|
||||||
|
src/autocomplete/Autocompleter.js
|
||||||
|
src/autocomplete/EmojiProvider.js
|
||||||
|
src/autocomplete/UserProvider.js
|
||||||
|
src/CallHandler.js
|
||||||
|
src/component-index.js
|
||||||
|
src/components/structures/ContextualMenu.js
|
||||||
|
src/components/structures/CreateRoom.js
|
||||||
|
src/components/structures/LoggedInView.js
|
||||||
|
src/components/structures/login/ForgotPassword.js
|
||||||
|
src/components/structures/login/Login.js
|
||||||
|
src/components/structures/login/Registration.js
|
||||||
|
src/components/structures/MessagePanel.js
|
||||||
|
src/components/structures/NotificationPanel.js
|
||||||
|
src/components/structures/RoomStatusBar.js
|
||||||
|
src/components/structures/RoomView.js
|
||||||
|
src/components/structures/ScrollPanel.js
|
||||||
|
src/components/structures/TimelinePanel.js
|
||||||
|
src/components/structures/UploadBar.js
|
||||||
|
src/components/views/avatars/BaseAvatar.js
|
||||||
|
src/components/views/avatars/MemberAvatar.js
|
||||||
|
src/components/views/create_room/RoomAlias.js
|
||||||
|
src/components/views/dialogs/ChatCreateOrReuseDialog.js
|
||||||
|
src/components/views/dialogs/DeactivateAccountDialog.js
|
||||||
|
src/components/views/dialogs/UnknownDeviceDialog.js
|
||||||
|
src/components/views/elements/AddressSelector.js
|
||||||
|
src/components/views/elements/DeviceVerifyButtons.js
|
||||||
|
src/components/views/elements/DirectorySearchBox.js
|
||||||
|
src/components/views/elements/EditableText.js
|
||||||
|
src/components/views/elements/MemberEventListSummary.js
|
||||||
|
src/components/views/elements/TintableSvg.js
|
||||||
|
src/components/views/elements/UserSelector.js
|
||||||
|
src/components/views/login/CountryDropdown.js
|
||||||
|
src/components/views/login/InteractiveAuthEntryComponents.js
|
||||||
|
src/components/views/login/PasswordLogin.js
|
||||||
|
src/components/views/login/RegistrationForm.js
|
||||||
|
src/components/views/login/ServerConfig.js
|
||||||
|
src/components/views/messages/MFileBody.js
|
||||||
|
src/components/views/messages/MImageBody.js
|
||||||
|
src/components/views/messages/RoomAvatarEvent.js
|
||||||
|
src/components/views/messages/TextualBody.js
|
||||||
|
src/components/views/room_settings/AliasSettings.js
|
||||||
|
src/components/views/room_settings/ColorSettings.js
|
||||||
|
src/components/views/room_settings/UrlPreviewSettings.js
|
||||||
|
src/components/views/rooms/Autocomplete.js
|
||||||
|
src/components/views/rooms/AuxPanel.js
|
||||||
|
src/components/views/rooms/EntityTile.js
|
||||||
|
src/components/views/rooms/EventTile.js
|
||||||
|
src/components/views/rooms/LinkPreviewWidget.js
|
||||||
|
src/components/views/rooms/MemberDeviceInfo.js
|
||||||
|
src/components/views/rooms/MemberInfo.js
|
||||||
|
src/components/views/rooms/MemberList.js
|
||||||
|
src/components/views/rooms/MemberTile.js
|
||||||
|
src/components/views/rooms/MessageComposer.js
|
||||||
|
src/components/views/rooms/MessageComposerInput.js
|
||||||
|
src/components/views/rooms/ReadReceiptMarker.js
|
||||||
|
src/components/views/rooms/RoomList.js
|
||||||
|
src/components/views/rooms/RoomPreviewBar.js
|
||||||
|
src/components/views/rooms/RoomSettings.js
|
||||||
|
src/components/views/rooms/RoomTile.js
|
||||||
|
src/components/views/rooms/SearchableEntityList.js
|
||||||
|
src/components/views/rooms/SearchResultTile.js
|
||||||
|
src/components/views/rooms/TopUnreadMessagesBar.js
|
||||||
|
src/components/views/rooms/UserTile.js
|
||||||
|
src/components/views/settings/AddPhoneNumber.js
|
||||||
|
src/components/views/settings/ChangeAvatar.js
|
||||||
|
src/components/views/settings/ChangeDisplayName.js
|
||||||
|
src/components/views/settings/ChangePassword.js
|
||||||
|
src/components/views/settings/DevicesPanel.js
|
||||||
|
src/ContentMessages.js
|
||||||
|
src/HtmlUtils.js
|
||||||
|
src/ImageUtils.js
|
||||||
|
src/languageHandler.js
|
||||||
|
src/linkify-matrix.js
|
||||||
|
src/Login.js
|
||||||
|
src/Markdown.js
|
||||||
|
src/MatrixClientPeg.js
|
||||||
|
src/Modal.js
|
||||||
|
src/Notifier.js
|
||||||
|
src/PlatformPeg.js
|
||||||
|
src/Presence.js
|
||||||
|
src/ratelimitedfunc.js
|
||||||
|
src/RichText.js
|
||||||
|
src/Roles.js
|
||||||
|
src/Rooms.js
|
||||||
|
src/ScalarAuthClient.js
|
||||||
|
src/UiEffects.js
|
||||||
|
src/Unread.js
|
||||||
|
src/utils/DecryptFile.js
|
||||||
|
src/utils/DMRoomMap.js
|
||||||
|
src/utils/FormattingUtils.js
|
||||||
|
src/utils/MultiInviter.js
|
||||||
|
src/utils/Receipt.js
|
||||||
|
src/Velociraptor.js
|
||||||
|
src/VelocityBounce.js
|
||||||
|
src/WhoIsTyping.js
|
||||||
|
src/wrappers/withMatrixClient.js
|
||||||
|
test/components/structures/login/Registration-test.js
|
||||||
|
test/components/structures/MessagePanel-test.js
|
||||||
|
test/components/structures/ScrollPanel-test.js
|
||||||
|
test/components/structures/TimelinePanel-test.js
|
||||||
|
test/components/views/dialogs/InteractiveAuthDialog-test.js
|
||||||
|
test/components/views/elements/MemberEventListSummary-test.js
|
||||||
|
test/components/views/login/RegistrationForm-test.js
|
||||||
|
test/components/views/rooms/MessageComposerInput-test.js
|
||||||
|
test/mock-clock.js
|
||||||
|
test/stores/RoomViewStore-test.js
|
117
.eslintrc
117
.eslintrc
|
@ -1,117 +0,0 @@
|
||||||
{
|
|
||||||
"parser": "babel-eslint",
|
|
||||||
"plugins": [
|
|
||||||
"react",
|
|
||||||
"flowtype"
|
|
||||||
],
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 6,
|
|
||||||
"sourceType": "module",
|
|
||||||
"ecmaFeatures": {
|
|
||||||
"jsx": true,
|
|
||||||
"impliedStrict": true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"env": {
|
|
||||||
"browser": true,
|
|
||||||
"amd": true,
|
|
||||||
"es6": true,
|
|
||||||
"node": true,
|
|
||||||
"mocha": true
|
|
||||||
},
|
|
||||||
"extends": ["eslint:recommended", "plugin:react/recommended"],
|
|
||||||
"rules": {
|
|
||||||
"no-undef": ["warn"],
|
|
||||||
"global-strict": ["off"],
|
|
||||||
"no-extra-semi": ["warn"],
|
|
||||||
"no-underscore-dangle": ["off"],
|
|
||||||
"no-console": ["off"],
|
|
||||||
"no-unused-vars": ["off"],
|
|
||||||
"no-trailing-spaces": ["warn", {
|
|
||||||
"skipBlankLines": true
|
|
||||||
}],
|
|
||||||
"no-unreachable": ["warn"],
|
|
||||||
"no-spaced-func": ["warn"],
|
|
||||||
"no-new-func": ["error"],
|
|
||||||
"no-new-wrappers": ["error"],
|
|
||||||
"no-invalid-regexp": ["error"],
|
|
||||||
"no-extra-bind": ["error"],
|
|
||||||
"no-magic-numbers": ["error", {
|
|
||||||
"ignore": [-1, 0, 1], // usually used in array/string indexing
|
|
||||||
"ignoreArrayIndexes": true,
|
|
||||||
"enforceConst": true,
|
|
||||||
"detectObjects": true
|
|
||||||
}],
|
|
||||||
"consistent-return": ["error"],
|
|
||||||
"valid-jsdoc": ["error"],
|
|
||||||
"no-use-before-define": ["error"],
|
|
||||||
"camelcase": ["warn"],
|
|
||||||
"array-callback-return": ["error"],
|
|
||||||
"dot-location": ["warn", "property"],
|
|
||||||
"guard-for-in": ["error"],
|
|
||||||
"no-useless-call": ["warn"],
|
|
||||||
"no-useless-escape": ["warn"],
|
|
||||||
"no-useless-concat": ["warn"],
|
|
||||||
"brace-style": ["warn", "1tbs"],
|
|
||||||
"comma-style": ["warn", "last"],
|
|
||||||
"space-before-function-paren": ["warn", "never"],
|
|
||||||
"space-before-blocks": ["warn", "always"],
|
|
||||||
"keyword-spacing": ["warn", {
|
|
||||||
"before": true,
|
|
||||||
"after": true
|
|
||||||
}],
|
|
||||||
|
|
||||||
// dangling commas required, but only for multiline objects/arrays
|
|
||||||
"comma-dangle": ["warn", "always-multiline"],
|
|
||||||
// always === instead of ==, unless dealing with null/undefined
|
|
||||||
"eqeqeq": ["error", "smart"],
|
|
||||||
// always use curly braces, even with single statements
|
|
||||||
"curly": ["error", "all"],
|
|
||||||
// phasing out var in favour of let/const is a good idea
|
|
||||||
"no-var": ["warn"],
|
|
||||||
// always require semicolons
|
|
||||||
"semi": ["error", "always"],
|
|
||||||
// prefer rest and spread over the Old Ways
|
|
||||||
"prefer-spread": ["warn"],
|
|
||||||
"prefer-rest-params": ["warn"],
|
|
||||||
|
|
||||||
/** react **/
|
|
||||||
|
|
||||||
// bind or arrow function in props causes performance issues
|
|
||||||
"react/jsx-no-bind": ["error", {
|
|
||||||
"ignoreRefs": true
|
|
||||||
}],
|
|
||||||
"react/jsx-key": ["error"],
|
|
||||||
"react/prefer-stateless-function": ["warn"],
|
|
||||||
|
|
||||||
/** flowtype **/
|
|
||||||
"flowtype/require-parameter-type": [
|
|
||||||
1,
|
|
||||||
{
|
|
||||||
"excludeArrowFunctions": true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"flowtype/define-flow-type": 1,
|
|
||||||
"flowtype/require-return-type": [
|
|
||||||
1,
|
|
||||||
"always",
|
|
||||||
{
|
|
||||||
"annotateUndefined": "never",
|
|
||||||
"excludeArrowFunctions": true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"flowtype/space-after-type-colon": [
|
|
||||||
1,
|
|
||||||
"always"
|
|
||||||
],
|
|
||||||
"flowtype/space-before-type-colon": [
|
|
||||||
1,
|
|
||||||
"never"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"settings": {
|
|
||||||
"flowtype": {
|
|
||||||
"onlyFilesWithFlowAnnotation": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
102
.eslintrc.js
Normal file
102
.eslintrc.js
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
// get the path of the js-sdk so we can extend the config
|
||||||
|
// eslint supports loading extended configs by module,
|
||||||
|
// but only if they come from a module that starts with eslint-config-
|
||||||
|
// So we load the filename directly (and it could be in node_modules/
|
||||||
|
// or or ../node_modules/ etc)
|
||||||
|
const matrixJsSdkPath = path.dirname(require.resolve('matrix-js-sdk'));
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
parser: "babel-eslint",
|
||||||
|
extends: [matrixJsSdkPath + "/.eslintrc.js"],
|
||||||
|
plugins: [
|
||||||
|
"react",
|
||||||
|
"flowtype",
|
||||||
|
"babel"
|
||||||
|
],
|
||||||
|
env: {
|
||||||
|
es6: true,
|
||||||
|
},
|
||||||
|
parserOptions: {
|
||||||
|
ecmaFeatures: {
|
||||||
|
jsx: true,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
// eslint's built in no-invalid-this rule breaks with class properties
|
||||||
|
"no-invalid-this": "off",
|
||||||
|
// so we replace it with a version that is class property aware
|
||||||
|
"babel/no-invalid-this": "error",
|
||||||
|
|
||||||
|
// We appear to follow this most of the time, so let's enforce it instead
|
||||||
|
// of occasionally following it (or catching it in review)
|
||||||
|
"keyword-spacing": "error",
|
||||||
|
|
||||||
|
/** react **/
|
||||||
|
// This just uses the react plugin to help eslint known when
|
||||||
|
// variables have been used in JSX
|
||||||
|
"react/jsx-uses-vars": "error",
|
||||||
|
// Don't mark React as unused if we're using JSX
|
||||||
|
"react/jsx-uses-react": "error",
|
||||||
|
|
||||||
|
// bind or arrow function in props causes performance issues
|
||||||
|
"react/jsx-no-bind": ["error", {
|
||||||
|
"ignoreRefs": true,
|
||||||
|
}],
|
||||||
|
"react/jsx-key": ["error"],
|
||||||
|
|
||||||
|
// Assert no spacing in JSX curly brackets
|
||||||
|
// <Element prop={ consideredError} prop={notConsideredError} />
|
||||||
|
//
|
||||||
|
// https://github.com/yannickcr/eslint-plugin-react/blob/HEAD/docs/rules/jsx-curly-spacing.md
|
||||||
|
"react/jsx-curly-spacing": ["error", {"when": "never", "children": {"when": "always"}}],
|
||||||
|
|
||||||
|
// Assert spacing before self-closing JSX tags, and no spacing before or
|
||||||
|
// after the closing slash, and no spacing after the opening bracket of
|
||||||
|
// the opening tag or closing tag.
|
||||||
|
//
|
||||||
|
// https://github.com/yannickcr/eslint-plugin-react/blob/HEAD/docs/rules/jsx-tag-spacing.md
|
||||||
|
"react/jsx-tag-spacing": ["error"],
|
||||||
|
|
||||||
|
/** flowtype **/
|
||||||
|
"flowtype/require-parameter-type": ["warn", {
|
||||||
|
"excludeArrowFunctions": true,
|
||||||
|
}],
|
||||||
|
"flowtype/define-flow-type": "warn",
|
||||||
|
"flowtype/require-return-type": ["warn",
|
||||||
|
"always",
|
||||||
|
{
|
||||||
|
"annotateUndefined": "never",
|
||||||
|
"excludeArrowFunctions": true,
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"flowtype/space-after-type-colon": ["warn", "always"],
|
||||||
|
"flowtype/space-before-type-colon": ["warn", "never"],
|
||||||
|
|
||||||
|
/*
|
||||||
|
* things that are errors in the js-sdk config that the current
|
||||||
|
* code does not adhere to, turned down to warn
|
||||||
|
*/
|
||||||
|
"max-len": ["warn", {
|
||||||
|
// apparently people believe the length limit shouldn't apply
|
||||||
|
// to JSX.
|
||||||
|
ignorePattern: '^\\s*<',
|
||||||
|
ignoreComments: true,
|
||||||
|
code: 120,
|
||||||
|
}],
|
||||||
|
"valid-jsdoc": ["warn"],
|
||||||
|
"new-cap": ["warn"],
|
||||||
|
"key-spacing": ["warn"],
|
||||||
|
"arrow-parens": ["warn"],
|
||||||
|
"prefer-const": ["warn"],
|
||||||
|
|
||||||
|
// crashes currently: https://github.com/eslint/eslint/issues/6274
|
||||||
|
"generator-star-spacing": "off",
|
||||||
|
},
|
||||||
|
settings: {
|
||||||
|
flowtype: {
|
||||||
|
onlyFilesWithFlowAnnotation: true
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
6
.flowconfig
Normal file
6
.flowconfig
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
[include]
|
||||||
|
src/**/*.js
|
||||||
|
test/**/*.js
|
||||||
|
|
||||||
|
[ignore]
|
||||||
|
node_modules/
|
5
.gitignore
vendored
5
.gitignore
vendored
|
@ -9,3 +9,8 @@ npm-debug.log
|
||||||
|
|
||||||
# test reports created by karma
|
# test reports created by karma
|
||||||
/karma-reports
|
/karma-reports
|
||||||
|
|
||||||
|
/.idea
|
||||||
|
/src/component-index.js
|
||||||
|
|
||||||
|
.DS_Store
|
||||||
|
|
33
.travis-test-riot.sh
Executable file
33
.travis-test-riot.sh
Executable file
|
@ -0,0 +1,33 @@
|
||||||
|
#!/bin/bash
|
||||||
|
#
|
||||||
|
# script which is run by the travis build (after `npm run test`).
|
||||||
|
#
|
||||||
|
# clones riot-web develop and runs the tests against our version of react-sdk.
|
||||||
|
|
||||||
|
set -ev
|
||||||
|
|
||||||
|
RIOT_WEB_DIR=riot-web
|
||||||
|
REACT_SDK_DIR=`pwd`
|
||||||
|
|
||||||
|
curbranch="${TRAVIS_PULL_REQUEST_BRANCH:-$TRAVIS_BRANCH}"
|
||||||
|
echo "Determined branch to be $curbranch"
|
||||||
|
|
||||||
|
git clone https://github.com/vector-im/riot-web.git \
|
||||||
|
"$RIOT_WEB_DIR"
|
||||||
|
|
||||||
|
cd "$RIOT_WEB_DIR"
|
||||||
|
|
||||||
|
git checkout "$curbranch" || git checkout develop
|
||||||
|
|
||||||
|
mkdir node_modules
|
||||||
|
npm install
|
||||||
|
|
||||||
|
# use the version of js-sdk we just used in the react-sdk tests
|
||||||
|
rm -r node_modules/matrix-js-sdk
|
||||||
|
ln -s "$REACT_SDK_DIR/node_modules/matrix-js-sdk" node_modules/matrix-js-sdk
|
||||||
|
|
||||||
|
# ... and, of course, the version of react-sdk we just built
|
||||||
|
rm -r node_modules/matrix-react-sdk
|
||||||
|
ln -s "$REACT_SDK_DIR" node_modules/matrix-react-sdk
|
||||||
|
|
||||||
|
npm run test
|
17
.travis.yml
17
.travis.yml
|
@ -1,3 +1,20 @@
|
||||||
|
# we need trusty for the chrome addon
|
||||||
|
dist: trusty
|
||||||
|
|
||||||
|
# we don't need sudo, so can run in a container, which makes startup much
|
||||||
|
# quicker.
|
||||||
|
#
|
||||||
|
# unfortunately we do temporarily require sudo as a workaround for
|
||||||
|
# https://github.com/travis-ci/travis-ci/issues/8836
|
||||||
|
sudo: required
|
||||||
|
|
||||||
language: node_js
|
language: node_js
|
||||||
node_js:
|
node_js:
|
||||||
- node # Latest stable version of nodejs.
|
- node # Latest stable version of nodejs.
|
||||||
|
addons:
|
||||||
|
chrome: stable
|
||||||
|
install:
|
||||||
|
- npm install
|
||||||
|
- (cd node_modules/matrix-js-sdk && npm install)
|
||||||
|
script:
|
||||||
|
./scripts/travis.sh
|
||||||
|
|
1738
CHANGELOG.md
1738
CHANGELOG.md
File diff suppressed because it is too large
Load diff
37
README.md
37
README.md
|
@ -12,17 +12,21 @@ a 'skin'. A skin provides:
|
||||||
* Zero or more 'modules' containing non-UI functionality
|
* Zero or more 'modules' containing non-UI functionality
|
||||||
|
|
||||||
**WARNING: As of July 2016, the skinning abstraction is broken due to rapid
|
**WARNING: As of July 2016, the skinning abstraction is broken due to rapid
|
||||||
development of `matrix-react-sdk` to meet the needs of Vector, the first app
|
development of `matrix-react-sdk` to meet the needs of Riot (codenamed Vector), the first app
|
||||||
to be built on top of the SDK** (https://github.com/vector-im/vector-web).
|
to be built on top of the SDK** (https://github.com/vector-im/riot-web).
|
||||||
Right now `matrix-react-sdk` depends on some functionality from `vector-web`
|
Right now `matrix-react-sdk` depends on some functionality from `riot-web`
|
||||||
(e.g. CSS), and `matrix-react-sdk` contains some Vector specific behaviour
|
(e.g. CSS), and `matrix-react-sdk` contains some Riot specific behaviour
|
||||||
(grep for 'vector'). This layering will be fixed asap once Vector development
|
(grep for 'vector'). This layering will be fixed asap once Riot development
|
||||||
has stabilised, but for now we do not advise trying to create new skins for
|
has stabilised, but for now we do not advise trying to create new skins for
|
||||||
matrix-react-sdk until the layers are clearly separated again.
|
matrix-react-sdk until the layers are clearly separated again.
|
||||||
|
|
||||||
In the interim, `vector-im/vector-web` and `matrix-org/matrix-react-sdk` should
|
In the interim, `vector-im/riot-web` and `matrix-org/matrix-react-sdk` should
|
||||||
be considered as a single project (for instance, matrix-react-sdk bugs
|
be considered as a single project (for instance, matrix-react-sdk bugs
|
||||||
are currently filed against vector-im/vector-web rather than this project).
|
are currently filed against vector-im/riot-web rather than this project).
|
||||||
|
|
||||||
|
Translation Status
|
||||||
|
==================
|
||||||
|
[![Translation status](https://translate.riot.im/widgets/riot-web/-/multi-auto.svg)](https://translate.riot.im/engage/riot-web/?utm_source=widget)
|
||||||
|
|
||||||
Developer Guide
|
Developer Guide
|
||||||
===============
|
===============
|
||||||
|
@ -42,17 +46,17 @@ Please follow the standard Matrix contributor's guide:
|
||||||
https://github.com/matrix-org/synapse/tree/master/CONTRIBUTING.rst
|
https://github.com/matrix-org/synapse/tree/master/CONTRIBUTING.rst
|
||||||
|
|
||||||
Please follow the Matrix JS/React code style as per:
|
Please follow the Matrix JS/React code style as per:
|
||||||
https://github.com/matrix-org/matrix-react-sdk/tree/master/code_style.rst
|
https://github.com/matrix-org/matrix-react-sdk/blob/master/code_style.md
|
||||||
|
|
||||||
Whilst the layering separation between matrix-react-sdk and Vector is broken
|
Whilst the layering separation between matrix-react-sdk and Riot is broken
|
||||||
(as of July 2016), code should be committed as follows:
|
(as of July 2016), code should be committed as follows:
|
||||||
* All new components: https://github.com/matrix-org/matrix-react-sdk/tree/master/src/components
|
* All new components: https://github.com/matrix-org/matrix-react-sdk/tree/master/src/components
|
||||||
* Vector-specific components: https://github.com/vector-im/vector-web/tree/master/src/components
|
* Riot-specific components: https://github.com/vector-im/riot-web/tree/master/src/components
|
||||||
* In practice, `matrix-react-sdk` is still evolving so fast that the maintenance
|
* In practice, `matrix-react-sdk` is still evolving so fast that the maintenance
|
||||||
burden of customising and overriding these components for Vector can seriously
|
burden of customising and overriding these components for Riot can seriously
|
||||||
impede development. So right now, there should be very few (if any) customisations for Vector.
|
impede development. So right now, there should be very few (if any) customisations for Riot.
|
||||||
* CSS for Matrix SDK components: https://github.com/vector-im/vector-web/tree/master/src/skins/vector/css/matrix-react-sdk
|
* CSS for Matrix SDK components: https://github.com/vector-im/riot-web/tree/master/src/skins/vector/css/matrix-react-sdk
|
||||||
* CSS for Vector-specific overrides and components: https://github.com/vector-im/vector-web/tree/master/src/skins/vector/css/vector-web
|
* CSS for Riot-specific overrides and components: https://github.com/vector-im/riot-web/tree/master/src/skins/vector/css/riot-web
|
||||||
|
|
||||||
React components in matrix-react-sdk are come in two different flavours:
|
React components in matrix-react-sdk are come in two different flavours:
|
||||||
'structures' and 'views'. Structures are stateful components which handle the
|
'structures' and 'views'. Structures are stateful components which handle the
|
||||||
|
@ -76,7 +80,7 @@ practices that anyone working with the SDK needs to be be aware of and uphold:
|
||||||
|
|
||||||
* The view's CSS file MUST have the same name (e.g. view/rooms/MessageTile.css).
|
* The view's CSS file MUST have the same name (e.g. view/rooms/MessageTile.css).
|
||||||
CSS for matrix-react-sdk currently resides in
|
CSS for matrix-react-sdk currently resides in
|
||||||
https://github.com/vector-im/vector-web/tree/master/src/skins/vector/css/matrix-react-sdk.
|
https://github.com/vector-im/riot-web/tree/master/src/skins/vector/css/matrix-react-sdk.
|
||||||
|
|
||||||
* Per-view CSS is optional - it could choose to inherit all its styling from
|
* Per-view CSS is optional - it could choose to inherit all its styling from
|
||||||
the context of the rest of the app, although this is unusual for any but
|
the context of the rest of the app, although this is unusual for any but
|
||||||
|
@ -129,7 +133,7 @@ from it.
|
||||||
Github Issues
|
Github Issues
|
||||||
=============
|
=============
|
||||||
|
|
||||||
All issues should be filed under https://github.com/vector-im/vector-web/issues
|
All issues should be filed under https://github.com/vector-im/riot-web/issues
|
||||||
for now.
|
for now.
|
||||||
|
|
||||||
OUTDATED: To Create Your Own Skin
|
OUTDATED: To Create Your Own Skin
|
||||||
|
@ -190,4 +194,3 @@ Alternative instructions:
|
||||||
* Create an index.html file pulling in your compiled javascript and the
|
* Create an index.html file pulling in your compiled javascript and the
|
||||||
CSS bundle from the skin you use. For now, you'll also need to manually
|
CSS bundle from the skin you use. For now, you'll also need to manually
|
||||||
import CSS from any skins that your skin inherts from.
|
import CSS from any skins that your skin inherts from.
|
||||||
|
|
||||||
|
|
|
@ -69,25 +69,41 @@ General Style
|
||||||
console.log("I am a fish"); // Bad
|
console.log("I am a fish"); // Bad
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
- No new line before else, catch, finally, etc:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
if (x) {
|
||||||
|
console.log("I am a fish");
|
||||||
|
} else {
|
||||||
|
console.log("I am a chimp"); // Good
|
||||||
|
}
|
||||||
|
|
||||||
|
if (x) {
|
||||||
|
console.log("I am a fish");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
console.log("I am a chimp"); // Bad
|
||||||
|
}
|
||||||
|
```
|
||||||
- Declare one variable per var statement (consistent with Node). Unless they
|
- Declare one variable per var statement (consistent with Node). Unless they
|
||||||
are simple and closely related. If you put the next declaration on a new line,
|
are simple and closely related. If you put the next declaration on a new line,
|
||||||
treat yourself to another `var`:
|
treat yourself to another `var`:
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
var key = "foo",
|
const key = "foo",
|
||||||
comparator = function(x, y) {
|
comparator = function(x, y) {
|
||||||
return x - y;
|
return x - y;
|
||||||
}; // Bad
|
}; // Bad
|
||||||
|
|
||||||
var key = "foo";
|
const key = "foo";
|
||||||
var comparator = function(x, y) {
|
const comparator = function(x, y) {
|
||||||
return x - y;
|
return x - y;
|
||||||
}; // Good
|
}; // Good
|
||||||
|
|
||||||
var x = 0, y = 0; // Fine
|
let x = 0, y = 0; // Fine
|
||||||
|
|
||||||
var x = 0;
|
let x = 0;
|
||||||
var y = 0; // Also fine
|
let y = 0; // Also fine
|
||||||
```
|
```
|
||||||
- A single line `if` is fine, all others have braces. This prevents errors when adding to the code.:
|
- A single line `if` is fine, all others have braces. This prevents errors when adding to the code.:
|
||||||
|
|
||||||
|
|
151
docs/settings.md
Normal file
151
docs/settings.md
Normal file
|
@ -0,0 +1,151 @@
|
||||||
|
# Settings Reference
|
||||||
|
|
||||||
|
This document serves as developer documentation for using "Granular Settings". Granular Settings allow users to specify different values for a setting at particular levels of interest. For example, a user may say that in a particular room they want URL previews off, but in all other rooms they want them enabled. The `SettingsStore` helps mask the complexity of dealing with the different levels and exposes easy to use getters and setters.
|
||||||
|
|
||||||
|
|
||||||
|
## Levels
|
||||||
|
|
||||||
|
Granular Settings rely on a series of known levels in order to use the correct value for the scenario. These levels, in order of prioirty, are:
|
||||||
|
* `device` - The current user's device
|
||||||
|
* `room-device` - The current user's device, but only when in a specific room
|
||||||
|
* `room-account` - The current user's account, but only when in a specific room
|
||||||
|
* `account` - The current user's account
|
||||||
|
* `room` - A specific room (setting for all members of the room)
|
||||||
|
* `config` - Values are defined by `config.json`
|
||||||
|
* `default` - The hardcoded default for the settings
|
||||||
|
|
||||||
|
Individual settings may control which levels are appropriate for them as part of the defaults. This is often to ensure that room administrators cannot force account-only settings upon participants.
|
||||||
|
|
||||||
|
|
||||||
|
## Settings
|
||||||
|
|
||||||
|
Settings are the different options a user may set or experience in the application. These are pre-defined in `src/settings/Settings.js` under the `SETTINGS` constant and have the following minimum requirements:
|
||||||
|
```
|
||||||
|
// The ID is used to reference the setting throughout the application. This must be unique.
|
||||||
|
"theSettingId": {
|
||||||
|
// The levels this setting supports is required. In `src/settings/Settings.js` there are various pre-set arrays
|
||||||
|
// for this option - they should be used where possible to avoid copy/pasting arrays across settings.
|
||||||
|
supportedLevels: [...],
|
||||||
|
|
||||||
|
// The default for this setting serves two purposes: It provides a value if the setting is not defined at other
|
||||||
|
// levels, and it serves to demonstrate the expected type to other developers. The value isn't enforced, but it
|
||||||
|
// should be respected throughout the code. The default may be any data type.
|
||||||
|
default: false,
|
||||||
|
|
||||||
|
// The display name has two notations: string and object. The object notation allows for different translatable
|
||||||
|
// strings to be used for different levels, while the string notation represents the string for all levels.
|
||||||
|
|
||||||
|
displayName: _td("Change something"), // effectively `displayName: { "default": _td("Change something") }`
|
||||||
|
displayName: {
|
||||||
|
"room": _td("Change something for participants of this room"),
|
||||||
|
|
||||||
|
// Note: the default will be used if the level requested (such as `device`) does not have a string defined here.
|
||||||
|
"default": _td("Change something"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Getting values for a setting
|
||||||
|
|
||||||
|
After importing `SettingsStore`, simply make a call to `SettingsStore.getValue`. The `roomId` parameter should always be supplied where possible, even if the setting does not have a per-room level value. This is to ensure that the value returned is best represented in the room, particularly if the setting ever gets a per-room level in the future.
|
||||||
|
|
||||||
|
In settings pages it is often desired to have the value at a particular level instead of getting the calculated value. Call `SettingsStore.getValueAt` to get the value of a setting at a particular level, and optionally make it explicitly at that level. By default `getValueAt` will traverse the tree starting at the provided level; making it explicit means it will not go beyond the provided level. When using `getValueAt`, please be sure to use `SettingLevel` to represent the target level.
|
||||||
|
|
||||||
|
### Setting values for a setting
|
||||||
|
|
||||||
|
Values are defined at particular levels and should be done in a safe manner. There are two checks to perform to ensure a clean save: is the level supported and can the user actually set the value. In most cases, neither should be an issue although there are circumstances where this changes. An example of a safe call is:
|
||||||
|
```javascript
|
||||||
|
const isSupported = SettingsStore.isLevelSupported(SettingLevel.ROOM);
|
||||||
|
if (isSupported) {
|
||||||
|
const canSetValue = SettingsStore.canSetValue("mySetting", "!curbf:matrix.org", SettingLevel.ROOM);
|
||||||
|
if (canSetValue) {
|
||||||
|
SettingsStore.setValue("mySetting", "!curbf:matrix.org", SettingLevel.ROOM, newValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
These checks may also be performed in different areas of the application to avoid the verbose example above. For instance, the component which allows changing the setting may be hidden conditionally on the above conditions.
|
||||||
|
|
||||||
|
##### `SettingsFlag` component
|
||||||
|
|
||||||
|
Where possible, the `SettingsFlag` component should be used to set simple "flip-a-bit" (true/false) settings. The `SettingsFlag` also supports simple radio button options, such as the theme the user would like to use.
|
||||||
|
```html
|
||||||
|
<SettingsFlag name="theSettingId"
|
||||||
|
level={SettingsLevel.ROOM}
|
||||||
|
roomId="!curbf:matrix.org"
|
||||||
|
label={_td("Your label here")} // optional, if falsey then the `SettingsStore` will be used
|
||||||
|
onChange={function(newValue) { }} // optional, called after saving
|
||||||
|
isExplicit={false} // this is passed along to `SettingsStore.getValueAt`, defaulting to false
|
||||||
|
manualSave={false} // if true, saving is delayed. You will need to call .save() on this component
|
||||||
|
|
||||||
|
// Options for radio buttons
|
||||||
|
group="your-radio-group" // this enables radio button support
|
||||||
|
value="yourValueHere" // the value for this particular option
|
||||||
|
/>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Getting the display name for a setting
|
||||||
|
|
||||||
|
Simply call `SettingsStore.getDisplayName`. The appropriate display name will be returned and automatically translated for you. If a display name cannot be found, it will return `null`.
|
||||||
|
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
Occasionally some parts of the application may be undergoing testing and are not quite production ready. These are commonly known to be behind a "labs flag". Features behind lab flags must go through the granular settings system, and look and act very much normal settings. The exception is that they must supply `isFeature: true` as part of the setting definition and should go through the helper functions on `SettingsStore`.
|
||||||
|
|
||||||
|
### Determining if a feature is enabled
|
||||||
|
|
||||||
|
A simple call to `SettingsStore.isFeatureEnabled` will tell you if the feature is enabled. This will perform all the required calculations to determine if the feature is enabled based upon the configuration and user selection.
|
||||||
|
|
||||||
|
### Enabling a feature
|
||||||
|
|
||||||
|
Features can only be enabled if the feature is in the `labs` state, otherwise this is a no-op. To find the current set of features in the `labs` state, call `SettingsStore.getLabsFeatures`. To set the value, call `SettingsStore.setFeatureEnabled`.
|
||||||
|
|
||||||
|
|
||||||
|
## Setting controllers
|
||||||
|
|
||||||
|
Settings may have environmental factors that affect their value or need additional code to be called when they are modified. A setting controller is able to override the calculated value for a setting and react to changes in that setting. Controllers are not a replacement for the level handlers and should only be used to ensure the environment is kept up to date with the setting where it is otherwise not possible. An example of this is the notification settings: they can only be considered enabled if the platform supports notifications, and enabling notifications requires additional steps to actually enable notifications.
|
||||||
|
|
||||||
|
For more information, see `src/settings/controllers/SettingController.js`.
|
||||||
|
|
||||||
|
|
||||||
|
## Local echo
|
||||||
|
|
||||||
|
`SettingsStore` will perform local echo on all settings to ensure that immediately getting values does not cause a split-brain scenario. As mentioned in the "Setting values for a setting" section, the appropriate checks should be done to ensure that the user is allowed to set the value. The local echo system assumes that the user has permission and that the request will go through successfully. The local echo only takes effect until the request to save a setting has completed (either successfully or otherwise).
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
SettingsStore.setValue(...).then(() => {
|
||||||
|
// The value has actually been stored at this point.
|
||||||
|
});
|
||||||
|
SettingsStore.getValue(...); // this will return the value set in `setValue` above.
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Maintainers Reference
|
||||||
|
|
||||||
|
The granular settings system has a few complex parts to power it. This section is to document how the `SettingsStore` is supposed to work.
|
||||||
|
|
||||||
|
### General information
|
||||||
|
|
||||||
|
The `SettingsStore` uses the hardcoded `LEVEL_ORDER` constant to ensure that it is using the correct override procedure. The array is checked from left to right, simulating the behaviour of overriding values from the higher levels. Each level should be defined in this array, including `default`.
|
||||||
|
|
||||||
|
Handlers (`src/settings/handlers/SettingsHandler.js`) represent a single level and are responsible for getting and setting values at that level. Handlers also provide additional information to the `SettingsStore` such as if the level is supported or if the current user may set values at the level. The `SettingsStore` will use the handler to enforce checks and manipulate settings. Handlers are also responsible for dealing with migration patterns or legacy settings for their level (for example, a setting being renamed or using a different key from other settings in the underlying store). Handlers are provided to the `SettingsStore` via the `LEVEL_HANDLERS` constant. `SettingsStore` will optimize lookups by only considering handlers that are supported on the platform.
|
||||||
|
|
||||||
|
Local echo is achieved through `src/settings/handlers/LocalEchoWrapper.js` which acts as a wrapper around a given handler. This is automatically applied to all defined `LEVEL_HANDLERS` and proxies the calls to the wrapped handler where possible. The echo is achieved by a simple object cache stored within the class itself. The cache is invalidated immediately upon the proxied save call succeeding or failing.
|
||||||
|
|
||||||
|
Controllers are notified of changes by the `SettingsStore`, and are given the opportunity to override values after the `SettingsStore` has deemed the value calculated. Controllers are invoked as the last possible step in the code.
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
Features automatically get considered as `disabled` if they are not listed in the `SdkConfig` or `enable_labs` is false/not set. Features are always checked against the configuration before going through the level order as they have the option of being forced-on or forced-off for the application. This is done by the `features` section and looks something like this:
|
||||||
|
|
||||||
|
```
|
||||||
|
"features": {
|
||||||
|
"feature_groups": "enable",
|
||||||
|
"feature_pinning": "disable", // the default
|
||||||
|
"feature_presence": "labs"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
If `enableLabs` is true in the configuration, the default for features becomes `"labs"`.
|
1
header
1
header
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
|
15
jenkins.sh
15
jenkins.sh
|
@ -2,21 +2,26 @@
|
||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
export KARMAFLAGS="--no-colors"
|
export NVM_DIR="$HOME/.nvm"
|
||||||
export NVM_DIR="/home/jenkins/.nvm"
|
|
||||||
[ -s "$NVM_DIR/nvm.sh" ] && . "$NVM_DIR/nvm.sh"
|
[ -s "$NVM_DIR/nvm.sh" ] && . "$NVM_DIR/nvm.sh"
|
||||||
nvm use 4
|
nvm use 6
|
||||||
|
|
||||||
set -x
|
set -x
|
||||||
|
|
||||||
# install the other dependencies
|
# install the other dependencies
|
||||||
npm install
|
npm install
|
||||||
|
|
||||||
|
# we may be using a dev branch of js-sdk in which case we need to build it
|
||||||
|
(cd node_modules/matrix-js-sdk && npm install)
|
||||||
|
|
||||||
# run the mocha tests
|
# run the mocha tests
|
||||||
npm run test
|
npm run test -- --no-colors
|
||||||
|
|
||||||
# run eslint
|
# run eslint
|
||||||
npm run lint -- -f checkstyle -o eslint.xml || true
|
npm run lintall -- -f checkstyle -o eslint.xml || true
|
||||||
|
|
||||||
|
# re-run the linter, excluding any files known to have errors or warnings.
|
||||||
|
npm run lintwithexclusions
|
||||||
|
|
||||||
# delete the old tarball, if it exists
|
# delete the old tarball, if it exists
|
||||||
rm -f matrix-react-sdk-*.tgz
|
rm -f matrix-react-sdk-*.tgz
|
||||||
|
|
|
@ -55,11 +55,18 @@ module.exports = function (config) {
|
||||||
// some images to reduce noise from the tests
|
// some images to reduce noise from the tests
|
||||||
{pattern: 'test/img/*', watched: false, included: false,
|
{pattern: 'test/img/*', watched: false, included: false,
|
||||||
served: true, nocache: false},
|
served: true, nocache: false},
|
||||||
|
// translation files
|
||||||
|
{pattern: 'src/i18n/strings/*', watcheed: false, included: false, served: true},
|
||||||
|
{pattern: 'test/i18n/*', watched: false, included: false, served: true},
|
||||||
],
|
],
|
||||||
|
|
||||||
// redirect img links to the karma server
|
|
||||||
proxies: {
|
proxies: {
|
||||||
|
// redirect img links to the karma server
|
||||||
"/img/": "/base/test/img/",
|
"/img/": "/base/test/img/",
|
||||||
|
// special languages.json file for the tests
|
||||||
|
"/i18n/languages.json": "/base/test/i18n/languages.json",
|
||||||
|
// and redirect i18n requests
|
||||||
|
"/i18n/": "/base/src/i18n/strings/",
|
||||||
},
|
},
|
||||||
|
|
||||||
// list of files to exclude
|
// list of files to exclude
|
||||||
|
@ -86,7 +93,18 @@ module.exports = function (config) {
|
||||||
// test results reporter to use
|
// test results reporter to use
|
||||||
// possible values: 'dots', 'progress'
|
// possible values: 'dots', 'progress'
|
||||||
// available reporters: https://npmjs.org/browse/keyword/karma-reporter
|
// available reporters: https://npmjs.org/browse/keyword/karma-reporter
|
||||||
reporters: ['progress', 'junit'],
|
reporters: ['logcapture', 'spec', 'junit', 'summary'],
|
||||||
|
|
||||||
|
specReporter: {
|
||||||
|
suppressErrorSummary: false, // do print error summary
|
||||||
|
suppressFailed: false, // do print information about failed tests
|
||||||
|
suppressPassed: false, // do print information about passed tests
|
||||||
|
showSpecTiming: true, // print the time elapsed for each spec
|
||||||
|
},
|
||||||
|
|
||||||
|
client: {
|
||||||
|
captureLogs: true,
|
||||||
|
},
|
||||||
|
|
||||||
// web server port
|
// web server port
|
||||||
port: 9876,
|
port: 9876,
|
||||||
|
@ -97,7 +115,10 @@ module.exports = function (config) {
|
||||||
// level of logging
|
// level of logging
|
||||||
// possible values: config.LOG_DISABLE || config.LOG_ERROR ||
|
// possible values: config.LOG_DISABLE || config.LOG_ERROR ||
|
||||||
// config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG
|
// config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG
|
||||||
logLevel: config.LOG_INFO,
|
//
|
||||||
|
// This is strictly for logs that would be generated by the browser itself and we
|
||||||
|
// don't want to log about missing images, which are emitted on LOG_WARN.
|
||||||
|
logLevel: config.LOG_ERROR,
|
||||||
|
|
||||||
// enable / disable watching file and executing tests whenever any file
|
// enable / disable watching file and executing tests whenever any file
|
||||||
// changes
|
// changes
|
||||||
|
@ -109,11 +130,25 @@ module.exports = function (config) {
|
||||||
browsers: [
|
browsers: [
|
||||||
'Chrome',
|
'Chrome',
|
||||||
//'PhantomJS',
|
//'PhantomJS',
|
||||||
|
//'ChromeHeadless',
|
||||||
],
|
],
|
||||||
|
|
||||||
|
customLaunchers: {
|
||||||
|
'ChromeHeadless': {
|
||||||
|
base: 'Chrome',
|
||||||
|
flags: [
|
||||||
|
// See https://chromium.googlesource.com/chromium/src/+/lkgr/headless/README.md
|
||||||
|
'--headless',
|
||||||
|
'--disable-gpu',
|
||||||
|
// Without a remote debugging port, Google Chrome exits immediately.
|
||||||
|
'--remote-debugging-port=9222',
|
||||||
|
],
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
// Continuous Integration mode
|
// Continuous Integration mode
|
||||||
// if true, Karma captures browsers, runs the tests and exits
|
// if true, Karma captures browsers, runs the tests and exits
|
||||||
singleRun: true,
|
// singleRun: false,
|
||||||
|
|
||||||
// Concurrency level
|
// Concurrency level
|
||||||
// how many browser should be started simultaneous
|
// how many browser should be started simultaneous
|
||||||
|
@ -135,17 +170,24 @@ module.exports = function (config) {
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
noParse: [
|
noParse: [
|
||||||
|
// for cross platform compatibility use [\\\/] as the path separator
|
||||||
|
// this ensures that the regex trips on both Windows and *nix
|
||||||
|
|
||||||
// don't parse the languages within highlight.js. They
|
// don't parse the languages within highlight.js. They
|
||||||
// cause stack overflows
|
// cause stack overflows
|
||||||
// (https://github.com/webpack/webpack/issues/1721), and
|
// (https://github.com/webpack/webpack/issues/1721), and
|
||||||
// there is no need for webpack to parse them - they can
|
// there is no need for webpack to parse them - they can
|
||||||
// just be included as-is.
|
// just be included as-is.
|
||||||
/highlight\.js\/lib\/languages/,
|
/highlight\.js[\\\/]lib[\\\/]languages/,
|
||||||
|
|
||||||
|
// olm takes ages for webpack to process, and it's already heavily
|
||||||
|
// optimised, so there is little to gain by us uglifying it.
|
||||||
|
/olm[\\\/](javascript[\\\/])?olm\.js$/,
|
||||||
|
|
||||||
// also disable parsing for sinon, because it
|
// also disable parsing for sinon, because it
|
||||||
// tries to do voodoo with 'require' which upsets
|
// tries to do voodoo with 'require' which upsets
|
||||||
// webpack (https://github.com/webpack/webpack/issues/304)
|
// webpack (https://github.com/webpack/webpack/issues/304)
|
||||||
/sinon\/pkg\/sinon\.js$/,
|
/sinon[\\\/]pkg[\\\/]sinon\.js$/,
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
resolve: {
|
resolve: {
|
||||||
|
@ -159,12 +201,24 @@ module.exports = function (config) {
|
||||||
'sinon': 'sinon/pkg/sinon.js',
|
'sinon': 'sinon/pkg/sinon.js',
|
||||||
},
|
},
|
||||||
root: [
|
root: [
|
||||||
path.resolve('./src'),
|
|
||||||
path.resolve('./test'),
|
path.resolve('./test'),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
devtool: 'inline-source-map',
|
devtool: 'inline-source-map',
|
||||||
|
externals: {
|
||||||
|
// Don't try to bundle electron: leave it as a commonjs dependency
|
||||||
|
// (the 'commonjs' here means it will output a 'require')
|
||||||
|
"electron": "commonjs electron",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
webpackMiddleware: {
|
||||||
|
stats: {
|
||||||
|
// don't fill the console up with a mahoosive list of modules
|
||||||
|
chunks: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
browserNoActivityTimeout: 15000,
|
browserNoActivityTimeout: 15000,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
6551
package-lock.json
generated
Normal file
6551
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
77
package.json
77
package.json
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "matrix-react-sdk",
|
"name": "matrix-react-sdk",
|
||||||
"version": "0.8.2",
|
"version": "0.11.3",
|
||||||
"description": "SDK for matrix.org using React",
|
"description": "SDK for matrix.org using React",
|
||||||
"author": "matrix.org",
|
"author": "matrix.org",
|
||||||
"repository": {
|
"repository": {
|
||||||
|
@ -10,6 +10,7 @@
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"main": "lib/index.js",
|
"main": "lib/index.js",
|
||||||
"files": [
|
"files": [
|
||||||
|
".eslintrc.js",
|
||||||
"CHANGELOG.md",
|
"CHANGELOG.md",
|
||||||
"CONTRIBUTING.rst",
|
"CONTRIBUTING.rst",
|
||||||
"LICENSE",
|
"LICENSE",
|
||||||
|
@ -27,56 +28,71 @@
|
||||||
"test"
|
"test"
|
||||||
],
|
],
|
||||||
"bin": {
|
"bin": {
|
||||||
"reskindex": "scripts/reskindex.js"
|
"reskindex": "scripts/reskindex.js",
|
||||||
|
"matrix-gen-i18n": "scripts/gen-i18n.js",
|
||||||
|
"matrix-prune-i18n": "scripts/prune-i18n.js"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"reskindex": "scripts/reskindex.js -h header",
|
"reskindex": "node scripts/reskindex.js -h header",
|
||||||
"build": "node scripts/babelcheck.js && babel src -d lib --source-maps",
|
"reskindex:watch": "node scripts/reskindex.js -h header -w",
|
||||||
"start": "node scripts/babelcheck.js && babel src -w -d lib --source-maps",
|
"i18n": "matrix-gen-i18n",
|
||||||
|
"prunei18n": "matrix-prune-i18n",
|
||||||
|
"build": "npm run reskindex && babel src -d lib --source-maps --copy-files",
|
||||||
|
"build:watch": "babel src -w -d lib --source-maps --copy-files",
|
||||||
|
"emoji-data-strip": "node scripts/emoji-data-strip.js",
|
||||||
|
"start": "parallelshell \"npm run build:watch\" \"npm run reskindex:watch\"",
|
||||||
"lint": "eslint src/",
|
"lint": "eslint src/",
|
||||||
"lintall": "eslint src/ test/",
|
"lintall": "eslint src/ test/",
|
||||||
|
"lintwithexclusions": "eslint --max-warnings 0 --ignore-path .eslintignore.errorfiles src test",
|
||||||
"clean": "rimraf lib",
|
"clean": "rimraf lib",
|
||||||
"prepublish": "npm run build && git rev-parse HEAD > git-revision.txt",
|
"prepublish": "npm run clean && npm run build && git rev-parse HEAD > git-revision.txt",
|
||||||
"test": "karma start $KARMAFLAGS --browsers PhantomJS",
|
"test": "karma start --single-run=true --browsers ChromeHeadless",
|
||||||
"test-multi": "karma start $KARMAFLAGS --single-run=false"
|
"test-multi": "karma start"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"babel-runtime": "^6.11.6",
|
"babel-runtime": "^6.11.6",
|
||||||
|
"bluebird": "^3.5.0",
|
||||||
"blueimp-canvas-to-blob": "^3.5.0",
|
"blueimp-canvas-to-blob": "^3.5.0",
|
||||||
"browser-encrypt-attachment": "^0.3.0",
|
"browser-encrypt-attachment": "^0.3.0",
|
||||||
"browser-request": "^0.3.3",
|
"browser-request": "^0.3.3",
|
||||||
"classnames": "^2.1.2",
|
"classnames": "^2.1.2",
|
||||||
"draft-js": "^0.8.1",
|
"commonmark": "^0.28.1",
|
||||||
"draft-js-export-html": "^0.5.0",
|
"counterpart": "^0.18.0",
|
||||||
"draft-js-export-markdown": "^0.2.0",
|
"draft-js": "^0.11.0-alpha",
|
||||||
"emojione": "2.2.3",
|
"draft-js-export-html": "^0.6.0",
|
||||||
"filesize": "^3.1.2",
|
"draft-js-export-markdown": "^0.3.0",
|
||||||
"flux": "^2.0.3",
|
"emojione": "2.2.7",
|
||||||
|
"file-saver": "^1.3.3",
|
||||||
|
"filesize": "3.5.6",
|
||||||
|
"flux": "2.1.1",
|
||||||
"fuse.js": "^2.2.0",
|
"fuse.js": "^2.2.0",
|
||||||
"glob": "^5.0.14",
|
"glob": "^5.0.14",
|
||||||
"highlight.js": "^8.9.1",
|
"highlight.js": "^8.9.1",
|
||||||
"isomorphic-fetch": "^2.2.1",
|
"isomorphic-fetch": "^2.2.1",
|
||||||
"linkifyjs": "^2.1.3",
|
"linkifyjs": "^2.1.3",
|
||||||
"lodash": "^4.13.1",
|
"lodash": "^4.13.1",
|
||||||
"marked": "^0.3.5",
|
"matrix-js-sdk": "0.9.2",
|
||||||
"matrix-js-sdk": "matrix-org/matrix-js-sdk#develop",
|
|
||||||
"optimist": "^0.6.1",
|
"optimist": "^0.6.1",
|
||||||
"q": "^1.4.1",
|
"prop-types": "^15.5.8",
|
||||||
|
"querystring": "^0.2.0",
|
||||||
"react": "^15.4.0",
|
"react": "^15.4.0",
|
||||||
"react-addons-css-transition-group": "15.3.2",
|
"react-addons-css-transition-group": "15.3.2",
|
||||||
|
"react-beautiful-dnd": "^4.0.0",
|
||||||
"react-dom": "^15.4.0",
|
"react-dom": "^15.4.0",
|
||||||
"react-gemini-scrollbar": "matrix-org/react-gemini-scrollbar#5e97aef",
|
"react-gemini-scrollbar": "matrix-org/react-gemini-scrollbar#5e97aef",
|
||||||
"sanitize-html": "^1.11.1",
|
"sanitize-html": "^1.14.1",
|
||||||
|
"text-encoding-utf-8": "^1.0.1",
|
||||||
|
"url": "^0.11.0",
|
||||||
"velocity-vector": "vector-im/velocity#059e3b2",
|
"velocity-vector": "vector-im/velocity#059e3b2",
|
||||||
"whatwg-fetch": "^1.0.0"
|
"whatwg-fetch": "^1.0.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"babel-cli": "^6.5.2",
|
"babel-cli": "^6.5.2",
|
||||||
"babel-core": "^6.14.0",
|
"babel-core": "^6.14.0",
|
||||||
"babel-eslint": "^6.1.0",
|
"babel-eslint": "^6.1.2",
|
||||||
"babel-loader": "^6.2.5",
|
"babel-loader": "^6.2.5",
|
||||||
"babel-plugin-add-module-exports": "^0.2.1",
|
"babel-plugin-add-module-exports": "^0.2.1",
|
||||||
"babel-plugin-transform-async-to-generator": "^6.16.0",
|
"babel-plugin-transform-async-to-bluebird": "^1.1.1",
|
||||||
"babel-plugin-transform-class-properties": "^6.16.0",
|
"babel-plugin-transform-class-properties": "^6.16.0",
|
||||||
"babel-plugin-transform-object-rest-spread": "^6.16.0",
|
"babel-plugin-transform-object-rest-spread": "^6.16.0",
|
||||||
"babel-plugin-transform-runtime": "^6.15.0",
|
"babel-plugin-transform-runtime": "^6.15.0",
|
||||||
|
@ -85,26 +101,35 @@
|
||||||
"babel-preset-es2016": "^6.11.3",
|
"babel-preset-es2016": "^6.11.3",
|
||||||
"babel-preset-es2017": "^6.14.0",
|
"babel-preset-es2017": "^6.14.0",
|
||||||
"babel-preset-react": "^6.11.1",
|
"babel-preset-react": "^6.11.1",
|
||||||
"eslint": "^2.13.1",
|
"chokidar": "^1.6.1",
|
||||||
"eslint-plugin-flowtype": "^2.17.0",
|
"eslint": "^3.13.1",
|
||||||
"eslint-plugin-react": "^6.2.1",
|
"eslint-config-google": "^0.7.1",
|
||||||
|
"eslint-plugin-babel": "^4.0.1",
|
||||||
|
"eslint-plugin-flowtype": "^2.30.0",
|
||||||
|
"eslint-plugin-react": "^7.4.0",
|
||||||
|
"estree-walker": "^0.5.0",
|
||||||
"expect": "^1.16.0",
|
"expect": "^1.16.0",
|
||||||
|
"flow-parser": "^0.57.3",
|
||||||
"json-loader": "^0.5.3",
|
"json-loader": "^0.5.3",
|
||||||
"karma": "^0.13.22",
|
"karma": "^1.7.0",
|
||||||
"karma-chrome-launcher": "^0.2.3",
|
"karma-chrome-launcher": "^0.2.3",
|
||||||
"karma-cli": "^0.1.2",
|
"karma-cli": "^0.1.2",
|
||||||
"karma-junit-reporter": "^0.4.1",
|
"karma-junit-reporter": "^0.4.1",
|
||||||
|
"karma-logcapture-reporter": "0.0.1",
|
||||||
"karma-mocha": "^0.2.2",
|
"karma-mocha": "^0.2.2",
|
||||||
"karma-phantomjs-launcher": "^1.0.0",
|
|
||||||
"karma-sourcemap-loader": "^0.3.7",
|
"karma-sourcemap-loader": "^0.3.7",
|
||||||
|
"karma-spec-reporter": "^0.0.31",
|
||||||
|
"karma-summary-reporter": "^1.3.3",
|
||||||
"karma-webpack": "^1.7.0",
|
"karma-webpack": "^1.7.0",
|
||||||
|
"matrix-react-test-utils": "^0.1.1",
|
||||||
"mocha": "^2.4.5",
|
"mocha": "^2.4.5",
|
||||||
"phantomjs-prebuilt": "^2.1.7",
|
"parallelshell": "^3.0.2",
|
||||||
"react-addons-test-utils": "^15.4.0",
|
"react-addons-test-utils": "^15.4.0",
|
||||||
"require-json": "0.0.1",
|
"require-json": "0.0.1",
|
||||||
"rimraf": "^2.4.3",
|
"rimraf": "^2.4.3",
|
||||||
"sinon": "^1.17.3",
|
"sinon": "^1.17.3",
|
||||||
"source-map-loader": "^0.1.5",
|
"source-map-loader": "^0.1.5",
|
||||||
|
"walk": "^2.3.9",
|
||||||
"webpack": "^1.12.14"
|
"webpack": "^1.12.14"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,22 +0,0 @@
|
||||||
#!/usr/bin/env node
|
|
||||||
|
|
||||||
var exec = require('child_process').exec;
|
|
||||||
|
|
||||||
// Makes sure the babel executable in the path is babel 6 (or greater), not
|
|
||||||
// babel 5, which it is if you upgrade from an older version of react-sdk and
|
|
||||||
// run 'npm install' since the package has changed to babel-cli, so 'babel'
|
|
||||||
// remains installed and the executable in node_modules/.bin remains as babel
|
|
||||||
// 5.
|
|
||||||
|
|
||||||
exec("babel -V", function (error, stdout, stderr) {
|
|
||||||
if ((error && error.code) || parseInt(stdout.substr(0,1), 10) < 6) {
|
|
||||||
console.log("\033[31m\033[1m"+
|
|
||||||
'*****************************************\n'+
|
|
||||||
'* matrix-react-sdk has moved to babel 6 *\n'+
|
|
||||||
'* Please "rm -rf node_modules && npm i" *\n'+
|
|
||||||
'* then restore links as appropriate *\n'+
|
|
||||||
'*****************************************\n'+
|
|
||||||
"\033[91m");
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
});
|
|
192
scripts/check-i18n.pl
Executable file
192
scripts/check-i18n.pl
Executable file
|
@ -0,0 +1,192 @@
|
||||||
|
#!/usr/bin/perl
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Cwd 'abs_path';
|
||||||
|
|
||||||
|
# script which checks how out of sync the i18ns are drifting
|
||||||
|
|
||||||
|
# example i18n format:
|
||||||
|
# "%(oneUser)sleft": "%(oneUser)sleft",
|
||||||
|
|
||||||
|
$|=1;
|
||||||
|
|
||||||
|
$0 =~ /^(.*\/)/;
|
||||||
|
my $i18ndir = abs_path($1."/../src/i18n/strings");
|
||||||
|
my $srcdir = abs_path($1."/../src");
|
||||||
|
|
||||||
|
my $en = read_i18n($i18ndir."/en_EN.json");
|
||||||
|
|
||||||
|
my $src_strings = read_src_strings($srcdir);
|
||||||
|
my $src = {};
|
||||||
|
|
||||||
|
print "Checking strings in src\n";
|
||||||
|
foreach my $tuple (@$src_strings) {
|
||||||
|
my ($s, $file) = (@$tuple);
|
||||||
|
$src->{$s} = $file;
|
||||||
|
if (!$en->{$s}) {
|
||||||
|
if ($en->{$s . '.'}) {
|
||||||
|
printf ("%50s %24s\t%s\n", $file, "en_EN has fullstop!", $s);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$s =~ /^(.*)\.?$/;
|
||||||
|
if ($en->{$1}) {
|
||||||
|
printf ("%50s %24s\t%s\n", $file, "en_EN lacks fullstop!", $s);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
printf ("%50s %24s\t%s\n", $file, "Translation missing!", $s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
print "\nChecking en_EN\n";
|
||||||
|
my $count = 0;
|
||||||
|
my $remaining_src = {};
|
||||||
|
foreach (keys %$src) { $remaining_src->{$_}++ };
|
||||||
|
|
||||||
|
foreach my $k (sort keys %$en) {
|
||||||
|
# crappy heuristic to ignore country codes for now...
|
||||||
|
next if ($k =~ /^(..|..-..)$/);
|
||||||
|
|
||||||
|
if ($en->{$k} ne $k) {
|
||||||
|
printf ("%50s %24s\t%s\n", "en_EN", "en_EN is not symmetrical", $k);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$src->{$k}) {
|
||||||
|
if ($src->{$k. '.'}) {
|
||||||
|
printf ("%50s %24s\t%s\n", $src->{$k. '.'}, "src has fullstop!", $k);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$k =~ /^(.*)\.?$/;
|
||||||
|
if ($src->{$1}) {
|
||||||
|
printf ("%50s %24s\t%s\n", $src->{$1}, "src lacks fullstop!", $k);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
printf ("%50s %24s\t%s\n", '???', "Not present in src?", $k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$count++;
|
||||||
|
delete $remaining_src->{$k};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
printf ("$count/" . (scalar keys %$src) . " strings found in src are present in en_EN\n");
|
||||||
|
foreach (keys %$remaining_src) {
|
||||||
|
print "missing: $_\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
opendir(DIR, $i18ndir) || die $!;
|
||||||
|
my @files = readdir(DIR);
|
||||||
|
closedir(DIR);
|
||||||
|
foreach my $lang (grep { -f "$i18ndir/$_" && !/(basefile|en_EN)\.json/ } @files) {
|
||||||
|
print "\nChecking $lang\n";
|
||||||
|
|
||||||
|
my $map = read_i18n($i18ndir."/".$lang);
|
||||||
|
my $count = 0;
|
||||||
|
|
||||||
|
my $remaining_en = {};
|
||||||
|
foreach (keys %$en) { $remaining_en->{$_}++ };
|
||||||
|
|
||||||
|
foreach my $k (sort keys %$map) {
|
||||||
|
{
|
||||||
|
no warnings 'uninitialized';
|
||||||
|
my $vars = {};
|
||||||
|
while ($k =~ /%\((.*?)\)s/g) {
|
||||||
|
$vars->{$1}++;
|
||||||
|
}
|
||||||
|
while ($map->{$k} =~ /%\((.*?)\)s/g) {
|
||||||
|
$vars->{$1}--;
|
||||||
|
}
|
||||||
|
foreach my $var (keys %$vars) {
|
||||||
|
if ($vars->{$var} != 0) {
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "Broken var ($var)s", $k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($en->{$k}) {
|
||||||
|
if ($map->{$k} eq $k) {
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "Untranslated string?", $k);
|
||||||
|
}
|
||||||
|
$count++;
|
||||||
|
delete $remaining_en->{$k};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if ($en->{$k . "."}) {
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "en_EN has fullstop!", $k);
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
$k =~ /^(.*)\.?$/;
|
||||||
|
if ($en->{$1}) {
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "en_EN lacks fullstop!", $k);
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "Not present in en_EN", $k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scalar keys %$remaining_en < 100) {
|
||||||
|
foreach (keys %$remaining_en) {
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "Not yet translated", $_);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf ("$count/" . (scalar keys %$en) . " strings translated\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
sub read_i18n {
|
||||||
|
my $path = shift;
|
||||||
|
my $map = {};
|
||||||
|
$path =~ /.*\/(.*)$/;
|
||||||
|
my $lang = $1;
|
||||||
|
|
||||||
|
open(FILE, "<", $path) || die $!;
|
||||||
|
while(<FILE>) {
|
||||||
|
if ($_ =~ m/^(\s+)"(.*?)"(: *)"(.*?)"(,?)$/) {
|
||||||
|
my ($indent, $src, $colon, $dst, $comma) = ($1, $2, $3, $4, $5);
|
||||||
|
$src =~ s/\\"/"/g;
|
||||||
|
$dst =~ s/\\"/"/g;
|
||||||
|
|
||||||
|
if ($map->{$src}) {
|
||||||
|
printf ("%10s %24s\t%s\n", $lang, "Duplicate translation!", $src);
|
||||||
|
}
|
||||||
|
$map->{$src} = $dst;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
close(FILE);
|
||||||
|
|
||||||
|
return $map;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub read_src_strings {
|
||||||
|
my $path = shift;
|
||||||
|
|
||||||
|
use File::Find;
|
||||||
|
use File::Slurp;
|
||||||
|
|
||||||
|
my $strings = [];
|
||||||
|
|
||||||
|
my @files;
|
||||||
|
find( sub { push @files, $File::Find::name if (-f $_ && /\.jsx?$/) }, $path );
|
||||||
|
foreach my $file (@files) {
|
||||||
|
my $src = read_file($file);
|
||||||
|
$src =~ s/'\s*\+\s*'//g;
|
||||||
|
$src =~ s/"\s*\+\s*"//g;
|
||||||
|
|
||||||
|
$file =~ s/^.*\/src/src/;
|
||||||
|
while ($src =~ /_t(?:Jsx)?\(\s*'(.*?[^\\])'/sg) {
|
||||||
|
my $s = $1;
|
||||||
|
$s =~ s/\\'/'/g;
|
||||||
|
push @$strings, [$s, $file];
|
||||||
|
}
|
||||||
|
while ($src =~ /_t(?:Jsx)?\(\s*"(.*?[^\\])"/sg) {
|
||||||
|
push @$strings, [$1, $file];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $strings;
|
||||||
|
}
|
47
scripts/copy-i18n.py
Executable file
47
scripts/copy-i18n.py
Executable file
|
@ -0,0 +1,47 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
|
||||||
|
if len(sys.argv) < 3:
|
||||||
|
print "Usage: %s <source> <dest>" % (sys.argv[0],)
|
||||||
|
print "eg. %s pt_BR.json pt.json" % (sys.argv[0],)
|
||||||
|
print
|
||||||
|
print "Adds any translations to <dest> that exist in <source> but not <dest>"
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
srcpath = sys.argv[1]
|
||||||
|
dstpath = sys.argv[2]
|
||||||
|
tmppath = dstpath + ".tmp"
|
||||||
|
|
||||||
|
with open(srcpath) as f:
|
||||||
|
src = json.load(f)
|
||||||
|
|
||||||
|
with open(dstpath) as f:
|
||||||
|
dst = json.load(f)
|
||||||
|
|
||||||
|
toAdd = {}
|
||||||
|
for k,v in src.iteritems():
|
||||||
|
if k not in dst:
|
||||||
|
print "Adding %s" % (k,)
|
||||||
|
toAdd[k] = v
|
||||||
|
|
||||||
|
# don't just json.dumps as we'll probably re-order all the keys (and they're
|
||||||
|
# not in any given order so we can't just sort_keys). Append them to the end.
|
||||||
|
with open(dstpath) as ifp:
|
||||||
|
with open(tmppath, 'w') as ofp:
|
||||||
|
for line in ifp:
|
||||||
|
strippedline = line.strip()
|
||||||
|
if strippedline in ('{', '}'):
|
||||||
|
ofp.write(line)
|
||||||
|
elif strippedline.endswith(','):
|
||||||
|
ofp.write(line)
|
||||||
|
else:
|
||||||
|
ofp.write(' '+strippedline+',')
|
||||||
|
toAddStr = json.dumps(toAdd, indent=4, separators=(',', ': '), ensure_ascii=False, encoding="utf8").strip("{}\n")
|
||||||
|
ofp.write("\n")
|
||||||
|
ofp.write(toAddStr.encode('utf8'))
|
||||||
|
ofp.write("\n")
|
||||||
|
|
||||||
|
os.rename(tmppath, dstpath)
|
26
scripts/emoji-data-strip.js
Normal file
26
scripts/emoji-data-strip.js
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
const EMOJI_DATA = require('emojione/emoji.json');
|
||||||
|
const EMOJI_SUPPORTED = Object.keys(require('emojione').emojioneList);
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
const output = Object.keys(EMOJI_DATA).map(
|
||||||
|
(key) => {
|
||||||
|
const datum = EMOJI_DATA[key];
|
||||||
|
const newDatum = {
|
||||||
|
name: datum.name,
|
||||||
|
shortname: datum.shortname,
|
||||||
|
category: datum.category,
|
||||||
|
emoji_order: datum.emoji_order,
|
||||||
|
};
|
||||||
|
if (datum.aliases_ascii.length > 0) {
|
||||||
|
newDatum.aliases_ascii = datum.aliases_ascii;
|
||||||
|
}
|
||||||
|
return newDatum;
|
||||||
|
}
|
||||||
|
).filter((datum) => {
|
||||||
|
return EMOJI_SUPPORTED.includes(datum.shortname);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Write to a file in src. Changes should be checked into git. This file is copied by
|
||||||
|
// babel using --copy-files
|
||||||
|
fs.writeFileSync('./src/stripped-emoji.json', JSON.stringify(output));
|
114
scripts/fix-i18n.pl
Executable file
114
scripts/fix-i18n.pl
Executable file
|
@ -0,0 +1,114 @@
|
||||||
|
#!/usr/bin/perl -ni
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
# script which synchronises i18n strings to include punctuation.
|
||||||
|
# i've cherry-picked ones which seem to have diverged between the different translations
|
||||||
|
# from TextForEvent, causing missing events all over the place
|
||||||
|
|
||||||
|
BEGIN {
|
||||||
|
$::fixups = [split(/\n/, <<EOT
|
||||||
|
%(targetName)s accepted the invitation for %(displayName)s.
|
||||||
|
%(targetName)s accepted an invitation.
|
||||||
|
%(senderName)s requested a VoIP conference.
|
||||||
|
%(senderName)s invited %(targetName)s.
|
||||||
|
%(senderName)s banned %(targetName)s.
|
||||||
|
%(senderName)s changed their display name from %(oldDisplayName)s to %(displayName)s.
|
||||||
|
%(senderName)s set their display name to %(displayName)s.
|
||||||
|
%(senderName)s removed their display name (%(oldDisplayName)s).
|
||||||
|
%(senderName)s removed their profile picture.
|
||||||
|
%(senderName)s changed their profile picture.
|
||||||
|
%(senderName)s set a profile picture.
|
||||||
|
VoIP conference started.
|
||||||
|
%(targetName)s joined the room.
|
||||||
|
VoIP conference finished.
|
||||||
|
%(targetName)s rejected the invitation.
|
||||||
|
%(targetName)s left the room.
|
||||||
|
%(senderName)s unbanned %(targetName)s.
|
||||||
|
%(senderName)s kicked %(targetName)s.
|
||||||
|
%(senderName)s withdrew %(targetName)s's inivitation.
|
||||||
|
%(targetName)s left the room.
|
||||||
|
%(senderDisplayName)s changed the topic to "%(topic)s".
|
||||||
|
%(senderDisplayName)s changed the room name to %(roomName)s.
|
||||||
|
%(senderDisplayName)s sent an image.
|
||||||
|
%(senderName)s answered the call.
|
||||||
|
%(senderName)s ended the call.
|
||||||
|
%(senderName)s placed a %(callType)s call.
|
||||||
|
%(senderName)s sent an invitation to %(targetDisplayName)s to join the room.
|
||||||
|
%(senderName)s turned on end-to-end encryption (algorithm %(algorithm)s).
|
||||||
|
%(senderName)s changed the power level of %(powerLevelDiffText)s.
|
||||||
|
For security, this session has been signed out. Please sign in again.
|
||||||
|
You need to log back in to generate end-to-end encryption keys for this device and submit the public key to your homeserver. This is a once off; sorry for the inconvenience.
|
||||||
|
A new password must be entered.
|
||||||
|
Guests can't set avatars. Please register.
|
||||||
|
Failed to set avatar.
|
||||||
|
Unable to verify email address.
|
||||||
|
Guests can't use labs features. Please register.
|
||||||
|
A new password must be entered.
|
||||||
|
Resetting password will currently reset any end-to-end encryption keys on all devices, making encrypted chat history unreadable, unless you first export your room keys and re-import them afterwards. In future this will be improved.
|
||||||
|
Guests cannot join this room even if explicitly invited.
|
||||||
|
Guest users can't invite users. Please register to invite.
|
||||||
|
This room is inaccessible to guests. You may be able to join if you register.
|
||||||
|
delete the alias.
|
||||||
|
remove %(name)s from the directory.
|
||||||
|
Conference call failed.
|
||||||
|
Conference calling is in development and may not be reliable.
|
||||||
|
Guest users can't create new rooms. Please register to create room and start a chat.
|
||||||
|
Server may be unavailable, overloaded, or you hit a bug.
|
||||||
|
Server unavailable, overloaded, or something else went wrong.
|
||||||
|
You are already in a call.
|
||||||
|
You cannot place VoIP calls in this browser.
|
||||||
|
You cannot place a call with yourself.
|
||||||
|
Your email address does not appear to be associated with a Matrix ID on this Homeserver.
|
||||||
|
Guest users can't upload files. Please register to upload.
|
||||||
|
Some of your messages have not been sent.
|
||||||
|
This room is private or inaccessible to guests. You may be able to join if you register.
|
||||||
|
Tried to load a specific point in this room's timeline, but was unable to find it.
|
||||||
|
Tried to load a specific point in this room's timeline, but you do not have permission to view the message in question.
|
||||||
|
This action cannot be performed by a guest user. Please register to be able to do this.
|
||||||
|
Tried to load a specific point in this room's timeline, but was unable to find it.
|
||||||
|
Tried to load a specific point in this room's timeline, but you do not have permission to view the message in question.
|
||||||
|
You are trying to access %(roomName)s.
|
||||||
|
You will not be able to undo this change as you are promoting the user to have the same power level as yourself.
|
||||||
|
EOT
|
||||||
|
)];
|
||||||
|
}
|
||||||
|
|
||||||
|
# example i18n format:
|
||||||
|
# "%(oneUser)sleft": "%(oneUser)sleft",
|
||||||
|
|
||||||
|
# script called with the line of the file to be checked
|
||||||
|
my $sub = 0;
|
||||||
|
if ($_ =~ m/^(\s+)"(.*?)"(: *)"(.*?)"(,?)$/) {
|
||||||
|
my ($indent, $src, $colon, $dst, $comma) = ($1, $2, $3, $4, $5);
|
||||||
|
$src =~ s/\\"/"/g;
|
||||||
|
$dst =~ s/\\"/"/g;
|
||||||
|
|
||||||
|
foreach my $fixup (@{$::fixups}) {
|
||||||
|
my $dotless_fixup = substr($fixup, 0, -1);
|
||||||
|
|
||||||
|
if ($src eq $dotless_fixup) {
|
||||||
|
print STDERR "fixing up src: $src\n";
|
||||||
|
$src .= '.';
|
||||||
|
$sub = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($ARGV !~ /(zh_Hans|zh_Hant|th)\.json$/ && $src eq $fixup && $dst !~ /\.$/) {
|
||||||
|
print STDERR "fixing up dst: $dst\n";
|
||||||
|
$dst .= '.';
|
||||||
|
$sub = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($sub) {
|
||||||
|
$src =~ s/"/\\"/g;
|
||||||
|
$dst =~ s/"/\\"/g;
|
||||||
|
print qq($indent"$src"$colon"$dst"$comma\n);
|
||||||
|
last;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$sub) {
|
||||||
|
print $_;
|
||||||
|
}
|
268
scripts/gen-i18n.js
Executable file
268
scripts/gen-i18n.js
Executable file
|
@ -0,0 +1,268 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Regenerates the translations en_EN file by walking the source tree and
|
||||||
|
* parsing each file with flow-parser. Emits a JSON file with the
|
||||||
|
* translatable strings mapped to themselves in the order they appeared
|
||||||
|
* in the files and grouped by the file they appeared in.
|
||||||
|
*
|
||||||
|
* Usage: node scripts/gen-i18n.js
|
||||||
|
*/
|
||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const walk = require('walk');
|
||||||
|
|
||||||
|
const flowParser = require('flow-parser');
|
||||||
|
const estreeWalker = require('estree-walker');
|
||||||
|
|
||||||
|
const TRANSLATIONS_FUNCS = ['_t', '_td'];
|
||||||
|
|
||||||
|
const INPUT_TRANSLATIONS_FILE = 'src/i18n/strings/en_EN.json';
|
||||||
|
const OUTPUT_FILE = 'src/i18n/strings/en_EN.json';
|
||||||
|
|
||||||
|
// NB. The sync version of walk is broken for single files so we walk
|
||||||
|
// all of res rather than just res/home.html.
|
||||||
|
// https://git.daplie.com/Daplie/node-walk/merge_requests/1 fixes it,
|
||||||
|
// or if we get bored waiting for it to be merged, we could switch
|
||||||
|
// to a project that's actively maintained.
|
||||||
|
const SEARCH_PATHS = ['src', 'res'];
|
||||||
|
|
||||||
|
const FLOW_PARSER_OPTS = {
|
||||||
|
esproposal_class_instance_fields: true,
|
||||||
|
esproposal_class_static_fields: true,
|
||||||
|
esproposal_decorators: true,
|
||||||
|
esproposal_export_star_as: true,
|
||||||
|
types: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
function getObjectValue(obj, key) {
|
||||||
|
for (const prop of obj.properties) {
|
||||||
|
if (prop.key.type == 'Identifier' && prop.key.name == key) {
|
||||||
|
return prop.value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTKey(arg) {
|
||||||
|
if (arg.type == 'Literal') {
|
||||||
|
return arg.value;
|
||||||
|
} else if (arg.type == 'BinaryExpression' && arg.operator == '+') {
|
||||||
|
return getTKey(arg.left) + getTKey(arg.right);
|
||||||
|
} else if (arg.type == 'TemplateLiteral') {
|
||||||
|
return arg.quasis.map((q) => {
|
||||||
|
return q.value.raw;
|
||||||
|
}).join('');
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getFormatStrings(str) {
|
||||||
|
// Match anything that starts with %
|
||||||
|
// We could make a regex that matched the full placeholder, but this
|
||||||
|
// would just not match invalid placeholders and so wouldn't help us
|
||||||
|
// detect the invalid ones.
|
||||||
|
// Also note that for simplicity, this just matches a % character and then
|
||||||
|
// anything up to the next % character (or a single %, or end of string).
|
||||||
|
const formatStringRe = /%([^%]+|%|$)/g;
|
||||||
|
const formatStrings = new Set();
|
||||||
|
|
||||||
|
let match;
|
||||||
|
while ( (match = formatStringRe.exec(str)) !== null ) {
|
||||||
|
const placeholder = match[1]; // Minus the leading '%'
|
||||||
|
if (placeholder === '%') continue; // Literal % is %%
|
||||||
|
|
||||||
|
const placeholderMatch = placeholder.match(/^\((.*?)\)(.)/);
|
||||||
|
if (placeholderMatch === null) {
|
||||||
|
throw new Error("Invalid format specifier: '"+match[0]+"'");
|
||||||
|
}
|
||||||
|
if (placeholderMatch.length < 3) {
|
||||||
|
throw new Error("Malformed format specifier");
|
||||||
|
}
|
||||||
|
const placeholderName = placeholderMatch[1];
|
||||||
|
const placeholderFormat = placeholderMatch[2];
|
||||||
|
|
||||||
|
if (placeholderFormat !== 's') {
|
||||||
|
throw new Error(`'${placeholderFormat}' used as format character: you probably meant 's'`);
|
||||||
|
}
|
||||||
|
|
||||||
|
formatStrings.add(placeholderName);
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatStrings;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTranslationsJs(file) {
|
||||||
|
const tree = flowParser.parse(fs.readFileSync(file, { encoding: 'utf8' }), FLOW_PARSER_OPTS);
|
||||||
|
|
||||||
|
const trs = new Set();
|
||||||
|
|
||||||
|
estreeWalker.walk(tree, {
|
||||||
|
enter: function(node, parent) {
|
||||||
|
if (
|
||||||
|
node.type == 'CallExpression' &&
|
||||||
|
TRANSLATIONS_FUNCS.includes(node.callee.name)
|
||||||
|
) {
|
||||||
|
const tKey = getTKey(node.arguments[0]);
|
||||||
|
// This happens whenever we call _t with non-literals (ie. whenever we've
|
||||||
|
// had to use a _td to compensate) so is expected.
|
||||||
|
if (tKey === null) return;
|
||||||
|
|
||||||
|
// check the format string against the args
|
||||||
|
// We only check _t: _td has no args
|
||||||
|
if (node.callee.name === '_t') {
|
||||||
|
try {
|
||||||
|
const placeholders = getFormatStrings(tKey);
|
||||||
|
for (const placeholder of placeholders) {
|
||||||
|
if (node.arguments.length < 2) {
|
||||||
|
throw new Error(`Placeholder found ('${placeholder}') but no substitutions given`);
|
||||||
|
}
|
||||||
|
const value = getObjectValue(node.arguments[1], placeholder);
|
||||||
|
if (value === null) {
|
||||||
|
throw new Error(`No value found for placeholder '${placeholder}'`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate tag replacements
|
||||||
|
if (node.arguments.length > 2) {
|
||||||
|
const tagMap = node.arguments[2];
|
||||||
|
for (const prop of tagMap.properties) {
|
||||||
|
if (prop.key.type === 'Literal') {
|
||||||
|
const tag = prop.key.value;
|
||||||
|
// RegExp same as in src/languageHandler.js
|
||||||
|
const regexp = new RegExp(`(<${tag}>(.*?)<\\/${tag}>|<${tag}>|<${tag}\\s*\\/>)`);
|
||||||
|
if (!tKey.match(regexp)) {
|
||||||
|
throw new Error(`No match for ${regexp} in ${tKey}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
console.log();
|
||||||
|
console.error(`ERROR: ${file}:${node.loc.start.line} ${tKey}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let isPlural = false;
|
||||||
|
if (node.arguments.length > 1 && node.arguments[1].type == 'ObjectExpression') {
|
||||||
|
const countVal = getObjectValue(node.arguments[1], 'count');
|
||||||
|
if (countVal) {
|
||||||
|
isPlural = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isPlural) {
|
||||||
|
trs.add(tKey + "|other");
|
||||||
|
const plurals = enPlurals[tKey];
|
||||||
|
if (plurals) {
|
||||||
|
for (const pluralType of Object.keys(plurals)) {
|
||||||
|
trs.add(tKey + "|" + pluralType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trs.add(tKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return trs;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTranslationsOther(file) {
|
||||||
|
const contents = fs.readFileSync(file, { encoding: 'utf8' });
|
||||||
|
|
||||||
|
const trs = new Set();
|
||||||
|
|
||||||
|
// Taken from riot-web src/components/structures/HomePage.js
|
||||||
|
const translationsRegex = /_t\(['"]([\s\S]*?)['"]\)/mg;
|
||||||
|
let matches;
|
||||||
|
while (matches = translationsRegex.exec(contents)) {
|
||||||
|
trs.add(matches[1]);
|
||||||
|
}
|
||||||
|
return trs;
|
||||||
|
}
|
||||||
|
|
||||||
|
// gather en_EN plural strings from the input translations file:
|
||||||
|
// the en_EN strings are all in the source with the exception of
|
||||||
|
// pluralised strings, which we need to pull in from elsewhere.
|
||||||
|
const inputTranslationsRaw = JSON.parse(fs.readFileSync(INPUT_TRANSLATIONS_FILE, { encoding: 'utf8' }));
|
||||||
|
const enPlurals = {};
|
||||||
|
|
||||||
|
for (const key of Object.keys(inputTranslationsRaw)) {
|
||||||
|
const parts = key.split("|");
|
||||||
|
if (parts.length > 1) {
|
||||||
|
const plurals = enPlurals[parts[0]] || {};
|
||||||
|
plurals[parts[1]] = inputTranslationsRaw[key];
|
||||||
|
enPlurals[parts[0]] = plurals;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const translatables = new Set();
|
||||||
|
|
||||||
|
const walkOpts = {
|
||||||
|
listeners: {
|
||||||
|
file: function(root, fileStats, next) {
|
||||||
|
const fullPath = path.join(root, fileStats.name);
|
||||||
|
|
||||||
|
let ltrs;
|
||||||
|
if (fileStats.name.endsWith('.js')) {
|
||||||
|
trs = getTranslationsJs(fullPath);
|
||||||
|
} else if (fileStats.name.endsWith('.html')) {
|
||||||
|
trs = getTranslationsOther(fullPath);
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.log(`${fullPath} (${trs.size} strings)`);
|
||||||
|
for (const tr of trs.values()) {
|
||||||
|
translatables.add(tr);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
for (const path of SEARCH_PATHS) {
|
||||||
|
if (fs.existsSync(path)) {
|
||||||
|
walk.walkSync(path, walkOpts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const trObj = {};
|
||||||
|
for (const tr of translatables) {
|
||||||
|
if (tr.includes("|")) {
|
||||||
|
if (inputTranslationsRaw[tr]) {
|
||||||
|
trObj[tr] = inputTranslationsRaw[tr];
|
||||||
|
} else {
|
||||||
|
trObj[tr] = tr.split("|")[0];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trObj[tr] = tr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.writeFileSync(
|
||||||
|
OUTPUT_FILE,
|
||||||
|
JSON.stringify(trObj, translatables.values(), 4) + "\n"
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log();
|
||||||
|
console.log(`Wrote ${translatables.size} strings to ${OUTPUT_FILE}`);
|
21
scripts/generate-eslint-error-ignore-file
Executable file
21
scripts/generate-eslint-error-ignore-file
Executable file
|
@ -0,0 +1,21 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# generates .eslintignore.errorfiles to list the files which have errors in,
|
||||||
|
# so that they can be ignored in future automated linting.
|
||||||
|
|
||||||
|
out=.eslintignore.errorfiles
|
||||||
|
|
||||||
|
cd `dirname $0`/..
|
||||||
|
|
||||||
|
echo "generating $out"
|
||||||
|
|
||||||
|
{
|
||||||
|
cat <<EOF
|
||||||
|
# autogenerated file: run scripts/generate-eslint-error-ignore-file to update.
|
||||||
|
|
||||||
|
EOF
|
||||||
|
|
||||||
|
./node_modules/.bin/eslint --no-ignore -f json src test |
|
||||||
|
jq -r '.[] | select((.errorCount + .warningCount) > 0) | .filePath' |
|
||||||
|
sed -e 's/.*matrix-react-sdk\///';
|
||||||
|
} > "$out"
|
68
scripts/prune-i18n.js
Executable file
68
scripts/prune-i18n.js
Executable file
|
@ -0,0 +1,68 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
/*
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Looks through all the translation files and removes any strings
|
||||||
|
* which don't appear in en_EN.json.
|
||||||
|
* Use this if you remove a translation, but merge any outstanding changes
|
||||||
|
* from weblate first or you'll need to resolve the conflict in weblate.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const I18NDIR = 'src/i18n/strings';
|
||||||
|
|
||||||
|
const enStringsRaw = JSON.parse(fs.readFileSync(path.join(I18NDIR, 'en_EN.json')));
|
||||||
|
|
||||||
|
const enStrings = new Set();
|
||||||
|
for (const str of Object.keys(enStringsRaw)) {
|
||||||
|
const parts = str.split('|');
|
||||||
|
if (parts.length > 1) {
|
||||||
|
enStrings.add(parts[0]);
|
||||||
|
} else {
|
||||||
|
enStrings.add(str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const filename of fs.readdirSync(I18NDIR)) {
|
||||||
|
if (filename === 'en_EN.json') continue;
|
||||||
|
if (filename === 'basefile.json') continue;
|
||||||
|
if (!filename.endsWith('.json')) continue;
|
||||||
|
|
||||||
|
const trs = JSON.parse(fs.readFileSync(path.join(I18NDIR, filename)));
|
||||||
|
const oldLen = Object.keys(trs).length;
|
||||||
|
for (const tr of Object.keys(trs)) {
|
||||||
|
const parts = tr.split('|');
|
||||||
|
const trKey = parts.length > 1 ? parts[0] : tr;
|
||||||
|
if (!enStrings.has(trKey)) {
|
||||||
|
delete trs[tr];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const removed = oldLen - Object.keys(trs).length;
|
||||||
|
if (removed > 0) {
|
||||||
|
console.log(`${filename}: removed ${removed} translations`);
|
||||||
|
// XXX: This is totally relying on the impl serialising the JSON object in the
|
||||||
|
// same order as they were parsed from the file. JSON.stringify() has a specific argument
|
||||||
|
// that can be used to control the order, but JSON.parse() lacks any kind of equivalent.
|
||||||
|
// Empirically this does maintain the order on my system, so I'm going to leave it like
|
||||||
|
// this for now.
|
||||||
|
fs.writeFileSync(path.join(I18NDIR, filename), JSON.stringify(trs, undefined, 4) + "\n");
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,53 +1,99 @@
|
||||||
#!/usr/bin/env node
|
#!/usr/bin/env node
|
||||||
|
|
||||||
var fs = require('fs');
|
var fs = require('fs');
|
||||||
var path = require('path');
|
var path = require('path');
|
||||||
var glob = require('glob');
|
var glob = require('glob');
|
||||||
|
|
||||||
var args = require('optimist').argv;
|
var args = require('optimist').argv;
|
||||||
|
var chokidar = require('chokidar');
|
||||||
var header = args.h || args.header;
|
|
||||||
|
|
||||||
var componentsDir = path.join('src', 'components');
|
|
||||||
|
|
||||||
var componentIndex = path.join('src', 'component-index.js');
|
var componentIndex = path.join('src', 'component-index.js');
|
||||||
|
var componentIndexTmp = componentIndex+".tmp";
|
||||||
|
var componentsDir = path.join('src', 'components');
|
||||||
|
var componentGlob = '**/*.js';
|
||||||
|
var prevFiles = [];
|
||||||
|
|
||||||
var packageJson = JSON.parse(fs.readFileSync('./package.json'));
|
function reskindex() {
|
||||||
|
var files = glob.sync(componentGlob, {cwd: componentsDir}).sort();
|
||||||
|
if (!filesHaveChanged(files, prevFiles)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
prevFiles = files;
|
||||||
|
|
||||||
var strm = fs.createWriteStream(componentIndex);
|
var header = args.h || args.header;
|
||||||
|
var packageJson = JSON.parse(fs.readFileSync('./package.json'));
|
||||||
|
|
||||||
if (header) {
|
var strm = fs.createWriteStream(componentIndexTmp);
|
||||||
strm.write(fs.readFileSync(header));
|
|
||||||
strm.write('\n');
|
if (header) {
|
||||||
|
strm.write(fs.readFileSync(header));
|
||||||
|
strm.write('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
strm.write("/*\n");
|
||||||
|
strm.write(" * THIS FILE IS AUTO-GENERATED\n");
|
||||||
|
strm.write(" * You can edit it you like, but your changes will be overwritten,\n");
|
||||||
|
strm.write(" * so you'd just be trying to swim upstream like a salmon.\n");
|
||||||
|
strm.write(" * You are not a salmon.\n");
|
||||||
|
strm.write(" */\n\n");
|
||||||
|
|
||||||
|
if (packageJson['matrix-react-parent']) {
|
||||||
|
const parentIndex = packageJson['matrix-react-parent'] +
|
||||||
|
'/lib/component-index';
|
||||||
|
strm.write(
|
||||||
|
`let components = require('${parentIndex}').components;
|
||||||
|
if (!components) {
|
||||||
|
throw new Error("'${parentIndex}' didn't export components");
|
||||||
|
}
|
||||||
|
`);
|
||||||
|
} else {
|
||||||
|
strm.write("let components = {};\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < files.length; ++i) {
|
||||||
|
var file = files[i].replace('.js', '');
|
||||||
|
|
||||||
|
var moduleName = (file.replace(/\//g, '.'));
|
||||||
|
var importName = moduleName.replace(/\./g, "$");
|
||||||
|
|
||||||
|
strm.write("import " + importName + " from './components/" + file + "';\n");
|
||||||
|
strm.write(importName + " && (components['"+moduleName+"'] = " + importName + ");");
|
||||||
|
strm.write('\n');
|
||||||
|
strm.uncork();
|
||||||
|
}
|
||||||
|
|
||||||
|
strm.write("export {components};\n");
|
||||||
|
strm.end();
|
||||||
|
fs.rename(componentIndexTmp, componentIndex, function(err) {
|
||||||
|
if(err) {
|
||||||
|
console.error("Error moving new index into place: " + err);
|
||||||
|
} else {
|
||||||
|
console.log('Reskindex: completed');
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
strm.write("/*\n");
|
// Expects both arrays of file names to be sorted
|
||||||
strm.write(" * THIS FILE IS AUTO-GENERATED\n");
|
function filesHaveChanged(files, prevFiles) {
|
||||||
strm.write(" * You can edit it you like, but your changes will be overwritten,\n");
|
if (files.length !== prevFiles.length) {
|
||||||
strm.write(" * so you'd just be trying to swim upstream like a salmon.\n");
|
return true;
|
||||||
strm.write(" * You are not a salmon.\n");
|
}
|
||||||
strm.write(" *\n");
|
// Check for name changes
|
||||||
strm.write(" * To update it, run:\n");
|
for (var i = 0; i < files.length; i++) {
|
||||||
strm.write(" * ./reskindex.js -h header\n");
|
if (prevFiles[i] !== files[i]) {
|
||||||
strm.write(" */\n\n");
|
return true;
|
||||||
|
}
|
||||||
if (packageJson['matrix-react-parent']) {
|
}
|
||||||
strm.write("module.exports.components = require('"+packageJson['matrix-react-parent']+"/lib/component-index').components;\n\n");
|
return false;
|
||||||
} else {
|
|
||||||
strm.write("module.exports.components = {};\n");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var files = glob.sync('**/*.js', {cwd: componentsDir}).sort();
|
// -w indicates watch mode where any FS events will trigger reskindex
|
||||||
for (var i = 0; i < files.length; ++i) {
|
if (!args.w) {
|
||||||
var file = files[i].replace('.js', '');
|
reskindex();
|
||||||
|
return;
|
||||||
var moduleName = (file.replace(/\//g, '.'));
|
|
||||||
var importName = moduleName.replace(/\./g, "$");
|
|
||||||
|
|
||||||
strm.write("import " + importName + " from './components/" + file + "';\n");
|
|
||||||
strm.write(importName + " && (module.exports.components['"+moduleName+"'] = " + importName + ");");
|
|
||||||
strm.write('\n');
|
|
||||||
strm.uncork();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
strm.end();
|
var watchDebouncer = null;
|
||||||
|
chokidar.watch(path.join(componentsDir, componentGlob)).on('all', (event, path) => {
|
||||||
|
if (path === componentIndex) return;
|
||||||
|
if (watchDebouncer) clearTimeout(watchDebouncer);
|
||||||
|
watchDebouncer = setTimeout(reskindex, 1000);
|
||||||
|
});
|
||||||
|
|
9
scripts/travis.sh
Executable file
9
scripts/travis.sh
Executable file
|
@ -0,0 +1,9 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -ex
|
||||||
|
|
||||||
|
npm run test
|
||||||
|
./.travis-test-riot.sh
|
||||||
|
|
||||||
|
# run the linter, but exclude any files known to have errors or warnings.
|
||||||
|
npm run lintwithexclusions
|
77
src/ActiveRoomObserver.js
Normal file
77
src/ActiveRoomObserver.js
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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 RoomViewStore from './stores/RoomViewStore';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Consumes changes from the RoomViewStore and notifies specific things
|
||||||
|
* about when the active room changes. Unlike listening for RoomViewStore
|
||||||
|
* changes, you can subscribe to only changes relevant to a particular
|
||||||
|
* room.
|
||||||
|
*
|
||||||
|
* TODO: If we introduce an observer for something else, factor out
|
||||||
|
* the adding / removing of listeners & emitting into a common class.
|
||||||
|
*/
|
||||||
|
class ActiveRoomObserver {
|
||||||
|
constructor() {
|
||||||
|
this._listeners = {};
|
||||||
|
|
||||||
|
this._activeRoomId = RoomViewStore.getRoomId();
|
||||||
|
// TODO: We could self-destruct when the last listener goes away, or at least
|
||||||
|
// stop listening.
|
||||||
|
this._roomStoreToken = RoomViewStore.addListener(this._onRoomViewStoreUpdate.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
addListener(roomId, listener) {
|
||||||
|
if (!this._listeners[roomId]) this._listeners[roomId] = [];
|
||||||
|
this._listeners[roomId].push(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
removeListener(roomId, listener) {
|
||||||
|
if (this._listeners[roomId]) {
|
||||||
|
const i = this._listeners[roomId].indexOf(listener);
|
||||||
|
if (i > -1) {
|
||||||
|
this._listeners[roomId].splice(i, 1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.warn("Unregistering unrecognised listener (roomId=" + roomId + ")");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_emit(roomId) {
|
||||||
|
if (!this._listeners[roomId]) return;
|
||||||
|
|
||||||
|
for (const l of this._listeners[roomId]) {
|
||||||
|
l.call();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_onRoomViewStoreUpdate() {
|
||||||
|
// emit for the old room ID
|
||||||
|
if (this._activeRoomId) this._emit(this._activeRoomId);
|
||||||
|
|
||||||
|
// update our cache
|
||||||
|
this._activeRoomId = RoomViewStore.getRoomId();
|
||||||
|
|
||||||
|
// and emit for the new one
|
||||||
|
if (this._activeRoomId) this._emit(this._activeRoomId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (global.mx_ActiveRoomObserver === undefined) {
|
||||||
|
global.mx_ActiveRoomObserver = new ActiveRoomObserver();
|
||||||
|
}
|
||||||
|
export default global.mx_ActiveRoomObserver;
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2016 OpenMarket Ltd
|
Copyright 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -14,14 +15,15 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Allows a user to add a third party identifier to their Home Server and,
|
* Allows a user to add a third party identifier to their Home Server and,
|
||||||
* optionally, the identity servers.
|
* optionally, the identity servers.
|
||||||
*
|
*
|
||||||
* This involves getting an email token from the identity server to "prove" that
|
* This involves getting an email token from the identity server to "prove" that
|
||||||
* the client owns the given email address, which is then passed to the
|
* the client owns the given email address, which is then passed to the
|
||||||
* add threepid API on the homeserver.
|
* add threepid API on the homeserver.
|
||||||
*/
|
*/
|
||||||
class AddThreepid {
|
class AddThreepid {
|
||||||
|
@ -42,8 +44,33 @@ class AddThreepid {
|
||||||
this.sessionId = res.sid;
|
this.sessionId = res.sid;
|
||||||
return res;
|
return res;
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
if (err.errcode == 'M_THREEPID_IN_USE') {
|
if (err.errcode === 'M_THREEPID_IN_USE') {
|
||||||
err.message = "This email address is already in use";
|
err.message = _t('This email address is already in use');
|
||||||
|
} else if (err.httpStatus) {
|
||||||
|
err.message = err.message + ` (Status ${err.httpStatus})`;
|
||||||
|
}
|
||||||
|
throw err;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to add a msisdn threepid. This will trigger a side-effect of
|
||||||
|
* sending a test message to the provided phone number.
|
||||||
|
* @param {string} phoneCountry The ISO 2 letter code of the country to resolve phoneNumber in
|
||||||
|
* @param {string} phoneNumber The national or international formatted phone number to add
|
||||||
|
* @param {boolean} bind If True, bind this phone number to this mxid on the Identity Server
|
||||||
|
* @return {Promise} Resolves when the text message has been sent. Then call haveMsisdnToken().
|
||||||
|
*/
|
||||||
|
addMsisdn(phoneCountry, phoneNumber, bind) {
|
||||||
|
this.bind = bind;
|
||||||
|
return MatrixClientPeg.get().requestAdd3pidMsisdnToken(
|
||||||
|
phoneCountry, phoneNumber, this.clientSecret, 1,
|
||||||
|
).then((res) => {
|
||||||
|
this.sessionId = res.sid;
|
||||||
|
return res;
|
||||||
|
}, function(err) {
|
||||||
|
if (err.errcode === 'M_THREEPID_IN_USE') {
|
||||||
|
err.message = _t('This phone number is already in use');
|
||||||
} else if (err.httpStatus) {
|
} else if (err.httpStatus) {
|
||||||
err.message = err.message + ` (Status ${err.httpStatus})`;
|
err.message = err.message + ` (Status ${err.httpStatus})`;
|
||||||
}
|
}
|
||||||
|
@ -53,26 +80,49 @@ class AddThreepid {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the email link has been clicked by attempting to add the threepid
|
* Checks if the email link has been clicked by attempting to add the threepid
|
||||||
* @return {Promise} Resolves if the password was reset. Rejects with an object
|
* @return {Promise} Resolves if the email address was added. Rejects with an object
|
||||||
* with a "message" property which contains a human-readable message detailing why
|
* with a "message" property which contains a human-readable message detailing why
|
||||||
* the reset failed, e.g. "There is no mapped matrix user ID for the given email address".
|
* the request failed.
|
||||||
*/
|
*/
|
||||||
checkEmailLinkClicked() {
|
checkEmailLinkClicked() {
|
||||||
var identityServerDomain = MatrixClientPeg.get().idBaseUrl.split("://")[1];
|
const identityServerDomain = MatrixClientPeg.get().idBaseUrl.split("://")[1];
|
||||||
return MatrixClientPeg.get().addThreePid({
|
return MatrixClientPeg.get().addThreePid({
|
||||||
sid: this.sessionId,
|
sid: this.sessionId,
|
||||||
client_secret: this.clientSecret,
|
client_secret: this.clientSecret,
|
||||||
id_server: identityServerDomain
|
id_server: identityServerDomain,
|
||||||
}, this.bind).catch(function(err) {
|
}, this.bind).catch(function(err) {
|
||||||
if (err.httpStatus === 401) {
|
if (err.httpStatus === 401) {
|
||||||
err.message = "Failed to verify email address: make sure you clicked the link in the email";
|
err.message = _t('Failed to verify email address: make sure you clicked the link in the email');
|
||||||
}
|
} else if (err.httpStatus) {
|
||||||
else if (err.httpStatus) {
|
|
||||||
err.message += ` (Status ${err.httpStatus})`;
|
err.message += ` (Status ${err.httpStatus})`;
|
||||||
}
|
}
|
||||||
throw err;
|
throw err;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes a phone number verification code as entered by the user and validates
|
||||||
|
* it with the ID server, then if successful, adds the phone number.
|
||||||
|
* @param {string} token phone number verification code as entered by the user
|
||||||
|
* @return {Promise} Resolves if the phone number was added. Rejects with an object
|
||||||
|
* with a "message" property which contains a human-readable message detailing why
|
||||||
|
* the request failed.
|
||||||
|
*/
|
||||||
|
haveMsisdnToken(token) {
|
||||||
|
return MatrixClientPeg.get().submitMsisdnToken(
|
||||||
|
this.sessionId, this.clientSecret, token,
|
||||||
|
).then((result) => {
|
||||||
|
if (result.errcode) {
|
||||||
|
throw result;
|
||||||
|
}
|
||||||
|
const identityServerDomain = MatrixClientPeg.get().idBaseUrl.split("://")[1];
|
||||||
|
return MatrixClientPeg.get().addThreePid({
|
||||||
|
sid: this.sessionId,
|
||||||
|
client_secret: this.clientSecret,
|
||||||
|
id_server: identityServerDomain,
|
||||||
|
}, this.bind);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = AddThreepid;
|
module.exports = AddThreepid;
|
||||||
|
|
231
src/Analytics.js
Normal file
231
src/Analytics.js
Normal file
|
@ -0,0 +1,231 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Michael Telatynski <7t3chguy@gmail.com>
|
||||||
|
|
||||||
|
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 { getCurrentLanguage, _t, _td } from './languageHandler';
|
||||||
|
import PlatformPeg from './PlatformPeg';
|
||||||
|
import SdkConfig, { DEFAULTS } from './SdkConfig';
|
||||||
|
import Modal from './Modal';
|
||||||
|
import sdk from './index';
|
||||||
|
|
||||||
|
function getRedactedHash() {
|
||||||
|
return window.location.hash.replace(/#\/(group|room|user)\/(.+)/, "#/$1/<redacted>");
|
||||||
|
}
|
||||||
|
|
||||||
|
function getRedactedUrl() {
|
||||||
|
// hardcoded url to make piwik happy
|
||||||
|
return 'https://riot.im/app/' + getRedactedHash();
|
||||||
|
}
|
||||||
|
|
||||||
|
const customVariables = {
|
||||||
|
'App Platform': {
|
||||||
|
id: 1,
|
||||||
|
expl: _td('The platform you\'re on'),
|
||||||
|
},
|
||||||
|
'App Version': {
|
||||||
|
id: 2,
|
||||||
|
expl: _td('The version of Riot.im'),
|
||||||
|
},
|
||||||
|
'User Type': {
|
||||||
|
id: 3,
|
||||||
|
expl: _td('Whether or not you\'re logged in (we don\'t record your user name)'),
|
||||||
|
},
|
||||||
|
'Chosen Language': {
|
||||||
|
id: 4,
|
||||||
|
expl: _td('Your language of choice'),
|
||||||
|
},
|
||||||
|
'Instance': {
|
||||||
|
id: 5,
|
||||||
|
expl: _td('Which officially provided instance you are using, if any'),
|
||||||
|
},
|
||||||
|
'RTE: Uses Richtext Mode': {
|
||||||
|
id: 6,
|
||||||
|
expl: _td('Whether or not you\'re using the Richtext mode of the Rich Text Editor'),
|
||||||
|
},
|
||||||
|
'Homeserver URL': {
|
||||||
|
id: 7,
|
||||||
|
expl: _td('Your homeserver\'s URL'),
|
||||||
|
},
|
||||||
|
'Identity Server URL': {
|
||||||
|
id: 8,
|
||||||
|
expl: _td('Your identity server\'s URL'),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
function whitelistRedact(whitelist, str) {
|
||||||
|
if (whitelist.includes(str)) return str;
|
||||||
|
return '<redacted>';
|
||||||
|
}
|
||||||
|
|
||||||
|
class Analytics {
|
||||||
|
constructor() {
|
||||||
|
this._paq = null;
|
||||||
|
this.disabled = true;
|
||||||
|
this.firstPage = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enable Analytics if initialized but disabled
|
||||||
|
* otherwise try and initalize, no-op if piwik config missing
|
||||||
|
*/
|
||||||
|
enable() {
|
||||||
|
if (this._paq || this._init()) {
|
||||||
|
this.disabled = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Disable Analytics calls, will not fully unload Piwik until a refresh,
|
||||||
|
* but this is second best, Piwik should not pull anything implicitly.
|
||||||
|
*/
|
||||||
|
disable() {
|
||||||
|
this.trackEvent('Analytics', 'opt-out');
|
||||||
|
this.disabled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
_init() {
|
||||||
|
const config = SdkConfig.get();
|
||||||
|
if (!config || !config.piwik || !config.piwik.url || !config.piwik.siteId) return;
|
||||||
|
|
||||||
|
const url = config.piwik.url;
|
||||||
|
const siteId = config.piwik.siteId;
|
||||||
|
const self = this;
|
||||||
|
|
||||||
|
window._paq = this._paq = window._paq || [];
|
||||||
|
|
||||||
|
this._paq.push(['setTrackerUrl', url+'piwik.php']);
|
||||||
|
this._paq.push(['setSiteId', siteId]);
|
||||||
|
|
||||||
|
this._paq.push(['trackAllContentImpressions']);
|
||||||
|
this._paq.push(['discardHashTag', false]);
|
||||||
|
this._paq.push(['enableHeartBeatTimer']);
|
||||||
|
// this._paq.push(['enableLinkTracking', true]);
|
||||||
|
|
||||||
|
const platform = PlatformPeg.get();
|
||||||
|
this._setVisitVariable('App Platform', platform.getHumanReadableName());
|
||||||
|
platform.getAppVersion().then((version) => {
|
||||||
|
this._setVisitVariable('App Version', version);
|
||||||
|
}).catch(() => {
|
||||||
|
this._setVisitVariable('App Version', 'unknown');
|
||||||
|
});
|
||||||
|
|
||||||
|
this._setVisitVariable('Chosen Language', getCurrentLanguage());
|
||||||
|
|
||||||
|
if (window.location.hostname === 'riot.im') {
|
||||||
|
this._setVisitVariable('Instance', window.location.pathname);
|
||||||
|
}
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
const g = document.createElement('script');
|
||||||
|
const s = document.getElementsByTagName('script')[0];
|
||||||
|
g.type='text/javascript'; g.async=true; g.defer=true; g.src=url+'piwik.js';
|
||||||
|
|
||||||
|
g.onload = function() {
|
||||||
|
console.log('Initialised anonymous analytics');
|
||||||
|
self._paq = window._paq;
|
||||||
|
};
|
||||||
|
|
||||||
|
s.parentNode.insertBefore(g, s);
|
||||||
|
})();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
trackPageChange() {
|
||||||
|
if (this.disabled) return;
|
||||||
|
if (this.firstPage) {
|
||||||
|
// De-duplicate first page
|
||||||
|
// router seems to hit the fn twice
|
||||||
|
this.firstPage = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._paq.push(['setCustomUrl', getRedactedUrl()]);
|
||||||
|
this._paq.push(['trackPageView']);
|
||||||
|
}
|
||||||
|
|
||||||
|
trackEvent(category, action, name) {
|
||||||
|
if (this.disabled) return;
|
||||||
|
this._paq.push(['trackEvent', category, action, name]);
|
||||||
|
}
|
||||||
|
|
||||||
|
logout() {
|
||||||
|
if (this.disabled) return;
|
||||||
|
this._paq.push(['deleteCookies']);
|
||||||
|
}
|
||||||
|
|
||||||
|
_setVisitVariable(key, value) {
|
||||||
|
this._paq.push(['setCustomVariable', customVariables[key].id, key, value, 'visit']);
|
||||||
|
}
|
||||||
|
|
||||||
|
setLoggedIn(isGuest, homeserverUrl, identityServerUrl) {
|
||||||
|
if (this.disabled) return;
|
||||||
|
|
||||||
|
const config = SdkConfig.get();
|
||||||
|
const whitelistedHSUrls = config.piwik.whitelistedHSUrls || DEFAULTS.piwik.whitelistedHSUrls;
|
||||||
|
const whitelistedISUrls = config.piwik.whitelistedISUrls || DEFAULTS.piwik.whitelistedISUrls;
|
||||||
|
|
||||||
|
this._setVisitVariable('User Type', isGuest ? 'Guest' : 'Logged In');
|
||||||
|
this._setVisitVariable('Homeserver URL', whitelistRedact(whitelistedHSUrls, homeserverUrl));
|
||||||
|
this._setVisitVariable('Identity Server URL', whitelistRedact(whitelistedISUrls, identityServerUrl));
|
||||||
|
}
|
||||||
|
|
||||||
|
setRichtextMode(state) {
|
||||||
|
if (this.disabled) return;
|
||||||
|
this._setVisitVariable('RTE: Uses Richtext Mode', state ? 'on' : 'off');
|
||||||
|
}
|
||||||
|
|
||||||
|
showDetailsModal() {
|
||||||
|
const Tracker = window.Piwik.getAsyncTracker();
|
||||||
|
const rows = Object.values(customVariables).map((v) => Tracker.getCustomVariable(v.id)).filter(Boolean);
|
||||||
|
|
||||||
|
const resolution = `${window.screen.width}x${window.screen.height}`;
|
||||||
|
|
||||||
|
const ErrorDialog = sdk.getComponent('dialogs.ErrorDialog');
|
||||||
|
Modal.createTrackedDialog('Analytics Details', '', ErrorDialog, {
|
||||||
|
title: _t('Analytics'),
|
||||||
|
description: <div>
|
||||||
|
<div>
|
||||||
|
{ _t('The information being sent to us to help make Riot.im better includes:') }
|
||||||
|
</div>
|
||||||
|
<table>
|
||||||
|
{ rows.map((row) => <tr key={row[0]}>
|
||||||
|
<td>{ _t(customVariables[row[0]].expl) }</td>
|
||||||
|
<td><code>{ row[1] }</code></td>
|
||||||
|
</tr>) }
|
||||||
|
</table>
|
||||||
|
<br />
|
||||||
|
<div>
|
||||||
|
{ _t('We also record each page you use in the app (currently <CurrentPageHash>), your User Agent'
|
||||||
|
+ ' (<CurrentUserAgent>) and your device resolution (<CurrentDeviceResolution>).',
|
||||||
|
{},
|
||||||
|
{
|
||||||
|
CurrentPageHash: <code>{ getRedactedHash() }</code>,
|
||||||
|
CurrentUserAgent: <code>{ navigator.userAgent }</code>,
|
||||||
|
CurrentDeviceResolution: <code>{ resolution }</code>,
|
||||||
|
},
|
||||||
|
) }
|
||||||
|
|
||||||
|
{ _t('Where this page includes identifiable information, such as a room, '
|
||||||
|
+ 'user or group ID, that data is removed before being sent to the server.') }
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!global.mxAnalytics) {
|
||||||
|
global.mxAnalytics = new Analytics();
|
||||||
|
}
|
||||||
|
module.exports = global.mxAnalytics;
|
|
@ -15,18 +15,18 @@ limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
var ContentRepo = require("matrix-js-sdk").ContentRepo;
|
import {ContentRepo} from 'matrix-js-sdk';
|
||||||
var MatrixClientPeg = require('./MatrixClientPeg');
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
avatarUrlForMember: function(member, width, height, resizeMethod) {
|
avatarUrlForMember: function(member, width, height, resizeMethod) {
|
||||||
var url = member.getAvatarUrl(
|
let url = member.getAvatarUrl(
|
||||||
MatrixClientPeg.get().getHomeserverUrl(),
|
MatrixClientPeg.get().getHomeserverUrl(),
|
||||||
width,
|
Math.floor(width * window.devicePixelRatio),
|
||||||
height,
|
Math.floor(height * window.devicePixelRatio),
|
||||||
resizeMethod,
|
resizeMethod,
|
||||||
false,
|
false,
|
||||||
false
|
false,
|
||||||
);
|
);
|
||||||
if (!url) {
|
if (!url) {
|
||||||
// member can be null here currently since on invites, the JS SDK
|
// member can be null here currently since on invites, the JS SDK
|
||||||
|
@ -38,9 +38,11 @@ module.exports = {
|
||||||
},
|
},
|
||||||
|
|
||||||
avatarUrlForUser: function(user, width, height, resizeMethod) {
|
avatarUrlForUser: function(user, width, height, resizeMethod) {
|
||||||
var url = ContentRepo.getHttpUriForMxc(
|
const url = ContentRepo.getHttpUriForMxc(
|
||||||
MatrixClientPeg.get().getHomeserverUrl(), user.avatarUrl,
|
MatrixClientPeg.get().getHomeserverUrl(), user.avatarUrl,
|
||||||
width, height, resizeMethod
|
Math.floor(width * window.devicePixelRatio),
|
||||||
|
Math.floor(height * window.devicePixelRatio),
|
||||||
|
resizeMethod,
|
||||||
);
|
);
|
||||||
if (!url || url.length === 0) {
|
if (!url || url.length === 0) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -49,12 +51,11 @@ module.exports = {
|
||||||
},
|
},
|
||||||
|
|
||||||
defaultAvatarUrlForString: function(s) {
|
defaultAvatarUrlForString: function(s) {
|
||||||
var images = [ '76cfa6', '50e2c2', 'f4c371' ];
|
const images = ['76cfa6', '50e2c2', 'f4c371'];
|
||||||
var total = 0;
|
let total = 0;
|
||||||
for (var i = 0; i < s.length; ++i) {
|
for (let i = 0; i < s.length; ++i) {
|
||||||
total += s.charCodeAt(i);
|
total += s.charCodeAt(i);
|
||||||
}
|
}
|
||||||
return 'img/' + images[total % images.length] + '.png';
|
return 'img/' + images[total % images.length] + '.png';
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,8 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import dis from './dispatcher';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Base class for classes that provide platform-specific functionality
|
* Base class for classes that provide platform-specific functionality
|
||||||
* eg. Setting an application badge or displaying notifications
|
* eg. Setting an application badge or displaying notifications
|
||||||
|
@ -27,6 +29,21 @@ export default class BasePlatform {
|
||||||
constructor() {
|
constructor() {
|
||||||
this.notificationCount = 0;
|
this.notificationCount = 0;
|
||||||
this.errorDidOccur = false;
|
this.errorDidOccur = false;
|
||||||
|
|
||||||
|
dis.register(this._onAction.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
_onAction(payload: Object) {
|
||||||
|
switch (payload.action) {
|
||||||
|
case 'on_logged_out':
|
||||||
|
this.setNotificationCount(0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Used primarily for Analytics
|
||||||
|
getHumanReadableName(): string {
|
||||||
|
return 'Base Platform';
|
||||||
}
|
}
|
||||||
|
|
||||||
setNotificationCount(count: number) {
|
setNotificationCount(count: number) {
|
||||||
|
@ -40,16 +57,18 @@ export default class BasePlatform {
|
||||||
/**
|
/**
|
||||||
* Returns true if the platform supports displaying
|
* Returns true if the platform supports displaying
|
||||||
* notifications, otherwise false.
|
* notifications, otherwise false.
|
||||||
|
* @returns {boolean} whether the platform supports displaying notifications
|
||||||
*/
|
*/
|
||||||
supportsNotifications() : boolean {
|
supportsNotifications(): boolean {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if the application currently has permission
|
* Returns true if the application currently has permission
|
||||||
* to display notifications. Otherwise false.
|
* to display notifications. Otherwise false.
|
||||||
|
* @returns {boolean} whether the application has permission to display notifications
|
||||||
*/
|
*/
|
||||||
maySendNotifications() : boolean {
|
maySendNotifications(): boolean {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,17 +79,42 @@ export default class BasePlatform {
|
||||||
* that is 'granted' if the user allowed the request or
|
* that is 'granted' if the user allowed the request or
|
||||||
* 'denied' otherwise.
|
* 'denied' otherwise.
|
||||||
*/
|
*/
|
||||||
requestNotificationPermission() : Promise<string> {
|
requestNotificationPermission(): Promise<string> {
|
||||||
}
|
}
|
||||||
|
|
||||||
displayNotification(title: string, msg: string, avatarUrl: string, room: Object) {
|
displayNotification(title: string, msg: string, avatarUrl: string, room: Object) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
loudNotification(ev: Event, room: Object) {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a promise that resolves to a string representing
|
* Returns a promise that resolves to a string representing
|
||||||
* the current version of the application.
|
* the current version of the application.
|
||||||
*/
|
*/
|
||||||
getAppVersion() {
|
getAppVersion(): Promise<string> {
|
||||||
throw new Error("getAppVersion not implemented!");
|
throw new Error("getAppVersion not implemented!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If it's not expected that capturing the screen will work
|
||||||
|
* with getUserMedia, return a string explaining why not.
|
||||||
|
* Otherwise, return null.
|
||||||
|
*/
|
||||||
|
screenCaptureErrorString(): string {
|
||||||
|
return "Not implemented";
|
||||||
|
}
|
||||||
|
|
||||||
|
isElectron(): boolean { return false; }
|
||||||
|
|
||||||
|
setupScreenSharingForIframe() {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restarts the application, without neccessarily reloading
|
||||||
|
* any application code
|
||||||
|
*/
|
||||||
|
reload() {
|
||||||
|
throw new Error("reload not implemented!");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 New Vector Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -51,32 +52,34 @@ limitations under the License.
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var MatrixClientPeg = require('./MatrixClientPeg');
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
var Modal = require('./Modal');
|
import PlatformPeg from './PlatformPeg';
|
||||||
var sdk = require('./index');
|
import Modal from './Modal';
|
||||||
var Matrix = require("matrix-js-sdk");
|
import sdk from './index';
|
||||||
var dis = require("./dispatcher");
|
import { _t } from './languageHandler';
|
||||||
|
import Matrix from 'matrix-js-sdk';
|
||||||
|
import dis from './dispatcher';
|
||||||
|
import { showUnknownDeviceDialogForCalls } from './cryptodevices';
|
||||||
|
|
||||||
global.mxCalls = {
|
global.mxCalls = {
|
||||||
//room_id: MatrixCall
|
//room_id: MatrixCall
|
||||||
};
|
};
|
||||||
var calls = global.mxCalls;
|
const calls = global.mxCalls;
|
||||||
var ConferenceHandler = null;
|
let ConferenceHandler = null;
|
||||||
|
|
||||||
var audioPromises = {};
|
const audioPromises = {};
|
||||||
|
|
||||||
function play(audioId) {
|
function play(audioId) {
|
||||||
// TODO: Attach an invisible element for this instead
|
// TODO: Attach an invisible element for this instead
|
||||||
// which listens?
|
// which listens?
|
||||||
var audio = document.getElementById(audioId);
|
const audio = document.getElementById(audioId);
|
||||||
if (audio) {
|
if (audio) {
|
||||||
if (audioPromises[audioId]) {
|
if (audioPromises[audioId]) {
|
||||||
audioPromises[audioId] = audioPromises[audioId].then(()=>{
|
audioPromises[audioId] = audioPromises[audioId].then(()=>{
|
||||||
audio.load();
|
audio.load();
|
||||||
return audio.play();
|
return audio.play();
|
||||||
});
|
});
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
audioPromises[audioId] = audio.play();
|
audioPromises[audioId] = audio.play();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -85,24 +88,67 @@ function play(audioId) {
|
||||||
function pause(audioId) {
|
function pause(audioId) {
|
||||||
// TODO: Attach an invisible element for this instead
|
// TODO: Attach an invisible element for this instead
|
||||||
// which listens?
|
// which listens?
|
||||||
var audio = document.getElementById(audioId);
|
const audio = document.getElementById(audioId);
|
||||||
if (audio) {
|
if (audio) {
|
||||||
if (audioPromises[audioId]) {
|
if (audioPromises[audioId]) {
|
||||||
audioPromises[audioId] = audioPromises[audioId].then(()=>audio.pause());
|
audioPromises[audioId] = audioPromises[audioId].then(()=>audio.pause());
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
// pause doesn't actually return a promise, but might as well do this for symmetry with play();
|
// pause doesn't actually return a promise, but might as well do this for symmetry with play();
|
||||||
audioPromises[audioId] = audio.pause();
|
audioPromises[audioId] = audio.pause();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function _reAttemptCall(call) {
|
||||||
|
if (call.direction === 'outbound') {
|
||||||
|
dis.dispatch({
|
||||||
|
action: 'place_call',
|
||||||
|
room_id: call.roomId,
|
||||||
|
type: call.type,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
call.answer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function _setCallListeners(call) {
|
function _setCallListeners(call) {
|
||||||
call.on("error", function(err) {
|
call.on("error", function(err) {
|
||||||
console.error("Call error: %s", err);
|
console.error("Call error: %s", err);
|
||||||
console.error(err.stack);
|
console.error(err.stack);
|
||||||
call.hangup();
|
if (err.code === 'unknown_devices') {
|
||||||
_setCallState(undefined, call.roomId, "ended");
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
||||||
|
|
||||||
|
Modal.createTrackedDialog('Call Failed', '', QuestionDialog, {
|
||||||
|
title: _t('Call Failed'),
|
||||||
|
description: _t(
|
||||||
|
"There are unknown devices in this room: "+
|
||||||
|
"if you proceed without verifying them, it will be "+
|
||||||
|
"possible for someone to eavesdrop on your call."
|
||||||
|
),
|
||||||
|
button: _t('Review Devices'),
|
||||||
|
onFinished: function(confirmed) {
|
||||||
|
if (confirmed) {
|
||||||
|
const room = MatrixClientPeg.get().getRoom(call.roomId);
|
||||||
|
showUnknownDeviceDialogForCalls(
|
||||||
|
MatrixClientPeg.get(),
|
||||||
|
room,
|
||||||
|
() => {
|
||||||
|
_reAttemptCall(call);
|
||||||
|
},
|
||||||
|
call.direction === 'outbound' ? _t("Call Anyway") : _t("Answer Anyway"),
|
||||||
|
call.direction === 'outbound' ? _t("Call") : _t("Answer"),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
|
||||||
|
Modal.createTrackedDialog('Call Failed', '', ErrorDialog, {
|
||||||
|
title: _t('Call Failed'),
|
||||||
|
description: err.message,
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
call.on("hangup", function() {
|
call.on("hangup", function() {
|
||||||
_setCallState(undefined, call.roomId, "ended");
|
_setCallState(undefined, call.roomId, "ended");
|
||||||
|
@ -113,38 +159,32 @@ function _setCallListeners(call) {
|
||||||
if (newState === "ringing") {
|
if (newState === "ringing") {
|
||||||
_setCallState(call, call.roomId, "ringing");
|
_setCallState(call, call.roomId, "ringing");
|
||||||
pause("ringbackAudio");
|
pause("ringbackAudio");
|
||||||
}
|
} else if (newState === "invite_sent") {
|
||||||
else if (newState === "invite_sent") {
|
|
||||||
_setCallState(call, call.roomId, "ringback");
|
_setCallState(call, call.roomId, "ringback");
|
||||||
play("ringbackAudio");
|
play("ringbackAudio");
|
||||||
}
|
} else if (newState === "ended" && oldState === "connected") {
|
||||||
else if (newState === "ended" && oldState === "connected") {
|
|
||||||
_setCallState(undefined, call.roomId, "ended");
|
_setCallState(undefined, call.roomId, "ended");
|
||||||
pause("ringbackAudio");
|
pause("ringbackAudio");
|
||||||
play("callendAudio");
|
play("callendAudio");
|
||||||
}
|
} else if (newState === "ended" && oldState === "invite_sent" &&
|
||||||
else if (newState === "ended" && oldState === "invite_sent" &&
|
|
||||||
(call.hangupParty === "remote" ||
|
(call.hangupParty === "remote" ||
|
||||||
(call.hangupParty === "local" && call.hangupReason === "invite_timeout")
|
(call.hangupParty === "local" && call.hangupReason === "invite_timeout")
|
||||||
)) {
|
)) {
|
||||||
_setCallState(call, call.roomId, "busy");
|
_setCallState(call, call.roomId, "busy");
|
||||||
pause("ringbackAudio");
|
pause("ringbackAudio");
|
||||||
play("busyAudio");
|
play("busyAudio");
|
||||||
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'Call Timeout', ErrorDialog, {
|
||||||
title: "Call Timeout",
|
title: _t('Call Timeout'),
|
||||||
description: "The remote side failed to pick up."
|
description: _t('The remote side failed to pick up') + '.',
|
||||||
});
|
});
|
||||||
}
|
} else if (oldState === "invite_sent") {
|
||||||
else if (oldState === "invite_sent") {
|
|
||||||
_setCallState(call, call.roomId, "stop_ringback");
|
_setCallState(call, call.roomId, "stop_ringback");
|
||||||
pause("ringbackAudio");
|
pause("ringbackAudio");
|
||||||
}
|
} else if (oldState === "ringing") {
|
||||||
else if (oldState === "ringing") {
|
|
||||||
_setCallState(call, call.roomId, "stop_ringing");
|
_setCallState(call, call.roomId, "stop_ringing");
|
||||||
pause("ringbackAudio");
|
pause("ringbackAudio");
|
||||||
}
|
} else if (newState === "connected") {
|
||||||
else if (newState === "connected") {
|
|
||||||
_setCallState(call, call.roomId, "connected");
|
_setCallState(call, call.roomId, "connected");
|
||||||
pause("ringbackAudio");
|
pause("ringbackAudio");
|
||||||
}
|
}
|
||||||
|
@ -153,15 +193,14 @@ function _setCallListeners(call) {
|
||||||
|
|
||||||
function _setCallState(call, roomId, status) {
|
function _setCallState(call, roomId, status) {
|
||||||
console.log(
|
console.log(
|
||||||
"Call state in %s changed to %s (%s)", roomId, status, (call ? call.call_state : "-")
|
"Call state in %s changed to %s (%s)", roomId, status, (call ? call.call_state : "-"),
|
||||||
);
|
);
|
||||||
calls[roomId] = call;
|
calls[roomId] = call;
|
||||||
|
|
||||||
if (status === "ringing") {
|
if (status === "ringing") {
|
||||||
play("ringAudio")
|
play("ringAudio");
|
||||||
}
|
} else if (call && call.call_state === "ringing") {
|
||||||
else if (call && call.call_state === "ringing") {
|
pause("ringAudio");
|
||||||
pause("ringAudio")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (call) {
|
if (call) {
|
||||||
|
@ -169,30 +208,38 @@ function _setCallState(call, roomId, status) {
|
||||||
}
|
}
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'call_state',
|
action: 'call_state',
|
||||||
room_id: roomId
|
room_id: roomId,
|
||||||
|
state: status,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function _onAction(payload) {
|
function _onAction(payload) {
|
||||||
function placeCall(newCall) {
|
function placeCall(newCall) {
|
||||||
_setCallListeners(newCall);
|
_setCallListeners(newCall);
|
||||||
_setCallState(newCall, newCall.roomId, "ringback");
|
|
||||||
if (payload.type === 'voice') {
|
if (payload.type === 'voice') {
|
||||||
newCall.placeVoiceCall();
|
newCall.placeVoiceCall();
|
||||||
}
|
} else if (payload.type === 'video') {
|
||||||
else if (payload.type === 'video') {
|
|
||||||
newCall.placeVideoCall(
|
newCall.placeVideoCall(
|
||||||
payload.remote_element,
|
payload.remote_element,
|
||||||
payload.local_element
|
payload.local_element,
|
||||||
);
|
);
|
||||||
}
|
} else if (payload.type === 'screensharing') {
|
||||||
else if (payload.type === 'screensharing') {
|
const screenCapErrorString = PlatformPeg.get().screenCaptureErrorString();
|
||||||
|
if (screenCapErrorString) {
|
||||||
|
_setCallState(undefined, newCall.roomId, "ended");
|
||||||
|
console.log("Can't capture screen: " + screenCapErrorString);
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Call Handler', 'Unable to capture screen', ErrorDialog, {
|
||||||
|
title: _t('Unable to capture screen'),
|
||||||
|
description: screenCapErrorString,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
newCall.placeScreenSharingCall(
|
newCall.placeScreenSharingCall(
|
||||||
payload.remote_element,
|
payload.remote_element,
|
||||||
payload.local_element
|
payload.local_element,
|
||||||
);
|
);
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
console.error("Unknown conf call type: %s", payload.type);
|
console.error("Unknown conf call type: %s", payload.type);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -201,9 +248,9 @@ function _onAction(payload) {
|
||||||
case 'place_call':
|
case 'place_call':
|
||||||
if (module.exports.getAnyActiveCall()) {
|
if (module.exports.getAnyActiveCall()) {
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'Existing Call', ErrorDialog, {
|
||||||
title: "Existing Call",
|
title: _t('Existing Call'),
|
||||||
description: "You are already in a call."
|
description: _t('You are already in a call.'),
|
||||||
});
|
});
|
||||||
return; // don't allow >1 call to be placed.
|
return; // don't allow >1 call to be placed.
|
||||||
}
|
}
|
||||||
|
@ -211,9 +258,9 @@ function _onAction(payload) {
|
||||||
// if the runtime env doesn't do VoIP, whine.
|
// if the runtime env doesn't do VoIP, whine.
|
||||||
if (!MatrixClientPeg.get().supportsVoip()) {
|
if (!MatrixClientPeg.get().supportsVoip()) {
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'VoIP is unsupported', ErrorDialog, {
|
||||||
title: "VoIP is unsupported",
|
title: _t('VoIP is unsupported'),
|
||||||
description: "You cannot place VoIP calls in this browser."
|
description: _t('You cannot place VoIP calls in this browser.'),
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -227,25 +274,21 @@ function _onAction(payload) {
|
||||||
var members = room.getJoinedMembers();
|
var members = room.getJoinedMembers();
|
||||||
if (members.length <= 1) {
|
if (members.length <= 1) {
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'Cannot place call with self', ErrorDialog, {
|
||||||
description: "You cannot place a call with yourself."
|
description: _t('You cannot place a call with yourself.'),
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
} else if (members.length === 2) {
|
||||||
else if (members.length === 2) {
|
|
||||||
console.log("Place %s call in %s", payload.type, payload.room_id);
|
console.log("Place %s call in %s", payload.type, payload.room_id);
|
||||||
var call = Matrix.createNewMatrixCall(
|
const call = Matrix.createNewMatrixCall(MatrixClientPeg.get(), payload.room_id);
|
||||||
MatrixClientPeg.get(), payload.room_id
|
|
||||||
);
|
|
||||||
placeCall(call);
|
placeCall(call);
|
||||||
}
|
} else { // > 2
|
||||||
else { // > 2
|
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: "place_conference_call",
|
action: "place_conference_call",
|
||||||
room_id: payload.room_id,
|
room_id: payload.room_id,
|
||||||
type: payload.type,
|
type: payload.type,
|
||||||
remote_element: payload.remote_element,
|
remote_element: payload.remote_element,
|
||||||
local_element: payload.local_element
|
local_element: payload.local_element,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -253,18 +296,16 @@ function _onAction(payload) {
|
||||||
console.log("Place conference call in %s", payload.room_id);
|
console.log("Place conference call in %s", payload.room_id);
|
||||||
if (!ConferenceHandler) {
|
if (!ConferenceHandler) {
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'Conference call unsupported client', ErrorDialog, {
|
||||||
description: "Conference calls are not supported in this client"
|
description: _t('Conference calls are not supported in this client'),
|
||||||
});
|
});
|
||||||
}
|
} else if (!MatrixClientPeg.get().supportsVoip()) {
|
||||||
else if (!MatrixClientPeg.get().supportsVoip()) {
|
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'VoIP is unsupported', ErrorDialog, {
|
||||||
title: "VoIP is unsupported",
|
title: _t('VoIP is unsupported'),
|
||||||
description: "You cannot place VoIP calls in this browser."
|
description: _t('You cannot place VoIP calls in this browser.'),
|
||||||
});
|
});
|
||||||
}
|
} else if (MatrixClientPeg.get().isRoomEncrypted(payload.room_id)) {
|
||||||
else if (MatrixClientPeg.get().isRoomEncrypted(payload.room_id)) {
|
|
||||||
// Conference calls are implemented by sending the media to central
|
// Conference calls are implemented by sending the media to central
|
||||||
// server which combines the audio from all the participants together
|
// server which combines the audio from all the participants together
|
||||||
// into a single stream. This is incompatible with end-to-end encryption
|
// into a single stream. This is incompatible with end-to-end encryption
|
||||||
|
@ -272,26 +313,26 @@ function _onAction(payload) {
|
||||||
// participant.
|
// participant.
|
||||||
// Therefore we disable conference calling in E2E rooms.
|
// Therefore we disable conference calling in E2E rooms.
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Call Handler', 'Conference calls unsupported e2e', ErrorDialog, {
|
||||||
description: "Conference calls are not supported in encrypted rooms",
|
description: _t('Conference calls are not supported in encrypted rooms'),
|
||||||
});
|
});
|
||||||
}
|
} else {
|
||||||
else {
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
||||||
var QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
Modal.createTrackedDialog('Call Handler', 'Conference calling in development', QuestionDialog, {
|
||||||
Modal.createDialog(QuestionDialog, {
|
title: _t('Warning!'),
|
||||||
title: "Warning!",
|
description: _t('Conference calling is in development and may not be reliable.'),
|
||||||
description: "Conference calling in Riot is in development and may not be reliable.",
|
onFinished: (confirm)=>{
|
||||||
onFinished: confirm=>{
|
|
||||||
if (confirm) {
|
if (confirm) {
|
||||||
ConferenceHandler.createNewMatrixCall(
|
ConferenceHandler.createNewMatrixCall(
|
||||||
MatrixClientPeg.get(), payload.room_id
|
MatrixClientPeg.get(), payload.room_id,
|
||||||
).done(function(call) {
|
).done(function(call) {
|
||||||
placeCall(call);
|
placeCall(call);
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
console.error("Conference call failed: " + err);
|
||||||
title: "Failed to set up conference call",
|
Modal.createTrackedDialog('Call Handler', 'Failed to set up conference call', ErrorDialog, {
|
||||||
description: "Conference call failed: " + err,
|
title: _t('Failed to set up conference call'),
|
||||||
|
description: _t('Conference call failed.') + ' ' + ((err && err.message) ? err.message : ''),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -332,7 +373,7 @@ function _onAction(payload) {
|
||||||
_setCallState(calls[payload.room_id], payload.room_id, "connected");
|
_setCallState(calls[payload.room_id], payload.room_id, "connected");
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: "view_room",
|
action: "view_room",
|
||||||
room_id: payload.room_id
|
room_id: payload.room_id,
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -343,9 +384,9 @@ if (!global.mxCallHandler) {
|
||||||
dis.register(_onAction);
|
dis.register(_onAction);
|
||||||
}
|
}
|
||||||
|
|
||||||
var callHandler = {
|
const callHandler = {
|
||||||
getCallForRoom: function(roomId) {
|
getCallForRoom: function(roomId) {
|
||||||
var call = module.exports.getCall(roomId);
|
let call = module.exports.getCall(roomId);
|
||||||
if (call) return call;
|
if (call) return call;
|
||||||
|
|
||||||
if (ConferenceHandler) {
|
if (ConferenceHandler) {
|
||||||
|
@ -361,8 +402,8 @@ var callHandler = {
|
||||||
},
|
},
|
||||||
|
|
||||||
getAnyActiveCall: function() {
|
getAnyActiveCall: function() {
|
||||||
var roomsWithCalls = Object.keys(calls);
|
const roomsWithCalls = Object.keys(calls);
|
||||||
for (var i = 0; i < roomsWithCalls.length; i++) {
|
for (let i = 0; i < roomsWithCalls.length; i++) {
|
||||||
if (calls[roomsWithCalls[i]] &&
|
if (calls[roomsWithCalls[i]] &&
|
||||||
calls[roomsWithCalls[i]].call_state !== "ended") {
|
calls[roomsWithCalls[i]].call_state !== "ended") {
|
||||||
return calls[roomsWithCalls[i]];
|
return calls[roomsWithCalls[i]];
|
||||||
|
@ -377,7 +418,7 @@ var callHandler = {
|
||||||
|
|
||||||
getConferenceHandler: function() {
|
getConferenceHandler: function() {
|
||||||
return ConferenceHandler;
|
return ConferenceHandler;
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
// Only things in here which actually need to be global are the
|
// Only things in here which actually need to be global are the
|
||||||
// calls list (done separately) and making sure we only register
|
// calls list (done separately) and making sure we only register
|
||||||
|
|
62
src/CallMediaHandler.js
Normal file
62
src/CallMediaHandler.js
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Michael Telatynski <7t3chguy@gmail.com>
|
||||||
|
|
||||||
|
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 Matrix from 'matrix-js-sdk';
|
||||||
|
import SettingsStore, {SettingLevel} from "./settings/SettingsStore";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
getDevices: function() {
|
||||||
|
// Only needed for Electron atm, though should work in modern browsers
|
||||||
|
// once permission has been granted to the webapp
|
||||||
|
return navigator.mediaDevices.enumerateDevices().then(function(devices) {
|
||||||
|
const audioIn = [];
|
||||||
|
const videoIn = [];
|
||||||
|
|
||||||
|
if (devices.some((device) => !device.label)) return false;
|
||||||
|
|
||||||
|
devices.forEach((device) => {
|
||||||
|
switch (device.kind) {
|
||||||
|
case 'audioinput': audioIn.push(device); break;
|
||||||
|
case 'videoinput': videoIn.push(device); break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// console.log("Loaded WebRTC Devices", mediaDevices);
|
||||||
|
return {
|
||||||
|
audioinput: audioIn,
|
||||||
|
videoinput: videoIn,
|
||||||
|
};
|
||||||
|
}, (error) => { console.log('Unable to refresh WebRTC Devices: ', error); });
|
||||||
|
},
|
||||||
|
|
||||||
|
loadDevices: function() {
|
||||||
|
const audioDeviceId = SettingsStore.getValue("webrtc_audioinput");
|
||||||
|
const videoDeviceId = SettingsStore.getValue("webrtc_videoinput");
|
||||||
|
|
||||||
|
Matrix.setMatrixCallAudioInput(audioDeviceId);
|
||||||
|
Matrix.setMatrixCallVideoInput(videoDeviceId);
|
||||||
|
},
|
||||||
|
|
||||||
|
setAudioInput: function(deviceId) {
|
||||||
|
SettingsStore.setValue("webrtc_audioinput", null, SettingLevel.DEVICE, deviceId);
|
||||||
|
Matrix.setMatrixCallAudioInput(deviceId);
|
||||||
|
},
|
||||||
|
|
||||||
|
setVideoInput: function(deviceId) {
|
||||||
|
SettingsStore.setValue("webrtc_videoinput", null, SettingLevel.DEVICE, deviceId);
|
||||||
|
Matrix.setMatrixCallVideoInput(deviceId);
|
||||||
|
},
|
||||||
|
};
|
84
src/ComposerHistoryManager.js
Normal file
84
src/ComposerHistoryManager.js
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
//@flow
|
||||||
|
/*
|
||||||
|
Copyright 2017 Aviral Dasgupta
|
||||||
|
|
||||||
|
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 {ContentState, convertToRaw, convertFromRaw} from 'draft-js';
|
||||||
|
import * as RichText from './RichText';
|
||||||
|
import Markdown from './Markdown';
|
||||||
|
import _clamp from 'lodash/clamp';
|
||||||
|
|
||||||
|
type MessageFormat = 'html' | 'markdown';
|
||||||
|
|
||||||
|
class HistoryItem {
|
||||||
|
|
||||||
|
// Keeping message for backwards-compatibility
|
||||||
|
message: string;
|
||||||
|
rawContentState: RawDraftContentState;
|
||||||
|
format: MessageFormat = 'html';
|
||||||
|
|
||||||
|
constructor(contentState: ?ContentState, format: ?MessageFormat) {
|
||||||
|
this.rawContentState = contentState ? convertToRaw(contentState) : null;
|
||||||
|
this.format = format;
|
||||||
|
}
|
||||||
|
|
||||||
|
toContentState(outputFormat: MessageFormat): ContentState {
|
||||||
|
const contentState = convertFromRaw(this.rawContentState);
|
||||||
|
if (outputFormat === 'markdown') {
|
||||||
|
if (this.format === 'html') {
|
||||||
|
return ContentState.createFromText(RichText.stateToMarkdown(contentState));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (this.format === 'markdown') {
|
||||||
|
return RichText.htmlToContentState(new Markdown(contentState.getPlainText()).toHTML());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// history item has format === outputFormat
|
||||||
|
return contentState;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class ComposerHistoryManager {
|
||||||
|
history: Array<HistoryItem> = [];
|
||||||
|
prefix: string;
|
||||||
|
lastIndex: number = 0;
|
||||||
|
currentIndex: number = 0;
|
||||||
|
|
||||||
|
constructor(roomId: string, prefix: string = 'mx_composer_history_') {
|
||||||
|
this.prefix = prefix + roomId;
|
||||||
|
|
||||||
|
// TODO: Performance issues?
|
||||||
|
let item;
|
||||||
|
for (; item = sessionStorage.getItem(`${this.prefix}[${this.currentIndex}]`); this.currentIndex++) {
|
||||||
|
this.history.push(
|
||||||
|
Object.assign(new HistoryItem(), JSON.parse(item)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
this.lastIndex = this.currentIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
save(contentState: ContentState, format: MessageFormat) {
|
||||||
|
const item = new HistoryItem(contentState, format);
|
||||||
|
this.history.push(item);
|
||||||
|
this.currentIndex = this.lastIndex + 1;
|
||||||
|
sessionStorage.setItem(`${this.prefix}[${this.lastIndex++}]`, JSON.stringify(item));
|
||||||
|
}
|
||||||
|
|
||||||
|
getItem(offset: number, format: MessageFormat): ?ContentState {
|
||||||
|
this.currentIndex = _clamp(this.currentIndex + offset, 0, this.lastIndex - 1);
|
||||||
|
const item = this.history[this.currentIndex];
|
||||||
|
return item ? item.toContentState(format) : null;
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,14 +16,15 @@ limitations under the License.
|
||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var q = require('q');
|
import Promise from 'bluebird';
|
||||||
var extend = require('./extend');
|
const extend = require('./extend');
|
||||||
var dis = require('./dispatcher');
|
const dis = require('./dispatcher');
|
||||||
var MatrixClientPeg = require('./MatrixClientPeg');
|
const MatrixClientPeg = require('./MatrixClientPeg');
|
||||||
var sdk = require('./index');
|
const sdk = require('./index');
|
||||||
var Modal = require('./Modal');
|
import { _t } from './languageHandler';
|
||||||
|
const Modal = require('./Modal');
|
||||||
|
|
||||||
var encrypt = require("browser-encrypt-attachment");
|
const encrypt = require("browser-encrypt-attachment");
|
||||||
|
|
||||||
// Polyfill for Canvas.toBlob API using Canvas.toDataURL
|
// Polyfill for Canvas.toBlob API using Canvas.toDataURL
|
||||||
require("blueimp-canvas-to-blob");
|
require("blueimp-canvas-to-blob");
|
||||||
|
@ -51,10 +52,10 @@ const MAX_HEIGHT = 600;
|
||||||
* and a thumbnail key.
|
* and a thumbnail key.
|
||||||
*/
|
*/
|
||||||
function createThumbnail(element, inputWidth, inputHeight, mimeType) {
|
function createThumbnail(element, inputWidth, inputHeight, mimeType) {
|
||||||
const deferred = q.defer();
|
const deferred = Promise.defer();
|
||||||
|
|
||||||
var targetWidth = inputWidth;
|
let targetWidth = inputWidth;
|
||||||
var targetHeight = inputHeight;
|
let targetHeight = inputHeight;
|
||||||
if (targetHeight > MAX_HEIGHT) {
|
if (targetHeight > MAX_HEIGHT) {
|
||||||
targetWidth = Math.floor(targetWidth * (MAX_HEIGHT / targetHeight));
|
targetWidth = Math.floor(targetWidth * (MAX_HEIGHT / targetHeight));
|
||||||
targetHeight = MAX_HEIGHT;
|
targetHeight = MAX_HEIGHT;
|
||||||
|
@ -80,7 +81,7 @@ function createThumbnail(element, inputWidth, inputHeight, mimeType) {
|
||||||
w: inputWidth,
|
w: inputWidth,
|
||||||
h: inputHeight,
|
h: inputHeight,
|
||||||
},
|
},
|
||||||
thumbnail: thumbnail
|
thumbnail: thumbnail,
|
||||||
});
|
});
|
||||||
}, mimeType);
|
}, mimeType);
|
||||||
|
|
||||||
|
@ -94,27 +95,21 @@ function createThumbnail(element, inputWidth, inputHeight, mimeType) {
|
||||||
* @return {Promise} A promise that resolves with the html image element.
|
* @return {Promise} A promise that resolves with the html image element.
|
||||||
*/
|
*/
|
||||||
function loadImageElement(imageFile) {
|
function loadImageElement(imageFile) {
|
||||||
const deferred = q.defer();
|
const deferred = Promise.defer();
|
||||||
|
|
||||||
// Load the file into an html element
|
// Load the file into an html element
|
||||||
const img = document.createElement("img");
|
const img = document.createElement("img");
|
||||||
|
const objectUrl = URL.createObjectURL(imageFile);
|
||||||
|
img.src = objectUrl;
|
||||||
|
|
||||||
const reader = new FileReader();
|
// Once ready, create a thumbnail
|
||||||
reader.onload = function(e) {
|
img.onload = function() {
|
||||||
img.src = e.target.result;
|
URL.revokeObjectURL(objectUrl);
|
||||||
|
deferred.resolve(img);
|
||||||
// Once ready, create a thumbnail
|
|
||||||
img.onload = function() {
|
|
||||||
deferred.resolve(img);
|
|
||||||
};
|
|
||||||
img.onerror = function(e) {
|
|
||||||
deferred.reject(e);
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
reader.onerror = function(e) {
|
img.onerror = function(e) {
|
||||||
deferred.reject(e);
|
deferred.reject(e);
|
||||||
};
|
};
|
||||||
reader.readAsDataURL(imageFile);
|
|
||||||
|
|
||||||
return deferred.promise;
|
return deferred.promise;
|
||||||
}
|
}
|
||||||
|
@ -128,12 +123,12 @@ function loadImageElement(imageFile) {
|
||||||
* @return {Promise} A promise that resolves with the attachment info.
|
* @return {Promise} A promise that resolves with the attachment info.
|
||||||
*/
|
*/
|
||||||
function infoForImageFile(matrixClient, roomId, imageFile) {
|
function infoForImageFile(matrixClient, roomId, imageFile) {
|
||||||
var thumbnailType = "image/png";
|
let thumbnailType = "image/png";
|
||||||
if (imageFile.type == "image/jpeg") {
|
if (imageFile.type == "image/jpeg") {
|
||||||
thumbnailType = "image/jpeg";
|
thumbnailType = "image/jpeg";
|
||||||
}
|
}
|
||||||
|
|
||||||
var imageInfo;
|
let imageInfo;
|
||||||
return loadImageElement(imageFile).then(function(img) {
|
return loadImageElement(imageFile).then(function(img) {
|
||||||
return createThumbnail(img, img.width, img.height, thumbnailType);
|
return createThumbnail(img, img.width, img.height, thumbnailType);
|
||||||
}).then(function(result) {
|
}).then(function(result) {
|
||||||
|
@ -153,7 +148,7 @@ function infoForImageFile(matrixClient, roomId, imageFile) {
|
||||||
* @return {Promise} A promise that resolves with the video image element.
|
* @return {Promise} A promise that resolves with the video image element.
|
||||||
*/
|
*/
|
||||||
function loadVideoElement(videoFile) {
|
function loadVideoElement(videoFile) {
|
||||||
const deferred = q.defer();
|
const deferred = Promise.defer();
|
||||||
|
|
||||||
// Load the file into an html element
|
// Load the file into an html element
|
||||||
const video = document.createElement("video");
|
const video = document.createElement("video");
|
||||||
|
@ -190,7 +185,7 @@ function loadVideoElement(videoFile) {
|
||||||
function infoForVideoFile(matrixClient, roomId, videoFile) {
|
function infoForVideoFile(matrixClient, roomId, videoFile) {
|
||||||
const thumbnailType = "image/jpeg";
|
const thumbnailType = "image/jpeg";
|
||||||
|
|
||||||
var videoInfo;
|
let videoInfo;
|
||||||
return loadVideoElement(videoFile).then(function(video) {
|
return loadVideoElement(videoFile).then(function(video) {
|
||||||
return createThumbnail(video, video.videoWidth, video.videoHeight, thumbnailType);
|
return createThumbnail(video, video.videoWidth, video.videoHeight, thumbnailType);
|
||||||
}).then(function(result) {
|
}).then(function(result) {
|
||||||
|
@ -209,7 +204,7 @@ function infoForVideoFile(matrixClient, roomId, videoFile) {
|
||||||
* is read.
|
* is read.
|
||||||
*/
|
*/
|
||||||
function readFileAsArrayBuffer(file) {
|
function readFileAsArrayBuffer(file) {
|
||||||
const deferred = q.defer();
|
const deferred = Promise.defer();
|
||||||
const reader = new FileReader();
|
const reader = new FileReader();
|
||||||
reader.onload = function(e) {
|
reader.onload = function(e) {
|
||||||
deferred.resolve(e.target.result);
|
deferred.resolve(e.target.result);
|
||||||
|
@ -228,11 +223,13 @@ function readFileAsArrayBuffer(file) {
|
||||||
* @param {MatrixClient} matrixClient The matrix client to upload the file with.
|
* @param {MatrixClient} matrixClient The matrix client to upload the file with.
|
||||||
* @param {String} roomId The ID of the room being uploaded to.
|
* @param {String} roomId The ID of the room being uploaded to.
|
||||||
* @param {File} file The file to upload.
|
* @param {File} file The file to upload.
|
||||||
|
* @param {Function?} progressHandler optional callback to be called when a chunk of
|
||||||
|
* data is uploaded.
|
||||||
* @return {Promise} A promise that resolves with an object.
|
* @return {Promise} A promise that resolves with an object.
|
||||||
* If the file is unencrypted then the object will have a "url" key.
|
* If the file is unencrypted then the object will have a "url" key.
|
||||||
* If the file is encrypted then the object will have a "file" key.
|
* If the file is encrypted then the object will have a "file" key.
|
||||||
*/
|
*/
|
||||||
function uploadFile(matrixClient, roomId, file) {
|
function uploadFile(matrixClient, roomId, file, progressHandler) {
|
||||||
if (matrixClient.isRoomEncrypted(roomId)) {
|
if (matrixClient.isRoomEncrypted(roomId)) {
|
||||||
// If the room is encrypted then encrypt the file before uploading it.
|
// If the room is encrypted then encrypt the file before uploading it.
|
||||||
// First read the file into memory.
|
// First read the file into memory.
|
||||||
|
@ -244,7 +241,9 @@ function uploadFile(matrixClient, roomId, file) {
|
||||||
const encryptInfo = encryptResult.info;
|
const encryptInfo = encryptResult.info;
|
||||||
// Pass the encrypted data as a Blob to the uploader.
|
// Pass the encrypted data as a Blob to the uploader.
|
||||||
const blob = new Blob([encryptResult.data]);
|
const blob = new Blob([encryptResult.data]);
|
||||||
return matrixClient.uploadContent(blob).then(function(url) {
|
return matrixClient.uploadContent(blob, {
|
||||||
|
progressHandler: progressHandler,
|
||||||
|
}).then(function(url) {
|
||||||
// If the attachment is encrypted then bundle the URL along
|
// If the attachment is encrypted then bundle the URL along
|
||||||
// with the information needed to decrypt the attachment and
|
// with the information needed to decrypt the attachment and
|
||||||
// add it under a file key.
|
// add it under a file key.
|
||||||
|
@ -256,7 +255,9 @@ function uploadFile(matrixClient, roomId, file) {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
const basePromise = matrixClient.uploadContent(file);
|
const basePromise = matrixClient.uploadContent(file, {
|
||||||
|
progressHandler: progressHandler,
|
||||||
|
});
|
||||||
const promise1 = basePromise.then(function(url) {
|
const promise1 = basePromise.then(function(url) {
|
||||||
// If the attachment isn't encrypted then include the URL directly.
|
// If the attachment isn't encrypted then include the URL directly.
|
||||||
return {"url": url};
|
return {"url": url};
|
||||||
|
@ -276,10 +277,10 @@ class ContentMessages {
|
||||||
|
|
||||||
sendContentToRoom(file, roomId, matrixClient) {
|
sendContentToRoom(file, roomId, matrixClient) {
|
||||||
const content = {
|
const content = {
|
||||||
body: file.name,
|
body: file.name || 'Attachment',
|
||||||
info: {
|
info: {
|
||||||
size: file.size,
|
size: file.size,
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
// if we have a mime type for the file, add it to the message metadata
|
// if we have a mime type for the file, add it to the message metadata
|
||||||
|
@ -287,13 +288,13 @@ class ContentMessages {
|
||||||
content.info.mimetype = file.type;
|
content.info.mimetype = file.type;
|
||||||
}
|
}
|
||||||
|
|
||||||
const def = q.defer();
|
const def = Promise.defer();
|
||||||
if (file.type.indexOf('image/') == 0) {
|
if (file.type.indexOf('image/') == 0) {
|
||||||
content.msgtype = 'm.image';
|
content.msgtype = 'm.image';
|
||||||
infoForImageFile(matrixClient, roomId, file).then(imageInfo=>{
|
infoForImageFile(matrixClient, roomId, file).then((imageInfo)=>{
|
||||||
extend(content.info, imageInfo);
|
extend(content.info, imageInfo);
|
||||||
def.resolve();
|
def.resolve();
|
||||||
}, error=>{
|
}, (error)=>{
|
||||||
console.error(error);
|
console.error(error);
|
||||||
content.msgtype = 'm.file';
|
content.msgtype = 'm.file';
|
||||||
def.resolve();
|
def.resolve();
|
||||||
|
@ -303,10 +304,10 @@ class ContentMessages {
|
||||||
def.resolve();
|
def.resolve();
|
||||||
} else if (file.type.indexOf('video/') == 0) {
|
} else if (file.type.indexOf('video/') == 0) {
|
||||||
content.msgtype = 'm.video';
|
content.msgtype = 'm.video';
|
||||||
infoForVideoFile(matrixClient, roomId, file).then(videoInfo=>{
|
infoForVideoFile(matrixClient, roomId, file).then((videoInfo)=>{
|
||||||
extend(content.info, videoInfo);
|
extend(content.info, videoInfo);
|
||||||
def.resolve();
|
def.resolve();
|
||||||
}, error=>{
|
}, (error)=>{
|
||||||
content.msgtype = 'm.file';
|
content.msgtype = 'm.file';
|
||||||
def.resolve();
|
def.resolve();
|
||||||
});
|
});
|
||||||
|
@ -316,7 +317,7 @@ class ContentMessages {
|
||||||
}
|
}
|
||||||
|
|
||||||
const upload = {
|
const upload = {
|
||||||
fileName: file.name,
|
fileName: file.name || 'Attachment',
|
||||||
roomId: roomId,
|
roomId: roomId,
|
||||||
total: 0,
|
total: 0,
|
||||||
loaded: 0,
|
loaded: 0,
|
||||||
|
@ -324,43 +325,44 @@ class ContentMessages {
|
||||||
this.inprogress.push(upload);
|
this.inprogress.push(upload);
|
||||||
dis.dispatch({action: 'upload_started'});
|
dis.dispatch({action: 'upload_started'});
|
||||||
|
|
||||||
var error;
|
let error;
|
||||||
|
|
||||||
|
function onProgress(ev) {
|
||||||
|
upload.total = ev.total;
|
||||||
|
upload.loaded = ev.loaded;
|
||||||
|
dis.dispatch({action: 'upload_progress', upload: upload});
|
||||||
|
}
|
||||||
|
|
||||||
return def.promise.then(function() {
|
return def.promise.then(function() {
|
||||||
// XXX: upload.promise must be the promise that
|
// XXX: upload.promise must be the promise that
|
||||||
// is returned by uploadFile as it has an abort()
|
// is returned by uploadFile as it has an abort()
|
||||||
// method hacked onto it.
|
// method hacked onto it.
|
||||||
upload.promise = uploadFile(
|
upload.promise = uploadFile(
|
||||||
matrixClient, roomId, file
|
matrixClient, roomId, file, onProgress,
|
||||||
);
|
);
|
||||||
return upload.promise.then(function(result) {
|
return upload.promise.then(function(result) {
|
||||||
content.file = result.file;
|
content.file = result.file;
|
||||||
content.url = result.url;
|
content.url = result.url;
|
||||||
});
|
});
|
||||||
}).progress(function(ev) {
|
|
||||||
if (ev) {
|
|
||||||
upload.total = ev.total;
|
|
||||||
upload.loaded = ev.loaded;
|
|
||||||
dis.dispatch({action: 'upload_progress', upload: upload});
|
|
||||||
}
|
|
||||||
}).then(function(url) {
|
}).then(function(url) {
|
||||||
return matrixClient.sendMessage(roomId, content);
|
return matrixClient.sendMessage(roomId, content);
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
error = err;
|
error = err;
|
||||||
if (!upload.canceled) {
|
if (!upload.canceled) {
|
||||||
var desc = "The file '"+upload.fileName+"' failed to upload.";
|
let desc = _t('The file \'%(fileName)s\' failed to upload', {fileName: upload.fileName}) + '.';
|
||||||
if (err.http_status == 413) {
|
if (err.http_status == 413) {
|
||||||
desc = "The file '"+upload.fileName+"' exceeds this home server's size limit for uploads";
|
desc = _t('The file \'%(fileName)s\' exceeds this home server\'s size limit for uploads', {fileName: upload.fileName});
|
||||||
}
|
}
|
||||||
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Upload failed', '', ErrorDialog, {
|
||||||
title: "Upload Failed",
|
title: _t('Upload Failed'),
|
||||||
description: desc
|
description: desc,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}).finally(() => {
|
}).finally(() => {
|
||||||
const inprogressKeys = Object.keys(this.inprogress);
|
const inprogressKeys = Object.keys(this.inprogress);
|
||||||
for (var i = 0; i < this.inprogress.length; ++i) {
|
for (let i = 0; i < this.inprogress.length; ++i) {
|
||||||
var k = inprogressKeys[i];
|
const k = inprogressKeys[i];
|
||||||
if (this.inprogress[k].promise === upload.promise) {
|
if (this.inprogress[k].promise === upload.promise) {
|
||||||
this.inprogress.splice(k, 1);
|
this.inprogress.splice(k, 1);
|
||||||
break;
|
break;
|
||||||
|
@ -368,8 +370,7 @@ class ContentMessages {
|
||||||
}
|
}
|
||||||
if (error) {
|
if (error) {
|
||||||
dis.dispatch({action: 'upload_failed', upload: upload});
|
dis.dispatch({action: 'upload_failed', upload: upload});
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
dis.dispatch({action: 'upload_finished', upload: upload});
|
dis.dispatch({action: 'upload_finished', upload: upload});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -381,9 +382,9 @@ class ContentMessages {
|
||||||
|
|
||||||
cancelUpload(promise) {
|
cancelUpload(promise) {
|
||||||
const inprogressKeys = Object.keys(this.inprogress);
|
const inprogressKeys = Object.keys(this.inprogress);
|
||||||
var upload;
|
let upload;
|
||||||
for (var i = 0; i < this.inprogress.length; ++i) {
|
for (let i = 0; i < this.inprogress.length; ++i) {
|
||||||
var k = inprogressKeys[i];
|
const k = inprogressKeys[i];
|
||||||
if (this.inprogress[k].promise === promise) {
|
if (this.inprogress[k].promise === promise) {
|
||||||
upload = this.inprogress[k];
|
upload = this.inprogress[k];
|
||||||
break;
|
break;
|
||||||
|
|
141
src/DateUtils.js
141
src/DateUtils.js
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -14,39 +15,113 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'use strict';
|
import { _t } from './languageHandler';
|
||||||
|
|
||||||
var days = ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"];
|
function getDaysArray() {
|
||||||
var months = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"];
|
return [
|
||||||
|
_t('Sun'),
|
||||||
module.exports = {
|
_t('Mon'),
|
||||||
formatDate: function(date) {
|
_t('Tue'),
|
||||||
// date.toLocaleTimeString is completely system dependent.
|
_t('Wed'),
|
||||||
// just go 24h for now
|
_t('Thu'),
|
||||||
function pad(n) {
|
_t('Fri'),
|
||||||
return (n < 10 ? '0' : '') + n;
|
_t('Sat'),
|
||||||
}
|
];
|
||||||
|
|
||||||
var now = new Date();
|
|
||||||
if (date.toDateString() === now.toDateString()) {
|
|
||||||
return pad(date.getHours()) + ':' + pad(date.getMinutes());
|
|
||||||
}
|
|
||||||
else if (now.getTime() - date.getTime() < 6 * 24 * 60 * 60 * 1000) {
|
|
||||||
return days[date.getDay()] + " " + pad(date.getHours()) + ':' + pad(date.getMinutes());
|
|
||||||
}
|
|
||||||
else /* if (now.getFullYear() === date.getFullYear()) */ {
|
|
||||||
return days[date.getDay()] + ", " + months[date.getMonth()] + " " + date.getDate() + " " + pad(date.getHours()) + ':' + pad(date.getMinutes());
|
|
||||||
}
|
|
||||||
/*
|
|
||||||
else {
|
|
||||||
return days[date.getDay()] + ", " + months[date.getMonth()] + " " + date.getDate() + " " + date.getFullYear() + " " + pad(date.getHours()) + ':' + pad(date.getMinutes());
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
},
|
|
||||||
|
|
||||||
formatTime: function(date) {
|
|
||||||
//return pad(date.getHours()) + ':' + pad(date.getMinutes());
|
|
||||||
return ('00' + date.getHours()).slice(-2) + ':' + ('00' + date.getMinutes()).slice(-2);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getMonthsArray() {
|
||||||
|
return [
|
||||||
|
_t('Jan'),
|
||||||
|
_t('Feb'),
|
||||||
|
_t('Mar'),
|
||||||
|
_t('Apr'),
|
||||||
|
_t('May'),
|
||||||
|
_t('Jun'),
|
||||||
|
_t('Jul'),
|
||||||
|
_t('Aug'),
|
||||||
|
_t('Sep'),
|
||||||
|
_t('Oct'),
|
||||||
|
_t('Nov'),
|
||||||
|
_t('Dec'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
function pad(n) {
|
||||||
|
return (n < 10 ? '0' : '') + n;
|
||||||
|
}
|
||||||
|
|
||||||
|
function twelveHourTime(date) {
|
||||||
|
let hours = date.getHours() % 12;
|
||||||
|
const minutes = pad(date.getMinutes());
|
||||||
|
const ampm = date.getHours() >= 12 ? _t('PM') : _t('AM');
|
||||||
|
hours = hours ? hours : 12; // convert 0 -> 12
|
||||||
|
return `${hours}:${minutes}${ampm}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatDate(date, showTwelveHour=false) {
|
||||||
|
const now = new Date();
|
||||||
|
const days = getDaysArray();
|
||||||
|
const months = getMonthsArray();
|
||||||
|
if (date.toDateString() === now.toDateString()) {
|
||||||
|
return formatTime(date, showTwelveHour);
|
||||||
|
} else if (now.getTime() - date.getTime() < 6 * 24 * 60 * 60 * 1000) {
|
||||||
|
// TODO: use standard date localize function provided in counterpart
|
||||||
|
return _t('%(weekDayName)s %(time)s', {
|
||||||
|
weekDayName: days[date.getDay()],
|
||||||
|
time: formatTime(date, showTwelveHour),
|
||||||
|
});
|
||||||
|
} else if (now.getFullYear() === date.getFullYear()) {
|
||||||
|
// TODO: use standard date localize function provided in counterpart
|
||||||
|
return _t('%(weekDayName)s, %(monthName)s %(day)s %(time)s', {
|
||||||
|
weekDayName: days[date.getDay()],
|
||||||
|
monthName: months[date.getMonth()],
|
||||||
|
day: date.getDate(),
|
||||||
|
time: formatTime(date, showTwelveHour),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return formatFullDate(date, showTwelveHour);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatFullDateNoTime(date) {
|
||||||
|
const days = getDaysArray();
|
||||||
|
const months = getMonthsArray();
|
||||||
|
return _t('%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s', {
|
||||||
|
weekDayName: days[date.getDay()],
|
||||||
|
monthName: months[date.getMonth()],
|
||||||
|
day: date.getDate(),
|
||||||
|
fullYear: date.getFullYear(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatFullDate(date, showTwelveHour=false) {
|
||||||
|
const days = getDaysArray();
|
||||||
|
const months = getMonthsArray();
|
||||||
|
return _t('%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s %(time)s', {
|
||||||
|
weekDayName: days[date.getDay()],
|
||||||
|
monthName: months[date.getMonth()],
|
||||||
|
day: date.getDate(),
|
||||||
|
fullYear: date.getFullYear(),
|
||||||
|
time: formatTime(date, showTwelveHour),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatTime(date, showTwelveHour=false) {
|
||||||
|
if (showTwelveHour) {
|
||||||
|
return twelveHourTime(date);
|
||||||
|
}
|
||||||
|
return pad(date.getHours()) + ':' + pad(date.getMinutes());
|
||||||
|
}
|
||||||
|
|
||||||
|
const MILLIS_IN_DAY = 86400000;
|
||||||
|
export function wantsDateSeparator(prevEventDate, nextEventDate) {
|
||||||
|
if (!nextEventDate || !prevEventDate) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
// Return early for events that are > 24h apart
|
||||||
|
if (Math.abs(prevEventDate.getTime() - nextEventDate.getTime()) > MILLIS_IN_DAY) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compare weekdays
|
||||||
|
return prevEventDate.getDay() !== nextEventDate.getDay();
|
||||||
|
}
|
||||||
|
|
|
@ -14,8 +14,7 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var React = require('react');
|
import sdk from './index';
|
||||||
var sdk = require('./index');
|
|
||||||
|
|
||||||
function isMatch(query, name, uid) {
|
function isMatch(query, name, uid) {
|
||||||
query = query.toLowerCase();
|
query = query.toLowerCase();
|
||||||
|
@ -33,8 +32,8 @@ function isMatch(query, name, uid) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// split spaces in name and try matching constituent parts
|
// split spaces in name and try matching constituent parts
|
||||||
var parts = name.split(" ");
|
const parts = name.split(" ");
|
||||||
for (var i = 0; i < parts.length; i++) {
|
for (let i = 0; i < parts.length; i++) {
|
||||||
if (parts[i].indexOf(query) === 0) {
|
if (parts[i].indexOf(query) === 0) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -67,7 +66,7 @@ class Entity {
|
||||||
|
|
||||||
class MemberEntity extends Entity {
|
class MemberEntity extends Entity {
|
||||||
getJsx() {
|
getJsx() {
|
||||||
var MemberTile = sdk.getComponent("rooms.MemberTile");
|
const MemberTile = sdk.getComponent("rooms.MemberTile");
|
||||||
return (
|
return (
|
||||||
<MemberTile key={this.model.userId} member={this.model} />
|
<MemberTile key={this.model.userId} member={this.model} />
|
||||||
);
|
);
|
||||||
|
@ -84,6 +83,7 @@ class UserEntity extends Entity {
|
||||||
super(model);
|
super(model);
|
||||||
this.showInviteButton = Boolean(showInviteButton);
|
this.showInviteButton = Boolean(showInviteButton);
|
||||||
this.inviteFn = inviteFn;
|
this.inviteFn = inviteFn;
|
||||||
|
this.onClick = this.onClick.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
onClick() {
|
onClick() {
|
||||||
|
@ -93,15 +93,15 @@ class UserEntity extends Entity {
|
||||||
}
|
}
|
||||||
|
|
||||||
getJsx() {
|
getJsx() {
|
||||||
var UserTile = sdk.getComponent("rooms.UserTile");
|
const UserTile = sdk.getComponent("rooms.UserTile");
|
||||||
return (
|
return (
|
||||||
<UserTile key={this.model.userId} user={this.model}
|
<UserTile key={this.model.userId} user={this.model}
|
||||||
showInviteButton={this.showInviteButton} onClick={this.onClick.bind(this)} />
|
showInviteButton={this.showInviteButton} onClick={this.onClick} />
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
matches(queryString) {
|
matches(queryString) {
|
||||||
var name = this.model.displayName || this.model.userId;
|
const name = this.model.displayName || this.model.userId;
|
||||||
return isMatch(queryString, name, this.model.userId);
|
return isMatch(queryString, name, this.model.userId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -109,7 +109,7 @@ class UserEntity extends Entity {
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
newEntity: function(jsx, matchFn) {
|
newEntity: function(jsx, matchFn) {
|
||||||
var entity = new Entity();
|
const entity = new Entity();
|
||||||
entity.getJsx = function() {
|
entity.getJsx = function() {
|
||||||
return jsx;
|
return jsx;
|
||||||
};
|
};
|
||||||
|
@ -136,6 +136,6 @@ module.exports = {
|
||||||
fromUsers: function(users, showInviteButton, inviteFn) {
|
fromUsers: function(users, showInviteButton, inviteFn) {
|
||||||
return users.map(function(u) {
|
return users.map(function(u) {
|
||||||
return new UserEntity(u, showInviteButton, inviteFn);
|
return new UserEntity(u, showInviteButton, inviteFn);
|
||||||
})
|
});
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
|
|
156
src/GroupAddressPicker.js
Normal file
156
src/GroupAddressPicker.js
Normal file
|
@ -0,0 +1,156 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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 Modal from './Modal';
|
||||||
|
import sdk from './';
|
||||||
|
import MultiInviter from './utils/MultiInviter';
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
import GroupStoreCache from './stores/GroupStoreCache';
|
||||||
|
|
||||||
|
export function showGroupInviteDialog(groupId) {
|
||||||
|
const description = <div>
|
||||||
|
<div>{ _t("Who would you like to add to this community?") }</div>
|
||||||
|
<div className="warning">
|
||||||
|
{ _t(
|
||||||
|
"Warning: any person you add to a community will be publicly "+
|
||||||
|
"visible to anyone who knows the community ID",
|
||||||
|
) }
|
||||||
|
</div>
|
||||||
|
</div>;
|
||||||
|
|
||||||
|
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
|
||||||
|
Modal.createTrackedDialog('Group Invite', '', AddressPickerDialog, {
|
||||||
|
title: _t("Invite new community members"),
|
||||||
|
description: description,
|
||||||
|
placeholder: _t("Name or matrix ID"),
|
||||||
|
button: _t("Invite to Community"),
|
||||||
|
validAddressTypes: ['mx-user-id'],
|
||||||
|
onFinished: (success, addrs) => {
|
||||||
|
if (!success) return;
|
||||||
|
|
||||||
|
_onGroupInviteFinished(groupId, addrs);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function showGroupAddRoomDialog(groupId) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
let addRoomsPublicly = false;
|
||||||
|
const onCheckboxClicked = (e) => {
|
||||||
|
addRoomsPublicly = e.target.checked;
|
||||||
|
};
|
||||||
|
const description = <div>
|
||||||
|
<div>{ _t("Which rooms would you like to add to this community?") }</div>
|
||||||
|
</div>;
|
||||||
|
|
||||||
|
const checkboxContainer = <label className="mx_GroupAddressPicker_checkboxContainer">
|
||||||
|
<input type="checkbox" onClick={onCheckboxClicked} />
|
||||||
|
<div>
|
||||||
|
{ _t("Show these rooms to non-members on the community page and room list?") }
|
||||||
|
</div>
|
||||||
|
</label>;
|
||||||
|
|
||||||
|
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
|
||||||
|
Modal.createTrackedDialog('Add Rooms to Group', '', AddressPickerDialog, {
|
||||||
|
title: _t("Add rooms to the community"),
|
||||||
|
description: description,
|
||||||
|
extraNode: checkboxContainer,
|
||||||
|
placeholder: _t("Room name or alias"),
|
||||||
|
button: _t("Add to community"),
|
||||||
|
pickerType: 'room',
|
||||||
|
validAddressTypes: ['mx-room-id'],
|
||||||
|
onFinished: (success, addrs) => {
|
||||||
|
if (!success) return;
|
||||||
|
|
||||||
|
_onGroupAddRoomFinished(groupId, addrs, addRoomsPublicly).then(resolve, reject);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function _onGroupInviteFinished(groupId, addrs) {
|
||||||
|
const multiInviter = new MultiInviter(groupId);
|
||||||
|
|
||||||
|
const addrTexts = addrs.map((addr) => addr.address);
|
||||||
|
|
||||||
|
multiInviter.invite(addrTexts).then((completionStates) => {
|
||||||
|
// Show user any errors
|
||||||
|
const errorList = [];
|
||||||
|
for (const addr of Object.keys(completionStates)) {
|
||||||
|
if (addrs[addr] === "error") {
|
||||||
|
errorList.push(addr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errorList.length > 0) {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite the following users to the group', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite the following users to %(groupId)s:", {groupId: groupId}),
|
||||||
|
description: errorList.join(", "),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}).catch((err) => {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite users to group', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite users to community"),
|
||||||
|
description: _t("Failed to invite users to %(groupId)s", {groupId: groupId}),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function _onGroupAddRoomFinished(groupId, addrs, addRoomsPublicly) {
|
||||||
|
const matrixClient = MatrixClientPeg.get();
|
||||||
|
const groupStore = GroupStoreCache.getGroupStore(groupId);
|
||||||
|
const errorList = [];
|
||||||
|
return Promise.all(addrs.map((addr) => {
|
||||||
|
return groupStore
|
||||||
|
.addRoomToGroup(addr.address, addRoomsPublicly)
|
||||||
|
.catch(() => { errorList.push(addr.address); })
|
||||||
|
.then(() => {
|
||||||
|
const roomId = addr.address;
|
||||||
|
const room = matrixClient.getRoom(roomId);
|
||||||
|
// Can the user change related groups?
|
||||||
|
if (!room || !room.currentState.mayClientSendStateEvent("m.room.related_groups", matrixClient)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Get the related groups
|
||||||
|
const relatedGroupsEvent = room.currentState.getStateEvents('m.room.related_groups', '');
|
||||||
|
const groups = relatedGroupsEvent ? relatedGroupsEvent.getContent().groups || [] : [];
|
||||||
|
|
||||||
|
// Add this group as related
|
||||||
|
if (!groups.includes(groupId)) {
|
||||||
|
groups.push(groupId);
|
||||||
|
return MatrixClientPeg.get().sendStateEvent(roomId, 'm.room.related_groups', {groups}, '');
|
||||||
|
}
|
||||||
|
}).reflect();
|
||||||
|
})).then(() => {
|
||||||
|
if (errorList.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog(
|
||||||
|
'Failed to add the following room to the group',
|
||||||
|
'', ErrorDialog,
|
||||||
|
{
|
||||||
|
title: _t(
|
||||||
|
"Failed to add the following rooms to %(groupId)s:",
|
||||||
|
{groupId},
|
||||||
|
),
|
||||||
|
description: errorList.join(", "),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
264
src/HtmlUtils.js
264
src/HtmlUtils.js
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 New Vector Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -16,40 +17,67 @@ limitations under the License.
|
||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var React = require('react');
|
const React = require('react');
|
||||||
var sanitizeHtml = require('sanitize-html');
|
const sanitizeHtml = require('sanitize-html');
|
||||||
var highlight = require('highlight.js');
|
const highlight = require('highlight.js');
|
||||||
var linkifyMatrix = require('./linkify-matrix');
|
const linkifyMatrix = require('./linkify-matrix');
|
||||||
import escape from 'lodash/escape';
|
import escape from 'lodash/escape';
|
||||||
import emojione from 'emojione';
|
import emojione from 'emojione';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
|
||||||
emojione.imagePathSVG = 'emojione/svg/';
|
emojione.imagePathSVG = 'emojione/svg/';
|
||||||
|
// Store PNG path for displaying many flags at once (for increased performance over SVG)
|
||||||
|
emojione.imagePathPNG = 'emojione/png/';
|
||||||
|
// Use SVGs for emojis
|
||||||
emojione.imageType = 'svg';
|
emojione.imageType = 'svg';
|
||||||
|
|
||||||
|
// Anything outside the basic multilingual plane will be a surrogate pair
|
||||||
|
const SURROGATE_PAIR_PATTERN = /([\ud800-\udbff])([\udc00-\udfff])/;
|
||||||
|
// And there a bunch more symbol characters that emojione has within the
|
||||||
|
// BMP, so this includes the ranges from 'letterlike symbols' to
|
||||||
|
// 'miscellaneous symbols and arrows' which should catch all of them
|
||||||
|
// (with plenty of false positives, but that's OK)
|
||||||
|
const SYMBOL_PATTERN = /([\u2100-\u2bff])/;
|
||||||
|
|
||||||
|
// And this is emojione's complete regex
|
||||||
const EMOJI_REGEX = new RegExp(emojione.unicodeRegexp+"+", "gi");
|
const EMOJI_REGEX = new RegExp(emojione.unicodeRegexp+"+", "gi");
|
||||||
|
const COLOR_REGEX = /^#[0-9a-fA-F]{6}$/;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return true if the given string contains emoji
|
||||||
|
* Uses a much, much simpler regex than emojione's so will give false
|
||||||
|
* positives, but useful for fast-path testing strings to see if they
|
||||||
|
* need emojification.
|
||||||
|
* unicodeToImage uses this function.
|
||||||
|
*/
|
||||||
|
export function containsEmoji(str) {
|
||||||
|
return SURROGATE_PAIR_PATTERN.test(str) || SYMBOL_PATTERN.test(str);
|
||||||
|
}
|
||||||
|
|
||||||
/* modified from https://github.com/Ranks/emojione/blob/master/lib/js/emojione.js
|
/* modified from https://github.com/Ranks/emojione/blob/master/lib/js/emojione.js
|
||||||
* because we want to include emoji shortnames in title text
|
* because we want to include emoji shortnames in title text
|
||||||
*/
|
*/
|
||||||
export function unicodeToImage(str) {
|
function unicodeToImage(str) {
|
||||||
let replaceWith, unicode, alt;
|
let replaceWith, unicode, alt, short, fname;
|
||||||
const mappedUnicode = emojione.mapUnicodeToShort();
|
const mappedUnicode = emojione.mapUnicodeToShort();
|
||||||
|
|
||||||
str = str.replace(emojione.regUnicode, function(unicodeChar) {
|
str = str.replace(emojione.regUnicode, function(unicodeChar) {
|
||||||
if ( (typeof unicodeChar === 'undefined') || (unicodeChar === '') || (!(unicodeChar in emojione.jsEscapeMap)) ) {
|
if ( (typeof unicodeChar === 'undefined') || (unicodeChar === '') || (!(unicodeChar in emojione.jsEscapeMap)) ) {
|
||||||
// if the unicodeChar doesnt exist just return the entire match
|
// if the unicodeChar doesnt exist just return the entire match
|
||||||
return unicodeChar;
|
return unicodeChar;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
// get the unicode codepoint from the actual char
|
// get the unicode codepoint from the actual char
|
||||||
unicode = emojione.jsEscapeMap[unicodeChar];
|
unicode = emojione.jsEscapeMap[unicodeChar];
|
||||||
|
|
||||||
|
short = mappedUnicode[unicode];
|
||||||
|
fname = emojione.emojioneList[short].fname;
|
||||||
|
|
||||||
// depending on the settings, we'll either add the native unicode as the alt tag, otherwise the shortname
|
// depending on the settings, we'll either add the native unicode as the alt tag, otherwise the shortname
|
||||||
alt = (emojione.unicodeAlt) ? emojione.convert(unicode.toUpperCase()) : mappedUnicode[unicode];
|
alt = (emojione.unicodeAlt) ? emojione.convert(unicode.toUpperCase()) : mappedUnicode[unicode];
|
||||||
const title = mappedUnicode[unicode];
|
const title = mappedUnicode[unicode];
|
||||||
|
|
||||||
replaceWith = `<img class="mx_emojione" title="${title}" alt="${alt}" src="${emojione.imagePathSVG}${unicode}.svg${emojione.cacheBustParam}"/>`;
|
replaceWith = `<img class="mx_emojione" title="${title}" alt="${alt}" src="${emojione.imagePathSVG}${fname}.svg${emojione.cacheBustParam}"/>`;
|
||||||
return replaceWith;
|
return replaceWith;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -57,7 +85,30 @@ export function unicodeToImage(str) {
|
||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function stripParagraphs(html: string): string {
|
/**
|
||||||
|
* Given one or more unicode characters (represented by unicode
|
||||||
|
* character number), return an image node with the corresponding
|
||||||
|
* emoji.
|
||||||
|
*
|
||||||
|
* @param alt {string} String to use for the image alt text
|
||||||
|
* @param useSvg {boolean} Whether to use SVG image src. If False, PNG will be used.
|
||||||
|
* @param unicode {integer} One or more integers representing unicode characters
|
||||||
|
* @returns A img node with the corresponding emoji
|
||||||
|
*/
|
||||||
|
export function charactersToImageNode(alt, useSvg, ...unicode) {
|
||||||
|
const fileName = unicode.map((u) => {
|
||||||
|
return u.toString(16);
|
||||||
|
}).join('-');
|
||||||
|
const path = useSvg ? emojione.imagePathSVG : emojione.imagePathPNG;
|
||||||
|
const fileType = useSvg ? 'svg' : 'png';
|
||||||
|
return <img
|
||||||
|
alt={alt}
|
||||||
|
src={`${path}${fileName}.${fileType}${emojione.cacheBustParam}`}
|
||||||
|
/>;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export function processHtmlForSending(html: string): string {
|
||||||
const contentDiv = document.createElement('div');
|
const contentDiv = document.createElement('div');
|
||||||
contentDiv.innerHTML = html;
|
contentDiv.innerHTML = html;
|
||||||
|
|
||||||
|
@ -66,10 +117,21 @@ export function stripParagraphs(html: string): string {
|
||||||
}
|
}
|
||||||
|
|
||||||
let contentHTML = "";
|
let contentHTML = "";
|
||||||
for (let i=0; i<contentDiv.children.length; i++) {
|
for (let i=0; i < contentDiv.children.length; i++) {
|
||||||
const element = contentDiv.children[i];
|
const element = contentDiv.children[i];
|
||||||
if (element.tagName.toLowerCase() === 'p') {
|
if (element.tagName.toLowerCase() === 'p') {
|
||||||
contentHTML += element.innerHTML + '<br />';
|
contentHTML += element.innerHTML;
|
||||||
|
// Don't add a <br /> for the last <p>
|
||||||
|
if (i !== contentDiv.children.length - 1) {
|
||||||
|
contentHTML += '<br />';
|
||||||
|
}
|
||||||
|
} else if (element.tagName.toLowerCase() === 'pre') {
|
||||||
|
// Replace "<br>\n" with "\n" within `<pre>` tags because the <br> is
|
||||||
|
// redundant. This is a workaround for a bug in draft-js-export-html:
|
||||||
|
// https://github.com/sstur/draft-js-export-html/issues/62
|
||||||
|
contentHTML += '<pre>' +
|
||||||
|
element.innerHTML.replace(/<br>\n/g, '\n').trim() +
|
||||||
|
'</pre>';
|
||||||
} else {
|
} else {
|
||||||
const temp = document.createElement('div');
|
const temp = document.createElement('div');
|
||||||
temp.appendChild(element.cloneNode(true));
|
temp.appendChild(element.cloneNode(true));
|
||||||
|
@ -80,32 +142,39 @@ export function stripParagraphs(html: string): string {
|
||||||
return contentHTML;
|
return contentHTML;
|
||||||
}
|
}
|
||||||
|
|
||||||
var sanitizeHtmlParams = {
|
/*
|
||||||
|
* Given an untrusted HTML string, return a React node with an sanitized version
|
||||||
|
* of that HTML.
|
||||||
|
*/
|
||||||
|
export function sanitizedHtmlNode(insaneHtml) {
|
||||||
|
const saneHtml = sanitizeHtml(insaneHtml, sanitizeHtmlParams);
|
||||||
|
|
||||||
|
return <div dangerouslySetInnerHTML={{ __html: saneHtml }} dir="auto" />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const sanitizeHtmlParams = {
|
||||||
allowedTags: [
|
allowedTags: [
|
||||||
'font', // custom to matrix for IRC-style font coloring
|
'font', // custom to matrix for IRC-style font coloring
|
||||||
'del', // for markdown
|
'del', // for markdown
|
||||||
// deliberately no h1/h2 to stop people shouting.
|
'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'blockquote', 'p', 'a', 'ul', 'ol', 'sup', 'sub',
|
||||||
'h3', 'h4', 'h5', 'h6', 'blockquote', 'p', 'a', 'ul', 'ol',
|
|
||||||
'nl', 'li', 'b', 'i', 'u', 'strong', 'em', 'strike', 'code', 'hr', 'br', 'div',
|
'nl', 'li', 'b', 'i', 'u', 'strong', 'em', 'strike', 'code', 'hr', 'br', 'div',
|
||||||
'table', 'thead', 'caption', 'tbody', 'tr', 'th', 'td', 'pre'
|
'table', 'thead', 'caption', 'tbody', 'tr', 'th', 'td', 'pre', 'span', 'img',
|
||||||
],
|
],
|
||||||
allowedAttributes: {
|
allowedAttributes: {
|
||||||
// custom ones first:
|
// custom ones first:
|
||||||
font: [ 'color' ], // custom to matrix
|
font: ['color', 'data-mx-bg-color', 'data-mx-color', 'style'], // custom to matrix
|
||||||
a: [ 'href', 'name', 'target', 'rel' ], // remote target: custom to matrix
|
span: ['data-mx-bg-color', 'data-mx-color', 'style'], // custom to matrix
|
||||||
// We don't currently allow img itself by default, but this
|
a: ['href', 'name', 'target', 'rel'], // remote target: custom to matrix
|
||||||
// would make sense if we did
|
img: ['src', 'width', 'height', 'alt', 'title'],
|
||||||
img: [ 'src' ],
|
ol: ['start'],
|
||||||
|
code: ['class'], // We don't actually allow all classes, we filter them in transformTags
|
||||||
},
|
},
|
||||||
// Lots of these won't come up by default because we don't allow them
|
// Lots of these won't come up by default because we don't allow them
|
||||||
selfClosing: [ 'img', 'br', 'hr', 'area', 'base', 'basefont', 'input', 'link', 'meta' ],
|
selfClosing: ['img', 'br', 'hr', 'area', 'base', 'basefont', 'input', 'link', 'meta'],
|
||||||
// URL schemes we permit
|
// URL schemes we permit
|
||||||
allowedSchemes: [ 'http', 'https', 'ftp', 'mailto' ],
|
allowedSchemes: ['http', 'https', 'ftp', 'mailto', 'magnet'],
|
||||||
|
|
||||||
// DO NOT USE. sanitize-html allows all URL starting with '//'
|
allowProtocolRelative: false,
|
||||||
// so this will always allow links to whatever scheme the
|
|
||||||
// host page is served over.
|
|
||||||
allowedSchemesByTag: {},
|
|
||||||
|
|
||||||
transformTags: { // custom to matrix
|
transformTags: { // custom to matrix
|
||||||
// add blank targets to all hyperlinks except vector URLs
|
// add blank targets to all hyperlinks except vector URLs
|
||||||
|
@ -113,28 +182,86 @@ var sanitizeHtmlParams = {
|
||||||
if (attribs.href) {
|
if (attribs.href) {
|
||||||
attribs.target = '_blank'; // by default
|
attribs.target = '_blank'; // by default
|
||||||
|
|
||||||
var m;
|
let m;
|
||||||
// FIXME: horrible duplication with linkify-matrix
|
// FIXME: horrible duplication with linkify-matrix
|
||||||
m = attribs.href.match(linkifyMatrix.VECTOR_URL_PATTERN);
|
m = attribs.href.match(linkifyMatrix.VECTOR_URL_PATTERN);
|
||||||
if (m) {
|
if (m) {
|
||||||
attribs.href = m[1];
|
attribs.href = m[1];
|
||||||
delete attribs.target;
|
delete attribs.target;
|
||||||
}
|
} else {
|
||||||
|
m = attribs.href.match(linkifyMatrix.MATRIXTO_URL_PATTERN);
|
||||||
m = attribs.href.match(linkifyMatrix.MATRIXTO_URL_PATTERN);
|
if (m) {
|
||||||
if (m) {
|
const entity = m[1];
|
||||||
var entity = m[1];
|
if (entity[0] === '@') {
|
||||||
if (entity[0] === '@') {
|
attribs.href = '#/user/' + entity;
|
||||||
attribs.href = '#/user/' + entity;
|
} else if (entity[0] === '#' || entity[0] === '!') {
|
||||||
|
attribs.href = '#/room/' + entity;
|
||||||
|
}
|
||||||
|
delete attribs.target;
|
||||||
}
|
}
|
||||||
else if (entity[0] === '#' || entity[0] === '!') {
|
|
||||||
attribs.href = '#/room/' + entity;
|
|
||||||
}
|
|
||||||
delete attribs.target;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
attribs.rel = 'noopener'; // https://mathiasbynens.github.io/rel-noopener/
|
attribs.rel = 'noopener'; // https://mathiasbynens.github.io/rel-noopener/
|
||||||
return { tagName: tagName, attribs : attribs };
|
return { tagName: tagName, attribs: attribs };
|
||||||
|
},
|
||||||
|
'img': function(tagName, attribs) {
|
||||||
|
// Strip out imgs that aren't `mxc` here instead of using allowedSchemesByTag
|
||||||
|
// because transformTags is used _before_ we filter by allowedSchemesByTag and
|
||||||
|
// we don't want to allow images with `https?` `src`s.
|
||||||
|
if (!attribs.src || !attribs.src.startsWith('mxc://')) {
|
||||||
|
return { tagName, attribs: {}};
|
||||||
|
}
|
||||||
|
attribs.src = MatrixClientPeg.get().mxcUrlToHttp(
|
||||||
|
attribs.src,
|
||||||
|
attribs.width || 800,
|
||||||
|
attribs.height || 600,
|
||||||
|
);
|
||||||
|
return { tagName: tagName, attribs: attribs };
|
||||||
|
},
|
||||||
|
'code': function(tagName, attribs) {
|
||||||
|
if (typeof attribs.class !== 'undefined') {
|
||||||
|
// Filter out all classes other than ones starting with language- for syntax highlighting.
|
||||||
|
const classes = attribs.class.split(/\s+/).filter(function(cl) {
|
||||||
|
return cl.startsWith('language-');
|
||||||
|
});
|
||||||
|
attribs.class = classes.join(' ');
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
tagName: tagName,
|
||||||
|
attribs: attribs,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
'*': function(tagName, attribs) {
|
||||||
|
// Delete any style previously assigned, style is an allowedTag for font and span
|
||||||
|
// because attributes are stripped after transforming
|
||||||
|
delete attribs.style;
|
||||||
|
|
||||||
|
// Sanitise and transform data-mx-color and data-mx-bg-color to their CSS
|
||||||
|
// equivalents
|
||||||
|
const customCSSMapper = {
|
||||||
|
'data-mx-color': 'color',
|
||||||
|
'data-mx-bg-color': 'background-color',
|
||||||
|
// $customAttributeKey: $cssAttributeKey
|
||||||
|
};
|
||||||
|
|
||||||
|
let style = "";
|
||||||
|
Object.keys(customCSSMapper).forEach((customAttributeKey) => {
|
||||||
|
const cssAttributeKey = customCSSMapper[customAttributeKey];
|
||||||
|
const customAttributeValue = attribs[customAttributeKey];
|
||||||
|
if (customAttributeValue &&
|
||||||
|
typeof customAttributeValue === 'string' &&
|
||||||
|
COLOR_REGEX.test(customAttributeValue)
|
||||||
|
) {
|
||||||
|
style += cssAttributeKey + ":" + customAttributeValue + ";";
|
||||||
|
delete attribs[customAttributeKey];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (style) {
|
||||||
|
attribs.style = style;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { tagName: tagName, attribs: attribs };
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -157,11 +284,11 @@ class BaseHighlighter {
|
||||||
* TextHighlighter).
|
* TextHighlighter).
|
||||||
*/
|
*/
|
||||||
applyHighlights(safeSnippet, safeHighlights) {
|
applyHighlights(safeSnippet, safeHighlights) {
|
||||||
var lastOffset = 0;
|
let lastOffset = 0;
|
||||||
var offset;
|
let offset;
|
||||||
var nodes = [];
|
let nodes = [];
|
||||||
|
|
||||||
var safeHighlight = safeHighlights[0];
|
const safeHighlight = safeHighlights[0];
|
||||||
while ((offset = safeSnippet.toLowerCase().indexOf(safeHighlight.toLowerCase(), lastOffset)) >= 0) {
|
while ((offset = safeSnippet.toLowerCase().indexOf(safeHighlight.toLowerCase(), lastOffset)) >= 0) {
|
||||||
// handle preamble
|
// handle preamble
|
||||||
if (offset > lastOffset) {
|
if (offset > lastOffset) {
|
||||||
|
@ -171,7 +298,7 @@ class BaseHighlighter {
|
||||||
|
|
||||||
// do highlight. use the original string rather than safeHighlight
|
// do highlight. use the original string rather than safeHighlight
|
||||||
// to preserve the original casing.
|
// to preserve the original casing.
|
||||||
var endOffset = offset + safeHighlight.length;
|
const endOffset = offset + safeHighlight.length;
|
||||||
nodes.push(this._processSnippet(safeSnippet.substring(offset, endOffset), true));
|
nodes.push(this._processSnippet(safeSnippet.substring(offset, endOffset), true));
|
||||||
|
|
||||||
lastOffset = endOffset;
|
lastOffset = endOffset;
|
||||||
|
@ -189,8 +316,7 @@ class BaseHighlighter {
|
||||||
if (safeHighlights[1]) {
|
if (safeHighlights[1]) {
|
||||||
// recurse into this range to check for the next set of highlight matches
|
// recurse into this range to check for the next set of highlight matches
|
||||||
return this.applyHighlights(safeSnippet, safeHighlights.slice(1));
|
return this.applyHighlights(safeSnippet, safeHighlights.slice(1));
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
// no more highlights to be found, just return the unhighlighted string
|
// no more highlights to be found, just return the unhighlighted string
|
||||||
return [this._processSnippet(safeSnippet, false)];
|
return [this._processSnippet(safeSnippet, false)];
|
||||||
}
|
}
|
||||||
|
@ -211,7 +337,7 @@ class HtmlHighlighter extends BaseHighlighter {
|
||||||
return snippet;
|
return snippet;
|
||||||
}
|
}
|
||||||
|
|
||||||
var span = "<span class=\""+this.highlightClass+"\">"
|
let span = "<span class=\""+this.highlightClass+"\">"
|
||||||
+ snippet + "</span>";
|
+ snippet + "</span>";
|
||||||
|
|
||||||
if (this.highlightLink) {
|
if (this.highlightLink) {
|
||||||
|
@ -236,15 +362,15 @@ class TextHighlighter extends BaseHighlighter {
|
||||||
* returns a React node
|
* returns a React node
|
||||||
*/
|
*/
|
||||||
_processSnippet(snippet, highlight) {
|
_processSnippet(snippet, highlight) {
|
||||||
var key = this._key++;
|
const key = this._key++;
|
||||||
|
|
||||||
var node =
|
let node =
|
||||||
<span key={key} className={highlight ? this.highlightClass : null }>
|
<span key={key} className={highlight ? this.highlightClass : null}>
|
||||||
{ snippet }
|
{ snippet }
|
||||||
</span>;
|
</span>;
|
||||||
|
|
||||||
if (highlight && this.highlightLink) {
|
if (highlight && this.highlightLink) {
|
||||||
node = <a key={key} href={this.highlightLink}>{node}</a>;
|
node = <a key={key} href={this.highlightLink}>{ node }</a>;
|
||||||
}
|
}
|
||||||
|
|
||||||
return node;
|
return node;
|
||||||
|
@ -259,22 +385,23 @@ class TextHighlighter extends BaseHighlighter {
|
||||||
* highlights: optional list of words to highlight, ordered by longest word first
|
* highlights: optional list of words to highlight, ordered by longest word first
|
||||||
*
|
*
|
||||||
* opts.highlightLink: optional href to add to highlighted words
|
* opts.highlightLink: optional href to add to highlighted words
|
||||||
|
* opts.disableBigEmoji: optional argument to disable the big emoji class.
|
||||||
*/
|
*/
|
||||||
export function bodyToHtml(content, highlights, opts) {
|
export function bodyToHtml(content, highlights, opts={}) {
|
||||||
opts = opts || {};
|
const isHtml = (content.format === "org.matrix.custom.html");
|
||||||
|
const body = isHtml ? content.formatted_body : escape(content.body);
|
||||||
|
|
||||||
var isHtml = (content.format === "org.matrix.custom.html");
|
let bodyHasEmoji = false;
|
||||||
let body = isHtml ? content.formatted_body : escape(content.body);
|
|
||||||
|
|
||||||
var safeBody;
|
let safeBody;
|
||||||
// XXX: We sanitize the HTML whilst also highlighting its text nodes, to avoid accidentally trying
|
// XXX: We sanitize the HTML whilst also highlighting its text nodes, to avoid accidentally trying
|
||||||
// to highlight HTML tags themselves. However, this does mean that we don't highlight textnodes which
|
// to highlight HTML tags themselves. However, this does mean that we don't highlight textnodes which
|
||||||
// are interrupted by HTML tags (not that we did before) - e.g. foo<span/>bar won't get highlighted
|
// are interrupted by HTML tags (not that we did before) - e.g. foo<span/>bar won't get highlighted
|
||||||
// by an attempt to search for 'foobar'. Then again, the search query probably wouldn't work either
|
// by an attempt to search for 'foobar'. Then again, the search query probably wouldn't work either
|
||||||
try {
|
try {
|
||||||
if (highlights && highlights.length > 0) {
|
if (highlights && highlights.length > 0) {
|
||||||
var highlighter = new HtmlHighlighter("mx_EventTile_searchHighlight", opts.highlightLink);
|
const highlighter = new HtmlHighlighter("mx_EventTile_searchHighlight", opts.highlightLink);
|
||||||
var safeHighlights = highlights.map(function(highlight) {
|
const safeHighlights = highlights.map(function(highlight) {
|
||||||
return sanitizeHtml(highlight, sanitizeHtmlParams);
|
return sanitizeHtml(highlight, sanitizeHtmlParams);
|
||||||
});
|
});
|
||||||
// XXX: hacky bodge to temporarily apply a textFilter to the sanitizeHtmlParams structure.
|
// XXX: hacky bodge to temporarily apply a textFilter to the sanitizeHtmlParams structure.
|
||||||
|
@ -283,23 +410,26 @@ export function bodyToHtml(content, highlights, opts) {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
safeBody = sanitizeHtml(body, sanitizeHtmlParams);
|
safeBody = sanitizeHtml(body, sanitizeHtmlParams);
|
||||||
safeBody = unicodeToImage(safeBody);
|
bodyHasEmoji = containsEmoji(body);
|
||||||
}
|
if (bodyHasEmoji) safeBody = unicodeToImage(safeBody);
|
||||||
finally {
|
} finally {
|
||||||
delete sanitizeHtmlParams.textFilter;
|
delete sanitizeHtmlParams.textFilter;
|
||||||
}
|
}
|
||||||
|
|
||||||
EMOJI_REGEX.lastIndex = 0;
|
let emojiBody = false;
|
||||||
let contentBodyTrimmed = content.body.trim();
|
if (!opts.disableBigEmoji && bodyHasEmoji) {
|
||||||
let match = EMOJI_REGEX.exec(contentBodyTrimmed);
|
EMOJI_REGEX.lastIndex = 0;
|
||||||
let emojiBody = match && match[0] && match[0].length === contentBodyTrimmed.length;
|
const contentBodyTrimmed = content.body !== undefined ? content.body.trim() : '';
|
||||||
|
const match = EMOJI_REGEX.exec(contentBodyTrimmed);
|
||||||
|
emojiBody = match && match[0] && match[0].length === contentBodyTrimmed.length;
|
||||||
|
}
|
||||||
|
|
||||||
const className = classNames({
|
const className = classNames({
|
||||||
'mx_EventTile_body': true,
|
'mx_EventTile_body': true,
|
||||||
'mx_EventTile_bigEmoji': emojiBody,
|
'mx_EventTile_bigEmoji': emojiBody,
|
||||||
'markdown-body': isHtml,
|
'markdown-body': isHtml,
|
||||||
});
|
});
|
||||||
return <span className={className} dangerouslySetInnerHTML={{ __html: safeBody }} />;
|
return <span className={className} dangerouslySetInnerHTML={{ __html: safeBody }} dir="auto" />;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function emojifyText(text) {
|
export function emojifyText(text) {
|
||||||
|
|
|
@ -42,16 +42,15 @@ module.exports = {
|
||||||
// no scaling needs to be applied
|
// no scaling needs to be applied
|
||||||
return fullHeight;
|
return fullHeight;
|
||||||
}
|
}
|
||||||
var widthMulti = thumbWidth / fullWidth;
|
const widthMulti = thumbWidth / fullWidth;
|
||||||
var heightMulti = thumbHeight / fullHeight;
|
const heightMulti = thumbHeight / fullHeight;
|
||||||
if (widthMulti < heightMulti) {
|
if (widthMulti < heightMulti) {
|
||||||
// width is the dominant dimension so scaling will be fixed on that
|
// width is the dominant dimension so scaling will be fixed on that
|
||||||
return Math.floor(widthMulti * fullHeight);
|
return Math.floor(widthMulti * fullHeight);
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
// height is the dominant dimension so scaling will be fixed on that
|
// height is the dominant dimension so scaling will be fixed on that
|
||||||
return Math.floor(heightMulti * fullHeight);
|
return Math.floor(heightMulti * fullHeight);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,83 +0,0 @@
|
||||||
/*
|
|
||||||
Copyright 2016 OpenMarket Ltd
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import MatrixClientPeg from './MatrixClientPeg';
|
|
||||||
import MultiInviter from './utils/MultiInviter';
|
|
||||||
|
|
||||||
const emailRegex = /^\S+@\S+\.\S+$/;
|
|
||||||
|
|
||||||
export function getAddressType(inputText) {
|
|
||||||
const isEmailAddress = /^\S+@\S+\.\S+$/.test(inputText);
|
|
||||||
const isMatrixId = inputText[0] === '@' && inputText.indexOf(":") > 0;
|
|
||||||
|
|
||||||
// sanity check the input for user IDs
|
|
||||||
if (isEmailAddress) {
|
|
||||||
return 'email';
|
|
||||||
} else if (isMatrixId) {
|
|
||||||
return 'mx';
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function inviteToRoom(roomId, addr) {
|
|
||||||
const addrType = getAddressType(addr);
|
|
||||||
|
|
||||||
if (addrType == 'email') {
|
|
||||||
return MatrixClientPeg.get().inviteByEmail(roomId, addr);
|
|
||||||
} else if (addrType == 'mx') {
|
|
||||||
return MatrixClientPeg.get().invite(roomId, addr);
|
|
||||||
} else {
|
|
||||||
throw new Error('Unsupported address');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Invites multiple addresses to a room
|
|
||||||
* Simpler interface to utils/MultiInviter but with
|
|
||||||
* no option to cancel.
|
|
||||||
*
|
|
||||||
* @param {roomId} The ID of the room to invite to
|
|
||||||
* @param {array} Array of strings of addresses to invite. May be matrix IDs or 3pids.
|
|
||||||
* @returns Promise
|
|
||||||
*/
|
|
||||||
export function inviteMultipleToRoom(roomId, addrs) {
|
|
||||||
this.inviter = new MultiInviter(roomId);
|
|
||||||
return this.inviter.invite(addrs);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks is the supplied address is valid
|
|
||||||
*
|
|
||||||
* @param {addr} The mx userId or email address to check
|
|
||||||
* @returns true, false, or null for unsure
|
|
||||||
*/
|
|
||||||
export function isValidAddress(addr) {
|
|
||||||
// Check if the addr is a valid type
|
|
||||||
var addrType = this.getAddressType(addr);
|
|
||||||
if (addrType === "mx") {
|
|
||||||
let user = MatrixClientPeg.get().getUser(addr);
|
|
||||||
if (user) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
} else if (addrType === "email") {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
138
src/KeyRequestHandler.js
Normal file
138
src/KeyRequestHandler.js
Normal file
|
@ -0,0 +1,138 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Vector Creations 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 sdk from './index';
|
||||||
|
import Modal from './Modal';
|
||||||
|
|
||||||
|
export default class KeyRequestHandler {
|
||||||
|
constructor(matrixClient) {
|
||||||
|
this._matrixClient = matrixClient;
|
||||||
|
|
||||||
|
// the user/device for which we currently have a dialog open
|
||||||
|
this._currentUser = null;
|
||||||
|
this._currentDevice = null;
|
||||||
|
|
||||||
|
// userId -> deviceId -> [keyRequest]
|
||||||
|
this._pendingKeyRequests = Object.create(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleKeyRequest(keyRequest) {
|
||||||
|
const userId = keyRequest.userId;
|
||||||
|
const deviceId = keyRequest.deviceId;
|
||||||
|
const requestId = keyRequest.requestId;
|
||||||
|
|
||||||
|
if (!this._pendingKeyRequests[userId]) {
|
||||||
|
this._pendingKeyRequests[userId] = Object.create(null);
|
||||||
|
}
|
||||||
|
if (!this._pendingKeyRequests[userId][deviceId]) {
|
||||||
|
this._pendingKeyRequests[userId][deviceId] = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if we already have this request
|
||||||
|
const requests = this._pendingKeyRequests[userId][deviceId];
|
||||||
|
if (requests.find((r) => r.requestId === requestId)) {
|
||||||
|
console.log("Already have this key request, ignoring");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
requests.push(keyRequest);
|
||||||
|
|
||||||
|
if (this._currentUser) {
|
||||||
|
// ignore for now
|
||||||
|
console.log("Key request, but we already have a dialog open");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this._processNextRequest();
|
||||||
|
}
|
||||||
|
|
||||||
|
handleKeyRequestCancellation(cancellation) {
|
||||||
|
// see if we can find the request in the queue
|
||||||
|
const userId = cancellation.userId;
|
||||||
|
const deviceId = cancellation.deviceId;
|
||||||
|
const requestId = cancellation.requestId;
|
||||||
|
|
||||||
|
if (userId === this._currentUser && deviceId === this._currentDevice) {
|
||||||
|
console.log(
|
||||||
|
"room key request cancellation for the user we currently have a"
|
||||||
|
+ " dialog open for",
|
||||||
|
);
|
||||||
|
// TODO: update the dialog. For now, we just ignore the
|
||||||
|
// cancellation.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this._pendingKeyRequests[userId]) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const requests = this._pendingKeyRequests[userId][deviceId];
|
||||||
|
if (!requests) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const idx = requests.findIndex((r) => r.requestId === requestId);
|
||||||
|
if (idx < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.log("Forgetting room key request");
|
||||||
|
requests.splice(idx, 1);
|
||||||
|
if (requests.length === 0) {
|
||||||
|
delete this._pendingKeyRequests[userId][deviceId];
|
||||||
|
if (Object.keys(this._pendingKeyRequests[userId]).length === 0) {
|
||||||
|
delete this._pendingKeyRequests[userId];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_processNextRequest() {
|
||||||
|
const userId = Object.keys(this._pendingKeyRequests)[0];
|
||||||
|
if (!userId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const deviceId = Object.keys(this._pendingKeyRequests[userId])[0];
|
||||||
|
if (!deviceId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.log(`Starting KeyShareDialog for ${userId}:${deviceId}`);
|
||||||
|
|
||||||
|
const finished = (r) => {
|
||||||
|
this._currentUser = null;
|
||||||
|
this._currentDevice = null;
|
||||||
|
|
||||||
|
if (r) {
|
||||||
|
for (const req of this._pendingKeyRequests[userId][deviceId]) {
|
||||||
|
req.share();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete this._pendingKeyRequests[userId][deviceId];
|
||||||
|
if (Object.keys(this._pendingKeyRequests[userId]).length === 0) {
|
||||||
|
delete this._pendingKeyRequests[userId];
|
||||||
|
}
|
||||||
|
|
||||||
|
this._processNextRequest();
|
||||||
|
};
|
||||||
|
|
||||||
|
const KeyShareDialog = sdk.getComponent("dialogs.KeyShareDialog");
|
||||||
|
Modal.createTrackedDialog('Key Share', 'Process Next Request', KeyShareDialog, {
|
||||||
|
matrixClient: this._matrixClient,
|
||||||
|
userId: userId,
|
||||||
|
deviceId: deviceId,
|
||||||
|
onFinished: finished,
|
||||||
|
});
|
||||||
|
this._currentUser = userId;
|
||||||
|
this._currentDevice = deviceId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
79
src/Keyboard.js
Normal file
79
src/Keyboard.js
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* a selection of key codes, as used in KeyboardEvent.keyCode */
|
||||||
|
export const KeyCode = {
|
||||||
|
BACKSPACE: 8,
|
||||||
|
TAB: 9,
|
||||||
|
ENTER: 13,
|
||||||
|
SHIFT: 16,
|
||||||
|
ESCAPE: 27,
|
||||||
|
SPACE: 32,
|
||||||
|
PAGE_UP: 33,
|
||||||
|
PAGE_DOWN: 34,
|
||||||
|
END: 35,
|
||||||
|
HOME: 36,
|
||||||
|
LEFT: 37,
|
||||||
|
UP: 38,
|
||||||
|
RIGHT: 39,
|
||||||
|
DOWN: 40,
|
||||||
|
DELETE: 46,
|
||||||
|
KEY_A: 65,
|
||||||
|
KEY_B: 66,
|
||||||
|
KEY_C: 67,
|
||||||
|
KEY_D: 68,
|
||||||
|
KEY_E: 69,
|
||||||
|
KEY_F: 70,
|
||||||
|
KEY_G: 71,
|
||||||
|
KEY_H: 72,
|
||||||
|
KEY_I: 73,
|
||||||
|
KEY_J: 74,
|
||||||
|
KEY_K: 75,
|
||||||
|
KEY_L: 76,
|
||||||
|
KEY_M: 77,
|
||||||
|
KEY_N: 78,
|
||||||
|
KEY_O: 79,
|
||||||
|
KEY_P: 80,
|
||||||
|
KEY_Q: 81,
|
||||||
|
KEY_R: 82,
|
||||||
|
KEY_S: 83,
|
||||||
|
KEY_T: 84,
|
||||||
|
KEY_U: 85,
|
||||||
|
KEY_V: 86,
|
||||||
|
KEY_W: 87,
|
||||||
|
KEY_X: 88,
|
||||||
|
KEY_Y: 89,
|
||||||
|
KEY_Z: 90,
|
||||||
|
};
|
||||||
|
|
||||||
|
export function isOnlyCtrlOrCmdKeyEvent(ev) {
|
||||||
|
const isMac = navigator.platform.toUpperCase().indexOf('MAC') >= 0;
|
||||||
|
if (isMac) {
|
||||||
|
return ev.metaKey && !ev.altKey && !ev.ctrlKey && !ev.shiftKey;
|
||||||
|
} else {
|
||||||
|
return ev.ctrlKey && !ev.altKey && !ev.metaKey && !ev.shiftKey;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isOnlyCtrlOrCmdIgnoreShiftKeyEvent(ev) {
|
||||||
|
const isMac = navigator.platform.toUpperCase().indexOf('MAC') >= 0;
|
||||||
|
if (isMac) {
|
||||||
|
return ev.metaKey && !ev.altKey && !ev.ctrlKey;
|
||||||
|
} else {
|
||||||
|
return ev.ctrlKey && !ev.altKey && !ev.metaKey;
|
||||||
|
}
|
||||||
|
}
|
410
src/Lifecycle.js
410
src/Lifecycle.js
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -14,42 +15,38 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import q from 'q';
|
import Promise from 'bluebird';
|
||||||
import Matrix from 'matrix-js-sdk';
|
import Matrix from 'matrix-js-sdk';
|
||||||
|
|
||||||
import MatrixClientPeg from './MatrixClientPeg';
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
import Notifier from './Notifier'
|
import createMatrixClient from './utils/createMatrixClient';
|
||||||
|
import Analytics from './Analytics';
|
||||||
|
import Notifier from './Notifier';
|
||||||
import UserActivity from './UserActivity';
|
import UserActivity from './UserActivity';
|
||||||
import Presence from './Presence';
|
import Presence from './Presence';
|
||||||
import dis from './dispatcher';
|
import dis from './dispatcher';
|
||||||
import DMRoomMap from './utils/DMRoomMap';
|
import DMRoomMap from './utils/DMRoomMap';
|
||||||
|
import RtsClient from './RtsClient';
|
||||||
|
import Modal from './Modal';
|
||||||
|
import sdk from './index';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called at startup, to attempt to build a logged-in Matrix session. It tries
|
* Called at startup, to attempt to build a logged-in Matrix session. It tries
|
||||||
* a number of things:
|
* a number of things:
|
||||||
*
|
*
|
||||||
* 0. if it looks like we are in the middle of a registration process, it does
|
|
||||||
* nothing.
|
|
||||||
*
|
*
|
||||||
* 1. if we have a loginToken in the (real) query params, it uses that to log
|
* 1. if we have a guest access token in the fragment query params, it uses
|
||||||
* in.
|
|
||||||
*
|
|
||||||
* 2. if we have a guest access token in the fragment query params, it uses
|
|
||||||
* that.
|
* that.
|
||||||
*
|
*
|
||||||
* 3. if an access token is stored in local storage (from a previous session),
|
* 2. if an access token is stored in local storage (from a previous session),
|
||||||
* it uses that.
|
* it uses that.
|
||||||
*
|
*
|
||||||
* 4. it attempts to auto-register as a guest user.
|
* 3. it attempts to auto-register as a guest user.
|
||||||
*
|
*
|
||||||
* If any of steps 1-4 are successful, it will call {setLoggedIn}, which in
|
* If any of steps 1-4 are successful, it will call {_doSetLoggedIn}, which in
|
||||||
* turn will raise on_logged_in and will_start_client events.
|
* turn will raise on_logged_in and will_start_client events.
|
||||||
*
|
*
|
||||||
* It returns a promise which resolves when the above process completes.
|
* @param {object} opts
|
||||||
*
|
|
||||||
* @param {object} opts.realQueryParams: string->string map of the
|
|
||||||
* query-parameters extracted from the real query-string of the starting
|
|
||||||
* URI.
|
|
||||||
*
|
*
|
||||||
* @param {object} opts.fragmentQueryParams: string->string map of the
|
* @param {object} opts.fragmentQueryParams: string->string map of the
|
||||||
* query-parameters extracted from the #-fragment of the starting URI.
|
* query-parameters extracted from the #-fragment of the starting URI.
|
||||||
|
@ -63,66 +60,72 @@ import DMRoomMap from './utils/DMRoomMap';
|
||||||
* @params {string} opts.guestIsUrl: homeserver URL. Only used if enableGuest is
|
* @params {string} opts.guestIsUrl: homeserver URL. Only used if enableGuest is
|
||||||
* true; defines the IS to use.
|
* true; defines the IS to use.
|
||||||
*
|
*
|
||||||
|
* @returns {Promise} a promise which resolves when the above process completes.
|
||||||
|
* Resolves to `true` if we ended up starting a session, or `false` if we
|
||||||
|
* failed.
|
||||||
*/
|
*/
|
||||||
export function loadSession(opts) {
|
export function loadSession(opts) {
|
||||||
const realQueryParams = opts.realQueryParams || {};
|
|
||||||
const fragmentQueryParams = opts.fragmentQueryParams || {};
|
const fragmentQueryParams = opts.fragmentQueryParams || {};
|
||||||
let enableGuest = opts.enableGuest || false;
|
let enableGuest = opts.enableGuest || false;
|
||||||
const guestHsUrl = opts.guestHsUrl;
|
const guestHsUrl = opts.guestHsUrl;
|
||||||
const guestIsUrl = opts.guestIsUrl;
|
const guestIsUrl = opts.guestIsUrl;
|
||||||
const defaultDeviceDisplayName = opts.defaultDeviceDisplayName;
|
const defaultDeviceDisplayName = opts.defaultDeviceDisplayName;
|
||||||
|
|
||||||
if (fragmentQueryParams.client_secret && fragmentQueryParams.sid) {
|
|
||||||
// this happens during email validation: the email contains a link to the
|
|
||||||
// IS, which in turn redirects back to vector. We let MatrixChat create a
|
|
||||||
// Registration component which completes the next stage of registration.
|
|
||||||
console.log("Not registering as guest: registration already in progress.");
|
|
||||||
return q();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!guestHsUrl) {
|
if (!guestHsUrl) {
|
||||||
console.warn("Cannot enable guest access: can't determine HS URL to use");
|
console.warn("Cannot enable guest access: can't determine HS URL to use");
|
||||||
enableGuest = false;
|
enableGuest = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (realQueryParams.loginToken) {
|
|
||||||
if (!realQueryParams.homeserver) {
|
|
||||||
console.warn("Cannot log in with token: can't determine HS URL to use");
|
|
||||||
} else {
|
|
||||||
return _loginWithToken(realQueryParams, defaultDeviceDisplayName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (enableGuest &&
|
if (enableGuest &&
|
||||||
fragmentQueryParams.guest_user_id &&
|
fragmentQueryParams.guest_user_id &&
|
||||||
fragmentQueryParams.guest_access_token
|
fragmentQueryParams.guest_access_token
|
||||||
) {
|
) {
|
||||||
console.log("Using guest access credentials");
|
console.log("Using guest access credentials");
|
||||||
setLoggedIn({
|
return _doSetLoggedIn({
|
||||||
userId: fragmentQueryParams.guest_user_id,
|
userId: fragmentQueryParams.guest_user_id,
|
||||||
accessToken: fragmentQueryParams.guest_access_token,
|
accessToken: fragmentQueryParams.guest_access_token,
|
||||||
homeserverUrl: guestHsUrl,
|
homeserverUrl: guestHsUrl,
|
||||||
identityServerUrl: guestIsUrl,
|
identityServerUrl: guestIsUrl,
|
||||||
guest: true,
|
guest: true,
|
||||||
});
|
}, true).then(() => true);
|
||||||
return q();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_restoreFromLocalStorage()) {
|
return _restoreFromLocalStorage().then((success) => {
|
||||||
return q();
|
if (success) {
|
||||||
}
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (enableGuest) {
|
if (enableGuest) {
|
||||||
return _registerAsGuest(guestHsUrl, guestIsUrl, defaultDeviceDisplayName);
|
return _registerAsGuest(guestHsUrl, guestIsUrl, defaultDeviceDisplayName);
|
||||||
}
|
}
|
||||||
|
|
||||||
// fall back to login screen
|
// fall back to login screen
|
||||||
return q();
|
return false;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function _loginWithToken(queryParams, defaultDeviceDisplayName) {
|
/**
|
||||||
|
* @param {Object} queryParams string->string map of the
|
||||||
|
* query-parameters extracted from the real query-string of the starting
|
||||||
|
* URI.
|
||||||
|
*
|
||||||
|
* @param {String} defaultDeviceDisplayName
|
||||||
|
*
|
||||||
|
* @returns {Promise} promise which resolves to true if we completed the token
|
||||||
|
* login, else false
|
||||||
|
*/
|
||||||
|
export function attemptTokenLogin(queryParams, defaultDeviceDisplayName) {
|
||||||
|
if (!queryParams.loginToken) {
|
||||||
|
return Promise.resolve(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!queryParams.homeserver) {
|
||||||
|
console.warn("Cannot log in with token: can't determine HS URL to use");
|
||||||
|
return Promise.resolve(false);
|
||||||
|
}
|
||||||
|
|
||||||
// create a temporary MatrixClient to do the login
|
// create a temporary MatrixClient to do the login
|
||||||
var client = Matrix.createClient({
|
const client = Matrix.createClient({
|
||||||
baseUrl: queryParams.homeserver,
|
baseUrl: queryParams.homeserver,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -133,28 +136,32 @@ function _loginWithToken(queryParams, defaultDeviceDisplayName) {
|
||||||
},
|
},
|
||||||
).then(function(data) {
|
).then(function(data) {
|
||||||
console.log("Logged in with token");
|
console.log("Logged in with token");
|
||||||
setLoggedIn({
|
return _clearStorage().then(() => {
|
||||||
userId: data.user_id,
|
_persistCredentialsToLocalStorage({
|
||||||
deviceId: data.device_id,
|
userId: data.user_id,
|
||||||
accessToken: data.access_token,
|
deviceId: data.device_id,
|
||||||
homeserverUrl: queryParams.homeserver,
|
accessToken: data.access_token,
|
||||||
identityServerUrl: queryParams.identityServer,
|
homeserverUrl: queryParams.homeserver,
|
||||||
guest: false,
|
identityServerUrl: queryParams.identityServer,
|
||||||
})
|
guest: false,
|
||||||
}, (err) => {
|
});
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
}).catch((err) => {
|
||||||
console.error("Failed to log in with login token: " + err + " " +
|
console.error("Failed to log in with login token: " + err + " " +
|
||||||
err.data);
|
err.data);
|
||||||
|
return false;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function _registerAsGuest(hsUrl, isUrl, defaultDeviceDisplayName) {
|
function _registerAsGuest(hsUrl, isUrl, defaultDeviceDisplayName) {
|
||||||
console.log("Doing guest login on %s", hsUrl);
|
console.log(`Doing guest login on ${hsUrl}`);
|
||||||
|
|
||||||
// TODO: we should probably de-duplicate this and Signup.Login.loginAsGuest.
|
// TODO: we should probably de-duplicate this and Login.loginAsGuest.
|
||||||
// Not really sure where the right home for it is.
|
// Not really sure where the right home for it is.
|
||||||
|
|
||||||
// create a temporary MatrixClient to do the login
|
// create a temporary MatrixClient to do the login
|
||||||
var client = Matrix.createClient({
|
const client = Matrix.createClient({
|
||||||
baseUrl: hsUrl,
|
baseUrl: hsUrl,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -163,119 +170,227 @@ function _registerAsGuest(hsUrl, isUrl, defaultDeviceDisplayName) {
|
||||||
initial_device_display_name: defaultDeviceDisplayName,
|
initial_device_display_name: defaultDeviceDisplayName,
|
||||||
},
|
},
|
||||||
}).then((creds) => {
|
}).then((creds) => {
|
||||||
console.log("Registered as guest: %s", creds.user_id);
|
console.log(`Registered as guest: ${creds.user_id}`);
|
||||||
setLoggedIn({
|
return _doSetLoggedIn({
|
||||||
userId: creds.user_id,
|
userId: creds.user_id,
|
||||||
deviceId: creds.device_id,
|
deviceId: creds.device_id,
|
||||||
accessToken: creds.access_token,
|
accessToken: creds.access_token,
|
||||||
homeserverUrl: hsUrl,
|
homeserverUrl: hsUrl,
|
||||||
identityServerUrl: isUrl,
|
identityServerUrl: isUrl,
|
||||||
guest: true,
|
guest: true,
|
||||||
});
|
}, true).then(() => true);
|
||||||
}, (err) => {
|
}, (err) => {
|
||||||
console.error("Failed to register as guest: " + err + " " + err.data);
|
console.error("Failed to register as guest: " + err + " " + err.data);
|
||||||
|
return false;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns true if a session is found in localstorage
|
// returns a promise which resolves to true if a session is found in
|
||||||
|
// localstorage
|
||||||
|
//
|
||||||
|
// N.B. Lifecycle.js should not maintain any further localStorage state, we
|
||||||
|
// are moving towards using SessionStore to keep track of state related
|
||||||
|
// to the current session (which is typically backed by localStorage).
|
||||||
|
//
|
||||||
|
// The plan is to gradually move the localStorage access done here into
|
||||||
|
// SessionStore to avoid bugs where the view becomes out-of-sync with
|
||||||
|
// localStorage (e.g. teamToken, isGuest etc.)
|
||||||
function _restoreFromLocalStorage() {
|
function _restoreFromLocalStorage() {
|
||||||
if (!localStorage) {
|
if (!localStorage) {
|
||||||
return false;
|
return Promise.resolve(false);
|
||||||
}
|
}
|
||||||
const hs_url = localStorage.getItem("mx_hs_url");
|
const hsUrl = localStorage.getItem("mx_hs_url");
|
||||||
const is_url = localStorage.getItem("mx_is_url") || 'https://matrix.org';
|
const isUrl = localStorage.getItem("mx_is_url") || 'https://matrix.org';
|
||||||
const access_token = localStorage.getItem("mx_access_token");
|
const accessToken = localStorage.getItem("mx_access_token");
|
||||||
const user_id = localStorage.getItem("mx_user_id");
|
const userId = localStorage.getItem("mx_user_id");
|
||||||
const device_id = localStorage.getItem("mx_device_id");
|
const deviceId = localStorage.getItem("mx_device_id");
|
||||||
|
|
||||||
let is_guest;
|
let isGuest;
|
||||||
if (localStorage.getItem("mx_is_guest") !== null) {
|
if (localStorage.getItem("mx_is_guest") !== null) {
|
||||||
is_guest = localStorage.getItem("mx_is_guest") === "true";
|
isGuest = localStorage.getItem("mx_is_guest") === "true";
|
||||||
} else {
|
} else {
|
||||||
// legacy key name
|
// legacy key name
|
||||||
is_guest = localStorage.getItem("matrix-is-guest") === "true";
|
isGuest = localStorage.getItem("matrix-is-guest") === "true";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (access_token && user_id && hs_url) {
|
if (accessToken && userId && hsUrl) {
|
||||||
console.log("Restoring session for %s", user_id);
|
console.log(`Restoring session for ${userId}`);
|
||||||
try {
|
try {
|
||||||
setLoggedIn({
|
return _doSetLoggedIn({
|
||||||
userId: user_id,
|
userId: userId,
|
||||||
deviceId: device_id,
|
deviceId: deviceId,
|
||||||
accessToken: access_token,
|
accessToken: accessToken,
|
||||||
homeserverUrl: hs_url,
|
homeserverUrl: hsUrl,
|
||||||
identityServerUrl: is_url,
|
identityServerUrl: isUrl,
|
||||||
guest: is_guest,
|
guest: isGuest,
|
||||||
});
|
}, false).then(() => true);
|
||||||
return true;
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.log("Unable to restore session", e);
|
return _handleRestoreFailure(e);
|
||||||
|
|
||||||
var msg = e.message;
|
|
||||||
if (msg == "OLM.BAD_LEGACY_ACCOUNT_PICKLE") {
|
|
||||||
msg = "You need to log back in to generate end-to-end encryption keys "
|
|
||||||
+ "for this device and submit the public key to your homeserver. "
|
|
||||||
+ "This is a once off; sorry for the inconvenience.";
|
|
||||||
}
|
|
||||||
|
|
||||||
// don't leak things into the new session
|
|
||||||
_clearLocalStorage();
|
|
||||||
|
|
||||||
throw new Error("Unable to restore previous session: " + msg);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
console.log("No previous session found.");
|
console.log("No previous session found.");
|
||||||
return false;
|
return Promise.resolve(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function _handleRestoreFailure(e) {
|
||||||
|
console.log("Unable to restore session", e);
|
||||||
|
|
||||||
|
const def = Promise.defer();
|
||||||
|
const SessionRestoreErrorDialog =
|
||||||
|
sdk.getComponent('views.dialogs.SessionRestoreErrorDialog');
|
||||||
|
|
||||||
|
Modal.createTrackedDialog('Session Restore Error', '', SessionRestoreErrorDialog, {
|
||||||
|
error: e.message,
|
||||||
|
onFinished: (success) => {
|
||||||
|
def.resolve(success);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return def.promise.then((success) => {
|
||||||
|
if (success) {
|
||||||
|
// user clicked continue.
|
||||||
|
_clearStorage();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// try, try again
|
||||||
|
return _restoreFromLocalStorage();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let rtsClient = null;
|
||||||
|
export function initRtsClient(url) {
|
||||||
|
if (url) {
|
||||||
|
rtsClient = new RtsClient(url);
|
||||||
|
} else {
|
||||||
|
rtsClient = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Transitions to a logged-in state using the given credentials
|
* Transitions to a logged-in state using the given credentials.
|
||||||
|
*
|
||||||
|
* Starts the matrix client and all other react-sdk services that
|
||||||
|
* listen for events while a session is logged in.
|
||||||
|
*
|
||||||
|
* Also stops the old MatrixClient and clears old credentials/etc out of
|
||||||
|
* storage before starting the new client.
|
||||||
|
*
|
||||||
* @param {MatrixClientCreds} credentials The credentials to use
|
* @param {MatrixClientCreds} credentials The credentials to use
|
||||||
|
*
|
||||||
|
* @returns {Promise} promise which resolves to the new MatrixClient once it has been started
|
||||||
*/
|
*/
|
||||||
export function setLoggedIn(credentials) {
|
export function setLoggedIn(credentials) {
|
||||||
credentials.guest = Boolean(credentials.guest);
|
stopMatrixClient();
|
||||||
console.log("setLoggedIn => %s (guest=%s) hs=%s",
|
return _doSetLoggedIn(credentials, true);
|
||||||
credentials.userId, credentials.guest,
|
}
|
||||||
credentials.homeserverUrl);
|
|
||||||
|
/**
|
||||||
|
* fires on_logging_in, optionally clears localstorage, persists new credentials
|
||||||
|
* to localstorage, starts the new client.
|
||||||
|
*
|
||||||
|
* @param {MatrixClientCreds} credentials
|
||||||
|
* @param {Boolean} clearStorage
|
||||||
|
*
|
||||||
|
* @returns {Promise} promise which resolves to the new MatrixClient once it has been started
|
||||||
|
*/
|
||||||
|
async function _doSetLoggedIn(credentials, clearStorage) {
|
||||||
|
credentials.guest = Boolean(credentials.guest);
|
||||||
|
|
||||||
|
console.log(
|
||||||
|
"setLoggedIn: mxid: " + credentials.userId +
|
||||||
|
" deviceId: " + credentials.deviceId +
|
||||||
|
" guest: " + credentials.guest +
|
||||||
|
" hs: " + credentials.homeserverUrl,
|
||||||
|
);
|
||||||
|
|
||||||
|
// This is dispatched to indicate that the user is still in the process of logging in
|
||||||
|
// because `teamPromise` may take some time to resolve, breaking the assumption that
|
||||||
|
// `setLoggedIn` takes an "instant" to complete, and dispatch `on_logged_in` a few ms
|
||||||
|
// later than MatrixChat might assume.
|
||||||
|
//
|
||||||
|
// we fire it *synchronously* to make sure it fires before on_logged_in.
|
||||||
|
// (dis.dispatch uses `setTimeout`, which does not guarantee ordering.)
|
||||||
|
dis.dispatch({action: 'on_logging_in'}, true);
|
||||||
|
|
||||||
|
if (clearStorage) {
|
||||||
|
await _clearStorage();
|
||||||
|
}
|
||||||
|
|
||||||
|
Analytics.setLoggedIn(credentials.guest, credentials.homeserverUrl, credentials.identityServerUrl);
|
||||||
|
|
||||||
|
// Resolves by default
|
||||||
|
let teamPromise = Promise.resolve(null);
|
||||||
|
|
||||||
|
|
||||||
// persist the session
|
|
||||||
if (localStorage) {
|
if (localStorage) {
|
||||||
try {
|
try {
|
||||||
localStorage.setItem("mx_hs_url", credentials.homeserverUrl);
|
_persistCredentialsToLocalStorage(credentials);
|
||||||
localStorage.setItem("mx_is_url", 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));
|
|
||||||
|
|
||||||
// if we didn't get a deviceId from the login, leave mx_device_id unset,
|
// The user registered as a PWLU (PassWord-Less User), the generated password
|
||||||
// rather than setting it to "undefined".
|
// is cached here such that the user can change it at a later time.
|
||||||
//
|
if (credentials.password) {
|
||||||
// (in this case MatrixClient doesn't bother with the crypto stuff
|
// Update SessionStore
|
||||||
// - that's fine for us).
|
dis.dispatch({
|
||||||
if (credentials.deviceId) {
|
action: 'cached_password',
|
||||||
localStorage.setItem("mx_device_id", credentials.deviceId);
|
cachedPassword: credentials.password,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log("Session persisted for %s", credentials.userId);
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.warn("Error using local storage: can't persist session!", e);
|
console.warn("Error using local storage: can't persist session!", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (rtsClient && !credentials.guest) {
|
||||||
|
teamPromise = rtsClient.login(credentials.userId).then((body) => {
|
||||||
|
if (body.team_token) {
|
||||||
|
localStorage.setItem("mx_team_token", body.team_token);
|
||||||
|
}
|
||||||
|
return body.team_token;
|
||||||
|
}, (err) => {
|
||||||
|
console.warn(`Failed to get team token on login: ${err}` );
|
||||||
|
return null;
|
||||||
|
});
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
console.warn("No local storage available: can't persist session!");
|
console.warn("No local storage available: can't persist session!");
|
||||||
}
|
}
|
||||||
|
|
||||||
MatrixClientPeg.replaceUsingCreds(credentials);
|
MatrixClientPeg.replaceUsingCreds(credentials);
|
||||||
|
|
||||||
dis.dispatch({action: 'on_logged_in'});
|
teamPromise.then((teamToken) => {
|
||||||
|
dis.dispatch({action: 'on_logged_in', teamToken: teamToken});
|
||||||
|
});
|
||||||
|
|
||||||
startMatrixClient();
|
startMatrixClient();
|
||||||
|
return MatrixClientPeg.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
function _persistCredentialsToLocalStorage(credentials) {
|
||||||
|
localStorage.setItem("mx_hs_url", credentials.homeserverUrl);
|
||||||
|
localStorage.setItem("mx_is_url", 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));
|
||||||
|
|
||||||
|
// if we didn't get a deviceId from the login, leave mx_device_id unset,
|
||||||
|
// rather than setting it to "undefined".
|
||||||
|
//
|
||||||
|
// (in this case MatrixClient doesn't bother with the crypto stuff
|
||||||
|
// - that's fine for us).
|
||||||
|
if (credentials.deviceId) {
|
||||||
|
localStorage.setItem("mx_device_id", credentials.deviceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`Session persisted for ${credentials.userId}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Logs the current session out and transitions to the logged-out state
|
* Logs the current session out and transitions to the logged-out state
|
||||||
*/
|
*/
|
||||||
export function logout() {
|
export function logout() {
|
||||||
|
if (!MatrixClientPeg.get()) return;
|
||||||
|
|
||||||
if (MatrixClientPeg.get().isGuest()) {
|
if (MatrixClientPeg.get().isGuest()) {
|
||||||
// logout doesn't work for guest sessions
|
// logout doesn't work for guest sessions
|
||||||
// Also we sometimes want to re-log in a guest session
|
// Also we sometimes want to re-log in a guest session
|
||||||
|
@ -289,7 +404,7 @@ export function logout() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
return MatrixClientPeg.get().logout().then(onLoggedOut,
|
MatrixClientPeg.get().logout().then(onLoggedOut,
|
||||||
(err) => {
|
(err) => {
|
||||||
// Just throwing an error here is going to be very unhelpful
|
// Just throwing an error here is going to be very unhelpful
|
||||||
// if you're trying to log out because your server's down and
|
// if you're trying to log out because your server's down and
|
||||||
|
@ -300,15 +415,17 @@ export function logout() {
|
||||||
// change your password).
|
// change your password).
|
||||||
console.log("Failed to call logout API: token will not be invalidated");
|
console.log("Failed to call logout API: token will not be invalidated");
|
||||||
onLoggedOut();
|
onLoggedOut();
|
||||||
}
|
},
|
||||||
);
|
).done();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Starts the matrix client and all other react-sdk services that
|
* Starts the matrix client and all other react-sdk services that
|
||||||
* listen for events while a session is logged in.
|
* listen for events while a session is logged in.
|
||||||
*/
|
*/
|
||||||
export function startMatrixClient() {
|
function startMatrixClient() {
|
||||||
|
console.log(`Lifecycle: Starting MatrixClient`);
|
||||||
|
|
||||||
// dispatch this before starting the matrix client: it's used
|
// dispatch this before starting the matrix client: it's used
|
||||||
// to add listeners for the 'sync' event so otherwise we'd have
|
// to add listeners for the 'sync' event so otherwise we'd have
|
||||||
// a race condition (and we need to dispatch synchronously for this
|
// a race condition (and we need to dispatch synchronously for this
|
||||||
|
@ -321,43 +438,58 @@ export function startMatrixClient() {
|
||||||
DMRoomMap.makeShared().start();
|
DMRoomMap.makeShared().start();
|
||||||
|
|
||||||
MatrixClientPeg.start();
|
MatrixClientPeg.start();
|
||||||
|
|
||||||
|
// dispatch that we finished starting up to wire up any other bits
|
||||||
|
// of the matrix client that cannot be set prior to starting up.
|
||||||
|
dis.dispatch({action: 'client_started'});
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Stops a running client and all related services, used after
|
* Stops a running client and all related services, and clears persistent
|
||||||
* a session has been logged out / ended.
|
* storage. Used after a session has been logged out.
|
||||||
*/
|
*/
|
||||||
export function onLoggedOut() {
|
export function onLoggedOut() {
|
||||||
_clearLocalStorage();
|
|
||||||
stopMatrixClient();
|
stopMatrixClient();
|
||||||
|
_clearStorage().done();
|
||||||
dis.dispatch({action: 'on_logged_out'});
|
dis.dispatch({action: 'on_logged_out'});
|
||||||
}
|
}
|
||||||
|
|
||||||
function _clearLocalStorage() {
|
/**
|
||||||
if (!window.localStorage) {
|
* @returns {Promise} promise which resolves once the stores have been cleared
|
||||||
return;
|
*/
|
||||||
}
|
function _clearStorage() {
|
||||||
const hsUrl = window.localStorage.getItem("mx_hs_url");
|
Analytics.logout();
|
||||||
const isUrl = window.localStorage.getItem("mx_is_url");
|
|
||||||
window.localStorage.clear();
|
|
||||||
|
|
||||||
// preserve our HS & IS URLs for convenience
|
if (window.localStorage) {
|
||||||
// N.B. we cache them in hsUrl/isUrl and can't really inline them
|
const hsUrl = window.localStorage.getItem("mx_hs_url");
|
||||||
// as getCurrentHsUrl() may call through to localStorage.
|
const isUrl = window.localStorage.getItem("mx_is_url");
|
||||||
// NB. We do clear the device ID (as well as all the settings)
|
window.localStorage.clear();
|
||||||
if (hsUrl) window.localStorage.setItem("mx_hs_url", hsUrl);
|
|
||||||
if (isUrl) window.localStorage.setItem("mx_is_url", isUrl);
|
// preserve our HS & IS URLs for convenience
|
||||||
|
// N.B. we cache them in hsUrl/isUrl and can't really inline them
|
||||||
|
// as getCurrentHsUrl() may call through to localStorage.
|
||||||
|
// NB. We do clear the device ID (as well as all the settings)
|
||||||
|
if (hsUrl) window.localStorage.setItem("mx_hs_url", hsUrl);
|
||||||
|
if (isUrl) window.localStorage.setItem("mx_is_url", isUrl);
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a temporary client to clear out the persistent stores.
|
||||||
|
const cli = createMatrixClient({
|
||||||
|
// we'll never make any requests, so can pass a bogus HS URL
|
||||||
|
baseUrl: "",
|
||||||
|
});
|
||||||
|
return cli.clearStores();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stop all the background processes related to the current client
|
* Stop all the background processes related to the current client.
|
||||||
*/
|
*/
|
||||||
export function stopMatrixClient() {
|
export function stopMatrixClient() {
|
||||||
Notifier.stop();
|
Notifier.stop();
|
||||||
UserActivity.stop();
|
UserActivity.stop();
|
||||||
Presence.stop();
|
Presence.stop();
|
||||||
if (DMRoomMap.shared()) DMRoomMap.shared().stop();
|
if (DMRoomMap.shared()) DMRoomMap.shared().stop();
|
||||||
var cli = MatrixClientPeg.get();
|
const cli = MatrixClientPeg.get();
|
||||||
if (cli) {
|
if (cli) {
|
||||||
cli.stopClient();
|
cli.stopClient();
|
||||||
cli.removeAllListeners();
|
cli.removeAllListeners();
|
||||||
|
|
243
src/Login.js
Normal file
243
src/Login.js
Normal file
|
@ -0,0 +1,243 @@
|
||||||
|
/*
|
||||||
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations 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 Matrix from "matrix-js-sdk";
|
||||||
|
import { _t } from "./languageHandler";
|
||||||
|
|
||||||
|
import Promise from 'bluebird';
|
||||||
|
import url from 'url';
|
||||||
|
|
||||||
|
export default class Login {
|
||||||
|
constructor(hsUrl, isUrl, fallbackHsUrl, opts) {
|
||||||
|
this._hsUrl = hsUrl;
|
||||||
|
this._isUrl = isUrl;
|
||||||
|
this._fallbackHsUrl = fallbackHsUrl;
|
||||||
|
this._currentFlowIndex = 0;
|
||||||
|
this._flows = [];
|
||||||
|
this._defaultDeviceDisplayName = opts.defaultDeviceDisplayName;
|
||||||
|
}
|
||||||
|
|
||||||
|
getHomeserverUrl() {
|
||||||
|
return this._hsUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
getIdentityServerUrl() {
|
||||||
|
return this._isUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
setHomeserverUrl(hsUrl) {
|
||||||
|
this._hsUrl = hsUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
setIdentityServerUrl(isUrl) {
|
||||||
|
this._isUrl = isUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a temporary MatrixClient, which can be used for login or register
|
||||||
|
* requests.
|
||||||
|
*/
|
||||||
|
_createTemporaryClient() {
|
||||||
|
return Matrix.createClient({
|
||||||
|
baseUrl: this._hsUrl,
|
||||||
|
idBaseUrl: this._isUrl,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
getFlows() {
|
||||||
|
const self = this;
|
||||||
|
const client = this._createTemporaryClient();
|
||||||
|
return client.loginFlows().then(function(result) {
|
||||||
|
self._flows = result.flows;
|
||||||
|
self._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 self._flows;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
chooseFlow(flowIndex) {
|
||||||
|
this._currentFlowIndex = flowIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
getCurrentFlowStep() {
|
||||||
|
// 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
loginAsGuest() {
|
||||||
|
const client = this._createTemporaryClient();
|
||||||
|
return client.registerGuest({
|
||||||
|
body: {
|
||||||
|
initial_device_display_name: this._defaultDeviceDisplayName,
|
||||||
|
},
|
||||||
|
}).then((creds) => {
|
||||||
|
return {
|
||||||
|
userId: creds.user_id,
|
||||||
|
deviceId: creds.device_id,
|
||||||
|
accessToken: creds.access_token,
|
||||||
|
homeserverUrl: this._hsUrl,
|
||||||
|
identityServerUrl: this._isUrl,
|
||||||
|
guest: true,
|
||||||
|
};
|
||||||
|
}, (error) => {
|
||||||
|
throw error;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
loginViaPassword(username, phoneCountry, phoneNumber, pass) {
|
||||||
|
const self = this;
|
||||||
|
|
||||||
|
const isEmail = username.indexOf("@") > 0;
|
||||||
|
|
||||||
|
let identifier;
|
||||||
|
let legacyParams; // parameters added to support old HSes
|
||||||
|
if (phoneCountry && phoneNumber) {
|
||||||
|
identifier = {
|
||||||
|
type: 'm.id.phone',
|
||||||
|
country: phoneCountry,
|
||||||
|
number: phoneNumber,
|
||||||
|
};
|
||||||
|
// No legacy support for phone number login
|
||||||
|
} else if (isEmail) {
|
||||||
|
identifier = {
|
||||||
|
type: 'm.id.thirdparty',
|
||||||
|
medium: 'email',
|
||||||
|
address: username,
|
||||||
|
};
|
||||||
|
legacyParams = {
|
||||||
|
medium: 'email',
|
||||||
|
address: username,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
identifier = {
|
||||||
|
type: 'm.id.user',
|
||||||
|
user: username,
|
||||||
|
};
|
||||||
|
legacyParams = {
|
||||||
|
user: username,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const loginParams = {
|
||||||
|
password: pass,
|
||||||
|
identifier: identifier,
|
||||||
|
initial_device_display_name: this._defaultDeviceDisplayName,
|
||||||
|
};
|
||||||
|
Object.assign(loginParams, legacyParams);
|
||||||
|
|
||||||
|
const client = this._createTemporaryClient();
|
||||||
|
|
||||||
|
const tryFallbackHs = (originalError) => {
|
||||||
|
const fbClient = Matrix.createClient({
|
||||||
|
baseUrl: self._fallbackHsUrl,
|
||||||
|
idBaseUrl: this._isUrl,
|
||||||
|
});
|
||||||
|
|
||||||
|
return fbClient.login('m.login.password', loginParams).then(function(data) {
|
||||||
|
return Promise.resolve({
|
||||||
|
homeserverUrl: self._fallbackHsUrl,
|
||||||
|
identityServerUrl: self._isUrl,
|
||||||
|
userId: data.user_id,
|
||||||
|
deviceId: data.device_id,
|
||||||
|
accessToken: data.access_token,
|
||||||
|
});
|
||||||
|
}).catch((fallback_error) => {
|
||||||
|
console.log("fallback HS login failed", fallback_error);
|
||||||
|
// throw the original error
|
||||||
|
throw originalError;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
const tryLowercaseUsername = (originalError) => {
|
||||||
|
const loginParamsLowercase = Object.assign({}, loginParams, {
|
||||||
|
user: username.toLowerCase(),
|
||||||
|
identifier: {
|
||||||
|
user: username.toLowerCase(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return client.login('m.login.password', loginParamsLowercase).then(function(data) {
|
||||||
|
return Promise.resolve({
|
||||||
|
homeserverUrl: self._hsUrl,
|
||||||
|
identityServerUrl: self._isUrl,
|
||||||
|
userId: data.user_id,
|
||||||
|
deviceId: data.device_id,
|
||||||
|
accessToken: data.access_token,
|
||||||
|
});
|
||||||
|
}).catch((fallback_error) => {
|
||||||
|
console.log("Lowercase username login failed", fallback_error);
|
||||||
|
// throw the original error
|
||||||
|
throw originalError;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
let originalLoginError = null;
|
||||||
|
return client.login('m.login.password', loginParams).then(function(data) {
|
||||||
|
return Promise.resolve({
|
||||||
|
homeserverUrl: self._hsUrl,
|
||||||
|
identityServerUrl: self._isUrl,
|
||||||
|
userId: data.user_id,
|
||||||
|
deviceId: data.device_id,
|
||||||
|
accessToken: data.access_token,
|
||||||
|
});
|
||||||
|
}).catch((error) => {
|
||||||
|
originalLoginError = error;
|
||||||
|
if (error.httpStatus === 403) {
|
||||||
|
if (self._fallbackHsUrl) {
|
||||||
|
return tryFallbackHs(originalLoginError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw originalLoginError;
|
||||||
|
}).catch((error) => {
|
||||||
|
// We apparently squash case at login serverside these days:
|
||||||
|
// https://github.com/matrix-org/synapse/blob/1189be43a2479f5adf034613e8d10e3f4f452eb9/synapse/handlers/auth.py#L475
|
||||||
|
// so this wasn't needed after all. Keeping the code around in case the
|
||||||
|
// the situation changes...
|
||||||
|
|
||||||
|
/*
|
||||||
|
if (
|
||||||
|
error.httpStatus === 403 &&
|
||||||
|
loginParams.identifier.type === 'm.id.user' &&
|
||||||
|
username.search(/[A-Z]/) > -1
|
||||||
|
) {
|
||||||
|
return tryLowercaseUsername(originalLoginError);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
throw originalLoginError;
|
||||||
|
}).catch((error) => {
|
||||||
|
console.log("Login failed", error);
|
||||||
|
throw error;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
redirectToCas() {
|
||||||
|
const client = this._createTemporaryClient();
|
||||||
|
const parsedUrl = url.parse(window.location.href, true);
|
||||||
|
|
||||||
|
// XXX: at this point, the fragment will always be #/login, which is no
|
||||||
|
// use to anyone. Ideally, we would get the intended fragment from
|
||||||
|
// MatrixChat.screenAfterLogin so that you could follow #/room links etc
|
||||||
|
// through a CAS login.
|
||||||
|
parsedUrl.hash = "";
|
||||||
|
|
||||||
|
parsedUrl.query["homeserver"] = client.getHomeserverUrl();
|
||||||
|
parsedUrl.query["identityServer"] = client.getIdentityServerUrl();
|
||||||
|
const casUrl = client.getCasLoginUrl(url.format(parsedUrl));
|
||||||
|
window.location.href = casUrl;
|
||||||
|
}
|
||||||
|
}
|
218
src/Markdown.js
218
src/Markdown.js
|
@ -14,115 +14,153 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import marked from 'marked';
|
import commonmark from 'commonmark';
|
||||||
|
import escape from 'lodash/escape';
|
||||||
|
|
||||||
// marked only applies the default options on the high
|
const ALLOWED_HTML_TAGS = ['sub', 'sup', 'del', 'u'];
|
||||||
// level marked() interface, so we do it here.
|
|
||||||
const marked_options = Object.assign({}, marked.defaults, {
|
// These types of node are definitely text
|
||||||
gfm: true,
|
const TEXT_NODES = ['text', 'softbreak', 'linebreak', 'paragraph', 'document'];
|
||||||
tables: true,
|
|
||||||
breaks: true,
|
function is_allowed_html_tag(node) {
|
||||||
pedantic: false,
|
// Regex won't work for tags with attrs, but we only
|
||||||
sanitize: true,
|
// allow <del> anyway.
|
||||||
smartLists: true,
|
const matches = /^<\/?(.*)>$/.exec(node.literal);
|
||||||
smartypants: false,
|
if (matches && matches.length == 2) {
|
||||||
xhtml: true, // return self closing tags (ie. <br /> not <br>)
|
const tag = matches[1];
|
||||||
});
|
return ALLOWED_HTML_TAGS.indexOf(tag) > -1;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function html_if_tag_allowed(node) {
|
||||||
|
if (is_allowed_html_tag(node)) {
|
||||||
|
this.lit(node.literal);
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
this.lit(escape(node.literal));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Returns true if the parse output containing the node
|
||||||
|
* comprises multiple block level elements (ie. lines),
|
||||||
|
* or false if it is only a single line.
|
||||||
|
*/
|
||||||
|
function is_multi_line(node) {
|
||||||
|
let par = node;
|
||||||
|
while (par.parent) {
|
||||||
|
par = par.parent;
|
||||||
|
}
|
||||||
|
return par.firstChild != par.lastChild;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class that wraps marked, adding the ability to see whether
|
* Class that wraps commonmark, adding the ability to see whether
|
||||||
* a given message actually uses any markdown syntax or whether
|
* a given message actually uses any markdown syntax or whether
|
||||||
* it's plain text.
|
* it's plain text.
|
||||||
*/
|
*/
|
||||||
export default class Markdown {
|
export default class Markdown {
|
||||||
constructor(input) {
|
constructor(input) {
|
||||||
const lexer = new marked.Lexer(marked_options);
|
this.input = input;
|
||||||
this.tokens = lexer.lex(input);
|
|
||||||
}
|
|
||||||
|
|
||||||
_copyTokens() {
|
const parser = new commonmark.Parser();
|
||||||
// copy tokens (the parser modifies its input arg)
|
this.parsed = parser.parse(this.input);
|
||||||
const tokens_copy = this.tokens.slice();
|
|
||||||
// it also has a 'links' property, because this is javascript
|
|
||||||
// and why wouldn't you have an array that also has properties?
|
|
||||||
return Object.assign(tokens_copy, this.tokens);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
isPlainText() {
|
isPlainText() {
|
||||||
// we determine if the message requires markdown by
|
const walker = this.parsed.walker();
|
||||||
// running the parser on the tokens with a dummy
|
|
||||||
// rendered and seeing if any of the renderer's
|
|
||||||
// functions are called other than those noted below.
|
|
||||||
// In case you were wondering, no we can't just examine
|
|
||||||
// the tokens because the tokens we have are only the
|
|
||||||
// output of the *first* tokenizer: any line-based
|
|
||||||
// markdown is processed by marked within Parser by
|
|
||||||
// the 'inline lexer'...
|
|
||||||
let is_plain = true;
|
|
||||||
|
|
||||||
function setNotPlain() {
|
let ev;
|
||||||
is_plain = false;
|
while ( (ev = walker.next()) ) {
|
||||||
}
|
const node = ev.node;
|
||||||
|
if (TEXT_NODES.indexOf(node.type) > -1) {
|
||||||
const dummy_renderer = {};
|
// definitely text
|
||||||
for (const k of Object.keys(marked.Renderer.prototype)) {
|
continue;
|
||||||
dummy_renderer[k] = setNotPlain;
|
} else if (node.type == 'html_inline' || node.type == 'html_block') {
|
||||||
}
|
// if it's an allowed html tag, we need to render it and therefore
|
||||||
// text and paragraph are just text
|
// we will need to use HTML. If it's not allowed, it's not HTML since
|
||||||
dummy_renderer.text = function(t){return t;}
|
// we'll just be treating it as text.
|
||||||
dummy_renderer.paragraph = function(t){return t;}
|
if (is_allowed_html_tag(node)) {
|
||||||
|
return false;
|
||||||
// ignore links where text is just the url:
|
}
|
||||||
// this ignores plain URLs that markdown has
|
} else {
|
||||||
// detected whilst preserving markdown syntax links
|
return false;
|
||||||
dummy_renderer.link = function(href, title, text) {
|
|
||||||
if (text != href) {
|
|
||||||
is_plain = false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
const dummy_options = Object.assign({}, marked_options, {
|
|
||||||
renderer: dummy_renderer,
|
|
||||||
});
|
|
||||||
const dummy_parser = new marked.Parser(dummy_options);
|
|
||||||
dummy_parser.parse(this._copyTokens());
|
|
||||||
|
|
||||||
return is_plain;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
toHTML() {
|
toHTML() {
|
||||||
const real_renderer = new marked.Renderer();
|
const renderer = new commonmark.HtmlRenderer({
|
||||||
real_renderer.link = function(href, title, text) {
|
safe: false,
|
||||||
// prevent marked from turning plain URLs
|
|
||||||
// into links, because its algorithm is fairly
|
|
||||||
// poor. Let's send plain URLs rather than
|
|
||||||
// badly linkified ones (the linkifier Vector
|
|
||||||
// uses on message display is way better, eg.
|
|
||||||
// handles URLs with closing parens at the end).
|
|
||||||
if (text == href) {
|
|
||||||
return href;
|
|
||||||
}
|
|
||||||
return marked.Renderer.prototype.link.apply(this, arguments);
|
|
||||||
}
|
|
||||||
|
|
||||||
real_renderer.paragraph = (text) => {
|
// Set soft breaks to hard HTML breaks: commonmark
|
||||||
// The tokens at the top level are the 'blocks', so if we
|
// puts softbreaks in for multiple lines in a blockquote,
|
||||||
// have more than one, there are multiple 'paragraphs'.
|
// so if these are just newline characters then the
|
||||||
// If there is only one top level token, just return the
|
// block quote ends up all on one line
|
||||||
// bare text: it's a single line of text and so should be
|
// (https://github.com/vector-im/riot-web/issues/3154)
|
||||||
// 'inline', rather than necessarily wrapped in its own
|
softbreak: '<br />',
|
||||||
// p tag. If, however, we have multiple tokens, each gets
|
|
||||||
// its own p tag to keep them as separate paragraphs.
|
|
||||||
if (this.tokens.length == 1) {
|
|
||||||
return text;
|
|
||||||
}
|
|
||||||
return '<p>' + text + '</p>';
|
|
||||||
}
|
|
||||||
|
|
||||||
const real_options = Object.assign({}, marked_options, {
|
|
||||||
renderer: real_renderer,
|
|
||||||
});
|
});
|
||||||
const real_parser = new marked.Parser(real_options);
|
const real_paragraph = renderer.paragraph;
|
||||||
return real_parser.parse(this._copyTokens());
|
|
||||||
|
renderer.paragraph = function(node, entering) {
|
||||||
|
// If there is only one top level node, just return the
|
||||||
|
// bare text: it's a single line of text and so should be
|
||||||
|
// 'inline', rather than unnecessarily wrapped in its own
|
||||||
|
// p tag. If, however, we have multiple nodes, each gets
|
||||||
|
// its own p tag to keep them as separate paragraphs.
|
||||||
|
if (is_multi_line(node)) {
|
||||||
|
real_paragraph.call(this, node, entering);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
renderer.html_inline = html_if_tag_allowed;
|
||||||
|
renderer.html_block = function(node) {
|
||||||
|
// as with `paragraph`, we only insert line breaks
|
||||||
|
// if there are multiple lines in the markdown.
|
||||||
|
const isMultiLine = is_multi_line(node);
|
||||||
|
|
||||||
|
if (isMultiLine) this.cr();
|
||||||
|
html_if_tag_allowed.call(this, node);
|
||||||
|
if (isMultiLine) this.cr();
|
||||||
|
};
|
||||||
|
|
||||||
|
return renderer.render(this.parsed);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Render the markdown message to plain text. That is, essentially
|
||||||
|
* just remove any backslashes escaping what would otherwise be
|
||||||
|
* markdown syntax
|
||||||
|
* (to fix https://github.com/vector-im/riot-web/issues/2870)
|
||||||
|
*/
|
||||||
|
toPlaintext() {
|
||||||
|
const renderer = new commonmark.HtmlRenderer({safe: false});
|
||||||
|
const real_paragraph = renderer.paragraph;
|
||||||
|
|
||||||
|
// The default `out` function only sends the input through an XML
|
||||||
|
// escaping function, which causes messages to be entity encoded,
|
||||||
|
// which we don't want in this case.
|
||||||
|
renderer.out = function(s) {
|
||||||
|
// The `lit` function adds a string literal to the output buffer.
|
||||||
|
this.lit(s);
|
||||||
|
};
|
||||||
|
|
||||||
|
renderer.paragraph = function(node, entering) {
|
||||||
|
// as with toHTML, only append lines to paragraphs if there are
|
||||||
|
// multiple paragraphs
|
||||||
|
if (is_multi_line(node)) {
|
||||||
|
if (!entering && node.next) {
|
||||||
|
this.lit('\n\n');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
renderer.html_block = function(node) {
|
||||||
|
this.lit(node.literal);
|
||||||
|
if (is_multi_line(node) && node.next) this.lit('\n\n');
|
||||||
|
};
|
||||||
|
|
||||||
|
return renderer.render(this.parsed);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd.
|
||||||
|
Copyright 2017 New Vector Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -16,12 +18,12 @@ limitations under the License.
|
||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
import Matrix from 'matrix-js-sdk';
|
|
||||||
import utils from 'matrix-js-sdk/lib/utils';
|
import utils from 'matrix-js-sdk/lib/utils';
|
||||||
import EventTimeline from 'matrix-js-sdk/lib/models/event-timeline';
|
import EventTimeline from 'matrix-js-sdk/lib/models/event-timeline';
|
||||||
import EventTimelineSet from 'matrix-js-sdk/lib/models/event-timeline-set';
|
import EventTimelineSet from 'matrix-js-sdk/lib/models/event-timeline-set';
|
||||||
|
import createMatrixClient from './utils/createMatrixClient';
|
||||||
const localStorage = window.localStorage;
|
import SettingsStore from './settings/SettingsStore';
|
||||||
|
import MatrixActionCreators from './actions/MatrixActionCreators';
|
||||||
|
|
||||||
interface MatrixClientCreds {
|
interface MatrixClientCreds {
|
||||||
homeserverUrl: string,
|
homeserverUrl: string,
|
||||||
|
@ -51,12 +53,25 @@ class MatrixClientPeg {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the script href passed to the IndexedDB web worker
|
||||||
|
* If set, a separate web worker will be started to run the IndexedDB
|
||||||
|
* queries on.
|
||||||
|
*
|
||||||
|
* @param {string} script href to the script to be passed to the web worker
|
||||||
|
*/
|
||||||
|
setIndexedDbWorkerScript(script) {
|
||||||
|
createMatrixClient.indexedDbWorkerScript = script;
|
||||||
|
}
|
||||||
|
|
||||||
get(): MatrixClient {
|
get(): MatrixClient {
|
||||||
return this.matrixClient;
|
return this.matrixClient;
|
||||||
}
|
}
|
||||||
|
|
||||||
unset() {
|
unset() {
|
||||||
this.matrixClient = null;
|
this.matrixClient = null;
|
||||||
|
|
||||||
|
MatrixActionCreators.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -67,11 +82,42 @@ class MatrixClientPeg {
|
||||||
this._createClient(creds);
|
this._createClient(creds);
|
||||||
}
|
}
|
||||||
|
|
||||||
start() {
|
async start() {
|
||||||
|
// try to initialise e2e on the new client
|
||||||
|
try {
|
||||||
|
// check that we have a version of the js-sdk which includes initCrypto
|
||||||
|
if (this.matrixClient.initCrypto) {
|
||||||
|
await this.matrixClient.initCrypto();
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// this can happen for a number of reasons, the most likely being
|
||||||
|
// that the olm library was missing. It's not fatal.
|
||||||
|
console.warn("Unable to initialise e2e: " + e);
|
||||||
|
}
|
||||||
|
|
||||||
const opts = utils.deepCopy(this.opts);
|
const opts = utils.deepCopy(this.opts);
|
||||||
// the react sdk doesn't work without this, so don't allow
|
// the react sdk doesn't work without this, so don't allow
|
||||||
opts.pendingEventOrdering = "detached";
|
opts.pendingEventOrdering = "detached";
|
||||||
|
opts.disablePresence = true; // we do this manually
|
||||||
|
|
||||||
|
try {
|
||||||
|
const promise = this.matrixClient.store.startup();
|
||||||
|
console.log(`MatrixClientPeg: waiting for MatrixClient store to initialise`);
|
||||||
|
await promise;
|
||||||
|
} catch (err) {
|
||||||
|
// log any errors when starting up the database (if one exists)
|
||||||
|
console.error(`Error starting matrixclient store: ${err}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// regardless of errors, start the client. If we did error out, we'll
|
||||||
|
// just end up doing a full initial /sync.
|
||||||
|
|
||||||
|
// Connect the matrix client to the dispatcher
|
||||||
|
MatrixActionCreators.start(this.matrixClient);
|
||||||
|
|
||||||
|
console.log(`MatrixClientPeg: really starting MatrixClient`);
|
||||||
this.get().startClient(opts);
|
this.get().startClient(opts);
|
||||||
|
console.log(`MatrixClientPeg: MatrixClient started`);
|
||||||
}
|
}
|
||||||
|
|
||||||
getCredentials(): MatrixClientCreds {
|
getCredentials(): MatrixClientCreds {
|
||||||
|
@ -99,20 +145,17 @@ class MatrixClientPeg {
|
||||||
}
|
}
|
||||||
|
|
||||||
_createClient(creds: MatrixClientCreds) {
|
_createClient(creds: MatrixClientCreds) {
|
||||||
var opts = {
|
const opts = {
|
||||||
baseUrl: creds.homeserverUrl,
|
baseUrl: creds.homeserverUrl,
|
||||||
idBaseUrl: creds.identityServerUrl,
|
idBaseUrl: creds.identityServerUrl,
|
||||||
accessToken: creds.accessToken,
|
accessToken: creds.accessToken,
|
||||||
userId: creds.userId,
|
userId: creds.userId,
|
||||||
deviceId: creds.deviceId,
|
deviceId: creds.deviceId,
|
||||||
timelineSupport: true,
|
timelineSupport: true,
|
||||||
|
forceTURN: SettingsStore.getValue('webRtcForceTURN', false),
|
||||||
};
|
};
|
||||||
|
|
||||||
if (localStorage) {
|
this.matrixClient = createMatrixClient(opts, this.indexedDbWorkerScript);
|
||||||
opts.sessionStore = new Matrix.WebStorageSessionStore(localStorage);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.matrixClient = Matrix.createClient(opts);
|
|
||||||
|
|
||||||
// we're going to add eventlisteners for each matrix event tile, so the
|
// we're going to add eventlisteners for each matrix event tile, so the
|
||||||
// potential number of event listeners is quite high.
|
// potential number of event listeners is quite high.
|
||||||
|
@ -120,8 +163,8 @@ class MatrixClientPeg {
|
||||||
|
|
||||||
this.matrixClient.setGuest(Boolean(creds.guest));
|
this.matrixClient.setGuest(Boolean(creds.guest));
|
||||||
|
|
||||||
var notifTimelineSet = new EventTimelineSet(null, {
|
const notifTimelineSet = new EventTimelineSet(null, {
|
||||||
timelineSupport: true
|
timelineSupport: true,
|
||||||
});
|
});
|
||||||
// XXX: what is our initial pagination token?! it somehow needs to be synchronised with /sync.
|
// XXX: what is our initial pagination token?! it somehow needs to be synchronised with /sync.
|
||||||
notifTimelineSet.getLiveTimeline().setPaginationToken("", EventTimeline.BACKWARDS);
|
notifTimelineSet.getLiveTimeline().setPaginationToken("", EventTimeline.BACKWARDS);
|
||||||
|
|
190
src/Modal.js
190
src/Modal.js
|
@ -17,46 +17,196 @@ limitations under the License.
|
||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var React = require('react');
|
const React = require('react');
|
||||||
var ReactDOM = require('react-dom');
|
const ReactDOM = require('react-dom');
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import Analytics from './Analytics';
|
||||||
|
import sdk from './index';
|
||||||
|
|
||||||
module.exports = {
|
const DIALOG_CONTAINER_ID = "mx_Dialog_Container";
|
||||||
DialogContainerId: "mx_Dialog_Container",
|
|
||||||
|
|
||||||
getOrCreateContainer: function() {
|
/**
|
||||||
var container = document.getElementById(this.DialogContainerId);
|
* Wrap an asynchronous loader function with a react component which shows a
|
||||||
|
* spinner until the real component loads.
|
||||||
|
*/
|
||||||
|
const AsyncWrapper = React.createClass({
|
||||||
|
propTypes: {
|
||||||
|
/** A function which takes a 'callback' argument which it will call
|
||||||
|
* with the real component once it loads.
|
||||||
|
*/
|
||||||
|
loader: PropTypes.func.isRequired,
|
||||||
|
},
|
||||||
|
|
||||||
|
getInitialState: function() {
|
||||||
|
return {
|
||||||
|
component: null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillMount: function() {
|
||||||
|
this._unmounted = false;
|
||||||
|
// XXX: temporary logging to try to diagnose
|
||||||
|
// https://github.com/vector-im/riot-web/issues/3148
|
||||||
|
console.log('Starting load of AsyncWrapper for modal');
|
||||||
|
this.props.loader((e) => {
|
||||||
|
// XXX: temporary logging to try to diagnose
|
||||||
|
// https://github.com/vector-im/riot-web/issues/3148
|
||||||
|
console.log('AsyncWrapper load completed with '+e.displayName);
|
||||||
|
if (this._unmounted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.setState({component: e});
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillUnmount: function() {
|
||||||
|
this._unmounted = true;
|
||||||
|
},
|
||||||
|
|
||||||
|
render: function() {
|
||||||
|
const {loader, ...otherProps} = this.props;
|
||||||
|
if (this.state.component) {
|
||||||
|
const Component = this.state.component;
|
||||||
|
return <Component {...otherProps} />;
|
||||||
|
} else {
|
||||||
|
// show a spinner until the component is loaded.
|
||||||
|
const Spinner = sdk.getComponent("elements.Spinner");
|
||||||
|
return <Spinner />;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
class ModalManager {
|
||||||
|
constructor() {
|
||||||
|
this._counter = 0;
|
||||||
|
|
||||||
|
/** list of the modals we have stacked up, with the most recent at [0] */
|
||||||
|
this._modals = [
|
||||||
|
/* {
|
||||||
|
elem: React component for this dialog
|
||||||
|
onFinished: caller-supplied onFinished callback
|
||||||
|
className: CSS class for the dialog wrapper div
|
||||||
|
} */
|
||||||
|
];
|
||||||
|
|
||||||
|
this.closeAll = this.closeAll.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
getOrCreateContainer() {
|
||||||
|
let container = document.getElementById(DIALOG_CONTAINER_ID);
|
||||||
|
|
||||||
if (!container) {
|
if (!container) {
|
||||||
container = document.createElement("div");
|
container = document.createElement("div");
|
||||||
container.id = this.DialogContainerId;
|
container.id = DIALOG_CONTAINER_ID;
|
||||||
document.body.appendChild(container);
|
document.body.appendChild(container);
|
||||||
}
|
}
|
||||||
|
|
||||||
return container;
|
return container;
|
||||||
},
|
}
|
||||||
|
|
||||||
createDialog: function (Element, props, className) {
|
createTrackedDialog(analyticsAction, analyticsInfo, Element, props, className) {
|
||||||
var self = this;
|
Analytics.trackEvent('Modal', analyticsAction, analyticsInfo);
|
||||||
|
return this.createDialog(Element, props, className);
|
||||||
|
}
|
||||||
|
|
||||||
// never call this via modal.close() from onFinished() otherwise it will loop
|
createDialog(Element, props, className) {
|
||||||
var closeDialog = function() {
|
return this.createDialogAsync((cb) => {cb(Element);}, props, className);
|
||||||
|
}
|
||||||
|
|
||||||
|
createTrackedDialogAsync(analyticsAction, analyticsInfo, loader, props, className) {
|
||||||
|
Analytics.trackEvent('Modal', analyticsAction, analyticsInfo);
|
||||||
|
return this.createDialogAsync(loader, props, className);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Open a modal view.
|
||||||
|
*
|
||||||
|
* This can be used to display a react component which is loaded as an asynchronous
|
||||||
|
* webpack component. To do this, set 'loader' as:
|
||||||
|
*
|
||||||
|
* (cb) => {
|
||||||
|
* require(['<module>'], cb);
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* @param {Function} loader a function which takes a 'callback' argument,
|
||||||
|
* which it should call with a React component which will be displayed as
|
||||||
|
* the modal view.
|
||||||
|
*
|
||||||
|
* @param {Object} props properties to pass to the displayed
|
||||||
|
* component. (We will also pass an 'onFinished' property.)
|
||||||
|
*
|
||||||
|
* @param {String} className CSS class to apply to the modal wrapper
|
||||||
|
*/
|
||||||
|
createDialogAsync(loader, props, className) {
|
||||||
|
const self = this;
|
||||||
|
const modal = {};
|
||||||
|
|
||||||
|
// never call this from onFinished() otherwise it will loop
|
||||||
|
//
|
||||||
|
// nb explicit function() rather than arrow function, to get `arguments`
|
||||||
|
const closeDialog = function() {
|
||||||
if (props && props.onFinished) props.onFinished.apply(null, arguments);
|
if (props && props.onFinished) props.onFinished.apply(null, arguments);
|
||||||
ReactDOM.unmountComponentAtNode(self.getOrCreateContainer());
|
const i = self._modals.indexOf(modal);
|
||||||
|
if (i >= 0) {
|
||||||
|
self._modals.splice(i, 1);
|
||||||
|
}
|
||||||
|
self._reRender();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// don't attempt to reuse the same AsyncWrapper for different dialogs,
|
||||||
|
// otherwise we'll get confused.
|
||||||
|
const modalCount = this._counter++;
|
||||||
|
|
||||||
// FIXME: If a dialog uses getDefaultProps it clobbers the onFinished
|
// FIXME: If a dialog uses getDefaultProps it clobbers the onFinished
|
||||||
// property set here so you can't close the dialog from a button click!
|
// property set here so you can't close the dialog from a button click!
|
||||||
var dialog = (
|
modal.elem = (
|
||||||
<div className={"mx_Dialog_wrapper " + className}>
|
<AsyncWrapper key={modalCount} loader={loader} {...props}
|
||||||
|
onFinished={closeDialog} />
|
||||||
|
);
|
||||||
|
modal.onFinished = props ? props.onFinished : null;
|
||||||
|
modal.className = className;
|
||||||
|
|
||||||
|
this._modals.unshift(modal);
|
||||||
|
|
||||||
|
this._reRender();
|
||||||
|
return {close: closeDialog};
|
||||||
|
}
|
||||||
|
|
||||||
|
closeAll() {
|
||||||
|
const modals = this._modals;
|
||||||
|
this._modals = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < modals.length; i++) {
|
||||||
|
const m = modals[i];
|
||||||
|
if (m.onFinished) {
|
||||||
|
m.onFinished(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this._reRender();
|
||||||
|
}
|
||||||
|
|
||||||
|
_reRender() {
|
||||||
|
if (this._modals.length == 0) {
|
||||||
|
ReactDOM.unmountComponentAtNode(this.getOrCreateContainer());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const modal = this._modals[0];
|
||||||
|
const dialog = (
|
||||||
|
<div className={"mx_Dialog_wrapper " + (modal.className ? modal.className : '')}>
|
||||||
<div className="mx_Dialog">
|
<div className="mx_Dialog">
|
||||||
<Element {...props} onFinished={closeDialog}/>
|
{ modal.elem }
|
||||||
</div>
|
</div>
|
||||||
<div className="mx_Dialog_background" onClick={ closeDialog.bind(this, false) }></div>
|
<div className="mx_Dialog_background" onClick={this.closeAll}></div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
ReactDOM.render(dialog, this.getOrCreateContainer());
|
ReactDOM.render(dialog, this.getOrCreateContainer());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {close: closeDialog};
|
if (!global.singletonModalManager) {
|
||||||
},
|
global.singletonModalManager = new ModalManager();
|
||||||
};
|
}
|
||||||
|
export default global.singletonModalManager;
|
||||||
|
|
197
src/Notifier.js
197
src/Notifier.js
|
@ -1,5 +1,7 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
Copyright 2017 New Vector Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -14,13 +16,16 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'use strict';
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
import PlatformPeg from './PlatformPeg';
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
import TextForEvent from './TextForEvent';
|
||||||
var PlatformPeg = require("./PlatformPeg");
|
import Analytics from './Analytics';
|
||||||
var TextForEvent = require('./TextForEvent');
|
import Avatar from './Avatar';
|
||||||
var Avatar = require('./Avatar');
|
import dis from './dispatcher';
|
||||||
var dis = require("./dispatcher");
|
import sdk from './index';
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
import Modal from './Modal';
|
||||||
|
import SettingsStore, {SettingLevel} from "./settings/SettingsStore";
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Dispatches:
|
* Dispatches:
|
||||||
|
@ -30,9 +35,16 @@ var dis = require("./dispatcher");
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var Notifier = {
|
const MAX_PENDING_ENCRYPTED = 20;
|
||||||
|
|
||||||
|
const Notifier = {
|
||||||
notifsByRoom: {},
|
notifsByRoom: {},
|
||||||
|
|
||||||
|
// A list of event IDs that we've received but need to wait until
|
||||||
|
// they're decrypted until we decide whether to notify for them
|
||||||
|
// or not
|
||||||
|
pendingEncryptedEventIds: [],
|
||||||
|
|
||||||
notificationMessageForEvent: function(ev) {
|
notificationMessageForEvent: function(ev) {
|
||||||
return TextForEvent.textForEvent(ev);
|
return TextForEvent.textForEvent(ev);
|
||||||
},
|
},
|
||||||
|
@ -49,16 +61,16 @@ var Notifier = {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var msg = this.notificationMessageForEvent(ev);
|
let msg = this.notificationMessageForEvent(ev);
|
||||||
if (!msg) return;
|
if (!msg) return;
|
||||||
|
|
||||||
var title;
|
let title;
|
||||||
if (!ev.sender || room.name == ev.sender.name) {
|
if (!ev.sender || room.name === ev.sender.name) {
|
||||||
title = room.name;
|
title = room.name;
|
||||||
// notificationMessageForEvent includes sender,
|
// notificationMessageForEvent includes sender,
|
||||||
// but we already have the sender here
|
// but we already have the sender here
|
||||||
if (ev.getContent().body) msg = ev.getContent().body;
|
if (ev.getContent().body) msg = ev.getContent().body;
|
||||||
} else if (ev.getType() == 'm.room.member') {
|
} else if (ev.getType() === 'm.room.member') {
|
||||||
// context is all in the message here, we don't need
|
// context is all in the message here, we don't need
|
||||||
// to display sender info
|
// to display sender info
|
||||||
title = room.name;
|
title = room.name;
|
||||||
|
@ -69,10 +81,11 @@ var Notifier = {
|
||||||
if (ev.getContent().body) msg = ev.getContent().body;
|
if (ev.getContent().body) msg = ev.getContent().body;
|
||||||
}
|
}
|
||||||
|
|
||||||
var avatarUrl = ev.sender ? Avatar.avatarUrlForMember(
|
if (!this.isBodyEnabled()) {
|
||||||
ev.sender, 40, 40, 'crop'
|
msg = '';
|
||||||
) : null;
|
}
|
||||||
|
|
||||||
|
const avatarUrl = ev.sender ? Avatar.avatarUrlForMember(ev.sender, 40, 40, 'crop') : null;
|
||||||
const notif = plaf.displayNotification(title, msg, avatarUrl, room);
|
const notif = plaf.displayNotification(title, msg, avatarUrl, room);
|
||||||
|
|
||||||
// if displayNotification returns non-null, the platform supports
|
// if displayNotification returns non-null, the platform supports
|
||||||
|
@ -84,31 +97,33 @@ var Notifier = {
|
||||||
},
|
},
|
||||||
|
|
||||||
_playAudioNotification: function(ev, room) {
|
_playAudioNotification: function(ev, room) {
|
||||||
var e = document.getElementById("messageAudio");
|
const e = document.getElementById("messageAudio");
|
||||||
if (e) {
|
if (e) {
|
||||||
e.load();
|
|
||||||
e.play();
|
e.play();
|
||||||
};
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
start: function() {
|
start: function() {
|
||||||
this.boundOnRoomTimeline = this.onRoomTimeline.bind(this);
|
this.boundOnEvent = this.onEvent.bind(this);
|
||||||
this.boundOnSyncStateChange = this.onSyncStateChange.bind(this);
|
this.boundOnSyncStateChange = this.onSyncStateChange.bind(this);
|
||||||
this.boundOnRoomReceipt = this.onRoomReceipt.bind(this);
|
this.boundOnRoomReceipt = this.onRoomReceipt.bind(this);
|
||||||
MatrixClientPeg.get().on('Room.timeline', this.boundOnRoomTimeline);
|
this.boundOnEventDecrypted = this.onEventDecrypted.bind(this);
|
||||||
MatrixClientPeg.get().on("Room.receipt", this.boundOnRoomReceipt);
|
MatrixClientPeg.get().on('event', this.boundOnEvent);
|
||||||
|
MatrixClientPeg.get().on('Room.receipt', this.boundOnRoomReceipt);
|
||||||
|
MatrixClientPeg.get().on('Event.decrypted', this.boundOnEventDecrypted);
|
||||||
MatrixClientPeg.get().on("sync", this.boundOnSyncStateChange);
|
MatrixClientPeg.get().on("sync", this.boundOnSyncStateChange);
|
||||||
this.toolbarHidden = false;
|
this.toolbarHidden = false;
|
||||||
this.isPrepared = false;
|
this.isSyncing = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
stop: function() {
|
stop: function() {
|
||||||
if (MatrixClientPeg.get()) {
|
if (MatrixClientPeg.get() && this.boundOnRoomTimeline) {
|
||||||
MatrixClientPeg.get().removeListener('Room.timeline', this.boundOnRoomTimeline);
|
MatrixClientPeg.get().removeListener('Event', this.boundOnEvent);
|
||||||
MatrixClientPeg.get().removeListener("Room.receipt", this.boundOnRoomReceipt);
|
MatrixClientPeg.get().removeListener('Room.receipt', this.boundOnRoomReceipt);
|
||||||
|
MatrixClientPeg.get().removeListener('Event.decrypted', this.boundOnEventDecrypted);
|
||||||
MatrixClientPeg.get().removeListener('sync', this.boundOnSyncStateChange);
|
MatrixClientPeg.get().removeListener('sync', this.boundOnSyncStateChange);
|
||||||
}
|
}
|
||||||
this.isPrepared = false;
|
this.isSyncing = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
supportsDesktopNotifications: function() {
|
supportsDesktopNotifications: function() {
|
||||||
|
@ -119,12 +134,17 @@ var Notifier = {
|
||||||
setEnabled: function(enable, callback) {
|
setEnabled: function(enable, callback) {
|
||||||
const plaf = PlatformPeg.get();
|
const plaf = PlatformPeg.get();
|
||||||
if (!plaf) return;
|
if (!plaf) return;
|
||||||
|
|
||||||
|
// Dev note: We don't set the "notificationsEnabled" setting to true here because it is a
|
||||||
|
// calculated value. It is determined based upon whether or not the master rule is enabled
|
||||||
|
// and other flags. Setting it here would cause a circular reference.
|
||||||
|
|
||||||
|
Analytics.trackEvent('Notifier', 'Set Enabled', enable);
|
||||||
|
|
||||||
// make sure that we persist the current setting audio_enabled setting
|
// make sure that we persist the current setting audio_enabled setting
|
||||||
// before changing anything
|
// before changing anything
|
||||||
if (global.localStorage) {
|
if (SettingsStore.isLevelSupported(SettingLevel.DEVICE)) {
|
||||||
if(global.localStorage.getItem('audio_notifications_enabled') == null) {
|
SettingsStore.setValue("audioNotificationsEnabled", null, SettingLevel.DEVICE, this.isEnabled());
|
||||||
this.setAudioEnabled(this.isEnabled());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (enable) {
|
if (enable) {
|
||||||
|
@ -132,116 +152,119 @@ var Notifier = {
|
||||||
plaf.requestNotificationPermission().done((result) => {
|
plaf.requestNotificationPermission().done((result) => {
|
||||||
if (result !== 'granted') {
|
if (result !== 'granted') {
|
||||||
// The permission request was dismissed or denied
|
// The permission request was dismissed or denied
|
||||||
|
// TODO: Support alternative branding in messaging
|
||||||
|
const description = result === 'denied'
|
||||||
|
? _t('Riot does not have permission to send you notifications - please check your browser settings')
|
||||||
|
: _t('Riot was not given permission to send notifications - please try again');
|
||||||
|
const ErrorDialog = sdk.getComponent('dialogs.ErrorDialog');
|
||||||
|
Modal.createTrackedDialog('Unable to enable Notifications', result, ErrorDialog, {
|
||||||
|
title: _t('Unable to enable Notifications'),
|
||||||
|
description,
|
||||||
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (global.localStorage) {
|
|
||||||
global.localStorage.setItem('notifications_enabled', 'true');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (callback) callback();
|
if (callback) callback();
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: "notifier_enabled",
|
action: "notifier_enabled",
|
||||||
value: true
|
value: true,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
// clear the notifications_hidden flag, so that if notifications are
|
// clear the notifications_hidden flag, so that if notifications are
|
||||||
// disabled again in the future, we will show the banner again.
|
// disabled again in the future, we will show the banner again.
|
||||||
this.setToolbarHidden(false);
|
this.setToolbarHidden(true);
|
||||||
} else {
|
} else {
|
||||||
if (!global.localStorage) return;
|
|
||||||
global.localStorage.setItem('notifications_enabled', 'false');
|
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: "notifier_enabled",
|
action: "notifier_enabled",
|
||||||
value: false
|
value: false,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
isEnabled: function() {
|
isEnabled: function() {
|
||||||
|
return this.isPossible() && SettingsStore.getValue("notificationsEnabled");
|
||||||
|
},
|
||||||
|
|
||||||
|
isPossible: function() {
|
||||||
const plaf = PlatformPeg.get();
|
const plaf = PlatformPeg.get();
|
||||||
if (!plaf) return false;
|
if (!plaf) return false;
|
||||||
if (!plaf.supportsNotifications()) return false;
|
if (!plaf.supportsNotifications()) return false;
|
||||||
if (!plaf.maySendNotifications()) return false;
|
if (!plaf.maySendNotifications()) return false;
|
||||||
|
|
||||||
if (!global.localStorage) return true;
|
return true; // possible, but not necessarily enabled
|
||||||
|
|
||||||
var enabled = global.localStorage.getItem('notifications_enabled');
|
|
||||||
if (enabled === null) return true;
|
|
||||||
return enabled === 'true';
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setAudioEnabled: function(enable) {
|
isBodyEnabled: function() {
|
||||||
if (!global.localStorage) return;
|
return this.isEnabled() && SettingsStore.getValue("notificationBodyEnabled");
|
||||||
global.localStorage.setItem('audio_notifications_enabled',
|
|
||||||
enable ? 'true' : 'false');
|
|
||||||
},
|
},
|
||||||
|
|
||||||
isAudioEnabled: function(enable) {
|
isAudioEnabled: function() {
|
||||||
if (!global.localStorage) return true;
|
return this.isEnabled() && SettingsStore.getValue("audioNotificationsEnabled");
|
||||||
var enabled = global.localStorage.getItem(
|
|
||||||
'audio_notifications_enabled');
|
|
||||||
// default to true if the popups are enabled
|
|
||||||
if (enabled === null) return this.isEnabled();
|
|
||||||
return enabled === 'true';
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setToolbarHidden: function(hidden, persistent = true) {
|
setToolbarHidden: function(hidden, persistent = true) {
|
||||||
this.toolbarHidden = hidden;
|
this.toolbarHidden = hidden;
|
||||||
|
|
||||||
|
Analytics.trackEvent('Notifier', 'Set Toolbar Hidden', hidden);
|
||||||
|
|
||||||
// XXX: why are we dispatching this here?
|
// XXX: why are we dispatching this here?
|
||||||
// this is nothing to do with notifier_enabled
|
// this is nothing to do with notifier_enabled
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: "notifier_enabled",
|
action: "notifier_enabled",
|
||||||
value: this.isEnabled()
|
value: this.isEnabled(),
|
||||||
});
|
});
|
||||||
|
|
||||||
// update the info to localStorage for persistent settings
|
// update the info to localStorage for persistent settings
|
||||||
if (persistent && global.localStorage) {
|
if (persistent && global.localStorage) {
|
||||||
global.localStorage.setItem('notifications_hidden', hidden);
|
global.localStorage.setItem("notifications_hidden", hidden);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
isToolbarHidden: function() {
|
isToolbarHidden: function() {
|
||||||
// Check localStorage for any such meta data
|
// Check localStorage for any such meta data
|
||||||
if (global.localStorage) {
|
if (global.localStorage) {
|
||||||
if (global.localStorage.getItem('notifications_hidden') === 'true') {
|
return global.localStorage.getItem("notifications_hidden") === "true";
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.toolbarHidden;
|
return this.toolbarHidden;
|
||||||
},
|
},
|
||||||
|
|
||||||
onSyncStateChange: function(state) {
|
onSyncStateChange: function(state) {
|
||||||
if (state === "PREPARED" || state === "SYNCING") {
|
if (state === "SYNCING") {
|
||||||
this.isPrepared = true;
|
this.isSyncing = true;
|
||||||
}
|
} else if (state === "STOPPED" || state === "ERROR") {
|
||||||
else if (state === "STOPPED" || state === "ERROR") {
|
this.isSyncing = false;
|
||||||
this.isPrepared = false;
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
onRoomTimeline: function(ev, room, toStartOfTimeline, removed, data) {
|
onEvent: function(ev) {
|
||||||
if (toStartOfTimeline) return;
|
if (!this.isSyncing) return; // don't alert for any messages initially
|
||||||
if (!room) return;
|
if (ev.sender && ev.sender.userId === MatrixClientPeg.get().credentials.userId) return;
|
||||||
if (!this.isPrepared) return; // don't alert for any messages initially
|
|
||||||
if (ev.sender && ev.sender.userId == MatrixClientPeg.get().credentials.userId) return;
|
|
||||||
if (data.timeline.getTimelineSet() !== room.getUnfilteredTimelineSet()) return;
|
|
||||||
|
|
||||||
var actions = MatrixClientPeg.get().getPushActionsForEvent(ev);
|
// If it's an encrypted event and the type is still 'm.room.encrypted',
|
||||||
if (actions && actions.notify) {
|
// it hasn't yet been decrypted, so wait until it is.
|
||||||
if (this.isEnabled()) {
|
if (ev.isBeingDecrypted() || ev.isDecryptionFailure()) {
|
||||||
this._displayPopupNotification(ev, room);
|
this.pendingEncryptedEventIds.push(ev.getId());
|
||||||
}
|
// don't let the list fill up indefinitely
|
||||||
if (actions.tweaks.sound && this.isAudioEnabled()) {
|
while (this.pendingEncryptedEventIds.length > MAX_PENDING_ENCRYPTED) {
|
||||||
this._playAudioNotification(ev, room);
|
this.pendingEncryptedEventIds.shift();
|
||||||
}
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this._evaluateEvent(ev);
|
||||||
|
},
|
||||||
|
|
||||||
|
onEventDecrypted: function(ev) {
|
||||||
|
const idx = this.pendingEncryptedEventIds.indexOf(ev.getId());
|
||||||
|
if (idx === -1) return;
|
||||||
|
|
||||||
|
this.pendingEncryptedEventIds.splice(idx, 1);
|
||||||
|
this._evaluateEvent(ev);
|
||||||
},
|
},
|
||||||
|
|
||||||
onRoomReceipt: function(ev, room) {
|
onRoomReceipt: function(ev, room) {
|
||||||
if (room.getUnreadNotificationCount() == 0) {
|
if (room.getUnreadNotificationCount() === 0) {
|
||||||
// ideally we would clear each notification when it was read,
|
// ideally we would clear each notification when it was read,
|
||||||
// but we have no way, given a read receipt, to know whether
|
// but we have no way, given a read receipt, to know whether
|
||||||
// the receipt comes before or after an event, so we can't
|
// the receipt comes before or after an event, so we can't
|
||||||
|
@ -256,7 +279,21 @@ var Notifier = {
|
||||||
}
|
}
|
||||||
delete this.notifsByRoom[room.roomId];
|
delete this.notifsByRoom[room.roomId];
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
|
||||||
|
_evaluateEvent: function(ev) {
|
||||||
|
const room = MatrixClientPeg.get().getRoom(ev.getRoomId());
|
||||||
|
const actions = MatrixClientPeg.get().getPushActionsForEvent(ev);
|
||||||
|
if (actions && actions.notify) {
|
||||||
|
if (this.isEnabled()) {
|
||||||
|
this._displayPopupNotification(ev, room);
|
||||||
|
}
|
||||||
|
if (actions.tweaks.sound && this.isAudioEnabled()) {
|
||||||
|
PlatformPeg.get().loudNotification(ev, room);
|
||||||
|
this._playAudioNotification(ev, room);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!global.mxNotifier) {
|
if (!global.mxNotifier) {
|
||||||
|
|
|
@ -23,8 +23,8 @@ limitations under the License.
|
||||||
* { key: $KEY, val: $VALUE, place: "add|del" }
|
* { key: $KEY, val: $VALUE, place: "add|del" }
|
||||||
*/
|
*/
|
||||||
module.exports.getKeyValueArrayDiffs = function(before, after) {
|
module.exports.getKeyValueArrayDiffs = function(before, after) {
|
||||||
var results = [];
|
const results = [];
|
||||||
var delta = {};
|
const delta = {};
|
||||||
Object.keys(before).forEach(function(beforeKey) {
|
Object.keys(before).forEach(function(beforeKey) {
|
||||||
delta[beforeKey] = delta[beforeKey] || 0; // init to 0 initially
|
delta[beforeKey] = delta[beforeKey] || 0; // init to 0 initially
|
||||||
delta[beforeKey]--; // keys present in the past have -ve values
|
delta[beforeKey]--; // keys present in the past have -ve values
|
||||||
|
@ -46,9 +46,9 @@ module.exports.getKeyValueArrayDiffs = function(before, after) {
|
||||||
results.push({ place: "del", key: muxedKey, val: beforeVal });
|
results.push({ place: "del", key: muxedKey, val: beforeVal });
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
case 0: // A mix of added/removed keys
|
case 0: {// A mix of added/removed keys
|
||||||
// compare old & new vals
|
// compare old & new vals
|
||||||
var itemDelta = {};
|
const itemDelta = {};
|
||||||
before[muxedKey].forEach(function(beforeVal) {
|
before[muxedKey].forEach(function(beforeVal) {
|
||||||
itemDelta[beforeVal] = itemDelta[beforeVal] || 0;
|
itemDelta[beforeVal] = itemDelta[beforeVal] || 0;
|
||||||
itemDelta[beforeVal]--;
|
itemDelta[beforeVal]--;
|
||||||
|
@ -64,13 +64,13 @@ module.exports.getKeyValueArrayDiffs = function(before, after) {
|
||||||
} else if (itemDelta[item] === -1) {
|
} else if (itemDelta[item] === -1) {
|
||||||
results.push({ place: "del", key: muxedKey, val: item });
|
results.push({ place: "del", key: muxedKey, val: item });
|
||||||
} else {
|
} else {
|
||||||
// itemDelta of 0 means it was unchanged between before/after
|
// itemDelta of 0 means it was unchanged between before/after
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
console.error("Calculated key delta of " + delta[muxedKey] +
|
console.error("Calculated key delta of " + delta[muxedKey] + " - this should never happen!");
|
||||||
" - this should never happen!");
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -79,8 +79,10 @@ module.exports.getKeyValueArrayDiffs = function(before, after) {
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Shallow-compare two objects for equality: each key and value must be
|
* Shallow-compare two objects for equality: each key and value must be identical
|
||||||
* identical
|
* @param {Object} objA First object to compare against the second
|
||||||
|
* @param {Object} objB Second object to compare against the first
|
||||||
|
* @return {boolean} whether the two objects have same key=values
|
||||||
*/
|
*/
|
||||||
module.exports.shallowEqual = function(objA, objB) {
|
module.exports.shallowEqual = function(objA, objB) {
|
||||||
if (objA === objB) {
|
if (objA === objB) {
|
||||||
|
@ -92,15 +94,15 @@ module.exports.shallowEqual = function(objA, objB) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
var keysA = Object.keys(objA);
|
const keysA = Object.keys(objA);
|
||||||
var keysB = Object.keys(objB);
|
const keysB = Object.keys(objB);
|
||||||
|
|
||||||
if (keysA.length !== keysB.length) {
|
if (keysA.length !== keysB.length) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var i = 0; i < keysA.length; i++) {
|
for (let i = 0; i < keysA.length; i++) {
|
||||||
var key = keysA[i];
|
const key = keysA[i];
|
||||||
if (!objB.hasOwnProperty(key) || objA[key] !== objB[key]) {
|
if (!objB.hasOwnProperty(key) || objA[key] !== objB[key]) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -16,9 +17,12 @@ limitations under the License.
|
||||||
|
|
||||||
/** The types of page which can be shown by the LoggedInView */
|
/** The types of page which can be shown by the LoggedInView */
|
||||||
export default {
|
export default {
|
||||||
|
HomePage: "home_page",
|
||||||
RoomView: "room_view",
|
RoomView: "room_view",
|
||||||
UserSettings: "user_settings",
|
UserSettings: "user_settings",
|
||||||
CreateRoom: "create_room",
|
CreateRoom: "create_room",
|
||||||
RoomDirectory: "room_directory",
|
RoomDirectory: "room_directory",
|
||||||
UserView: "user_view",
|
UserView: "user_view",
|
||||||
|
GroupView: "group_view",
|
||||||
|
MyGroups: "my_groups",
|
||||||
};
|
};
|
||||||
|
|
|
@ -14,7 +14,8 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var Matrix = require("matrix-js-sdk");
|
import * as Matrix from 'matrix-js-sdk';
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Allows a user to reset their password on a homeserver.
|
* Allows a user to reset their password on a homeserver.
|
||||||
|
@ -33,7 +34,7 @@ class PasswordReset {
|
||||||
constructor(homeserverUrl, identityUrl) {
|
constructor(homeserverUrl, identityUrl) {
|
||||||
this.client = Matrix.createClient({
|
this.client = Matrix.createClient({
|
||||||
baseUrl: homeserverUrl,
|
baseUrl: homeserverUrl,
|
||||||
idBaseUrl: identityUrl
|
idBaseUrl: identityUrl,
|
||||||
});
|
});
|
||||||
this.clientSecret = this.client.generateClientSecret();
|
this.clientSecret = this.client.generateClientSecret();
|
||||||
this.identityServerDomain = identityUrl.split("://")[1];
|
this.identityServerDomain = identityUrl.split("://")[1];
|
||||||
|
@ -52,8 +53,8 @@ class PasswordReset {
|
||||||
this.sessionId = res.sid;
|
this.sessionId = res.sid;
|
||||||
return res;
|
return res;
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
if (err.errcode == 'M_THREEPID_NOT_FOUND') {
|
if (err.errcode === 'M_THREEPID_NOT_FOUND') {
|
||||||
err.message = "This email address was not found";
|
err.message = _t('This email address was not found');
|
||||||
} else if (err.httpStatus) {
|
} else if (err.httpStatus) {
|
||||||
err.message = err.message + ` (Status ${err.httpStatus})`;
|
err.message = err.message + ` (Status ${err.httpStatus})`;
|
||||||
}
|
}
|
||||||
|
@ -74,16 +75,15 @@ class PasswordReset {
|
||||||
threepid_creds: {
|
threepid_creds: {
|
||||||
sid: this.sessionId,
|
sid: this.sessionId,
|
||||||
client_secret: this.clientSecret,
|
client_secret: this.clientSecret,
|
||||||
id_server: this.identityServerDomain
|
id_server: this.identityServerDomain,
|
||||||
}
|
},
|
||||||
}, this.password).catch(function(err) {
|
}, this.password).catch(function(err) {
|
||||||
if (err.httpStatus === 401) {
|
if (err.httpStatus === 401) {
|
||||||
err.message = "Failed to verify email address: make sure you clicked the link in the email";
|
err.message = _t('Failed to verify email address: make sure you clicked the link in the email');
|
||||||
}
|
} else if (err.httpStatus === 404) {
|
||||||
else if (err.httpStatus === 404) {
|
err.message =
|
||||||
err.message = "Your email address does not appear to be associated with a Matrix ID on this Homeserver.";
|
_t('Your email address does not appear to be associated with a Matrix ID on this Homeserver.');
|
||||||
}
|
} else if (err.httpStatus) {
|
||||||
else if (err.httpStatus) {
|
|
||||||
err.message += ` (Status ${err.httpStatus})`;
|
err.message += ` (Status ${err.httpStatus})`;
|
||||||
}
|
}
|
||||||
throw err;
|
throw err;
|
||||||
|
|
|
@ -14,12 +14,12 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
const MatrixClientPeg = require("./MatrixClientPeg");
|
||||||
var dis = require("./dispatcher");
|
const dis = require("./dispatcher");
|
||||||
|
|
||||||
// Time in ms after that a user is considered as unavailable/away
|
// Time in ms after that a user is considered as unavailable/away
|
||||||
var UNAVAILABLE_TIME_MS = 3 * 60 * 1000; // 3 mins
|
const UNAVAILABLE_TIME_MS = 3 * 60 * 1000; // 3 mins
|
||||||
var PRESENCE_STATES = ["online", "offline", "unavailable"];
|
const PRESENCE_STATES = ["online", "offline", "unavailable"];
|
||||||
|
|
||||||
class Presence {
|
class Presence {
|
||||||
|
|
||||||
|
@ -56,13 +56,27 @@ class Presence {
|
||||||
return this.state;
|
return this.state;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current status message.
|
||||||
|
* @returns {String} the status message, may be null
|
||||||
|
*/
|
||||||
|
getStatusMessage() {
|
||||||
|
return this.statusMessage;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the presence state.
|
* Set the presence state.
|
||||||
* If the state has changed, the Home Server will be notified.
|
* If the state has changed, the Home Server will be notified.
|
||||||
* @param {string} newState the new presence state (see PRESENCE enum)
|
* @param {string} newState the new presence state (see PRESENCE enum)
|
||||||
|
* @param {String} statusMessage an optional status message for the presence
|
||||||
|
* @param {boolean} maintain true to have this status maintained by this tracker
|
||||||
*/
|
*/
|
||||||
setState(newState) {
|
setState(newState, statusMessage=null, maintain=false) {
|
||||||
if (newState === this.state) {
|
if (this.maintain) {
|
||||||
|
// Don't update presence if we're maintaining a particular status
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (newState === this.state && statusMessage === this.statusMessage) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (PRESENCE_STATES.indexOf(newState) === -1) {
|
if (PRESENCE_STATES.indexOf(newState) === -1) {
|
||||||
|
@ -71,22 +85,38 @@ class Presence {
|
||||||
if (!this.running) {
|
if (!this.running) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
var old_state = this.state;
|
const old_state = this.state;
|
||||||
|
const old_message = this.statusMessage;
|
||||||
this.state = newState;
|
this.state = newState;
|
||||||
|
this.statusMessage = statusMessage;
|
||||||
|
this.maintain = maintain;
|
||||||
|
|
||||||
if (MatrixClientPeg.get().isGuest()) {
|
if (MatrixClientPeg.get().isGuest()) {
|
||||||
return; // don't try to set presence when a guest; it won't work.
|
return; // don't try to set presence when a guest; it won't work.
|
||||||
}
|
}
|
||||||
|
|
||||||
var self = this;
|
const updateContent = {
|
||||||
MatrixClientPeg.get().setPresence(this.state).done(function() {
|
presence: this.state,
|
||||||
|
status_msg: this.statusMessage ? this.statusMessage : '',
|
||||||
|
};
|
||||||
|
|
||||||
|
const self = this;
|
||||||
|
MatrixClientPeg.get().setPresence(updateContent).done(function() {
|
||||||
console.log("Presence: %s", newState);
|
console.log("Presence: %s", newState);
|
||||||
|
|
||||||
|
// We have to dispatch because the js-sdk is unreliable at telling us about our own presence
|
||||||
|
dis.dispatch({action: "self_presence_updated", statusInfo: updateContent});
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
console.error("Failed to set presence: %s", err);
|
console.error("Failed to set presence: %s", err);
|
||||||
self.state = old_state;
|
self.state = old_state;
|
||||||
|
self.statusMessage = old_message;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
stopMaintainingStatus() {
|
||||||
|
this.maintain = false;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Callback called when the user made no action on the page for UNAVAILABLE_TIME ms.
|
* Callback called when the user made no action on the page for UNAVAILABLE_TIME ms.
|
||||||
* @private
|
* @private
|
||||||
|
@ -95,7 +125,8 @@ class Presence {
|
||||||
this.setState("unavailable");
|
this.setState("unavailable");
|
||||||
}
|
}
|
||||||
|
|
||||||
_onUserActivity() {
|
_onUserActivity(payload) {
|
||||||
|
if (payload.action === "sync_state" || payload.action === "self_presence_updated") return;
|
||||||
this._resetTimer();
|
this._resetTimer();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,14 +135,14 @@ class Presence {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
_resetTimer() {
|
_resetTimer() {
|
||||||
var self = this;
|
const self = this;
|
||||||
this.setState("online");
|
this.setState("online");
|
||||||
// Re-arm the timer
|
// Re-arm the timer
|
||||||
clearTimeout(this.timer);
|
clearTimeout(this.timer);
|
||||||
this.timer = setTimeout(function() {
|
this.timer = setTimeout(function() {
|
||||||
self._onUnavailableTimerFire();
|
self._onUnavailableTimerFire();
|
||||||
}, UNAVAILABLE_TIME_MS);
|
}, UNAVAILABLE_TIME_MS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = new Presence();
|
module.exports = new Presence();
|
||||||
|
|
|
@ -14,31 +14,44 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var MatrixClientPeg = require('./MatrixClientPeg');
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
var dis = require('./dispatcher');
|
import dis from './dispatcher';
|
||||||
|
import { EventStatus } from 'matrix-js-sdk';
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
|
resendUnsentEvents: function(room) {
|
||||||
|
room.getPendingEvents().filter(function(ev) {
|
||||||
|
return ev.status === EventStatus.NOT_SENT;
|
||||||
|
}).forEach(function(event) {
|
||||||
|
module.exports.resend(event);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
cancelUnsentEvents: function(room) {
|
||||||
|
room.getPendingEvents().filter(function(ev) {
|
||||||
|
return ev.status === EventStatus.NOT_SENT;
|
||||||
|
}).forEach(function(event) {
|
||||||
|
module.exports.removeFromQueue(event);
|
||||||
|
});
|
||||||
|
},
|
||||||
resend: function(event) {
|
resend: function(event) {
|
||||||
MatrixClientPeg.get().resendEvent(
|
const room = MatrixClientPeg.get().getRoom(event.getRoomId());
|
||||||
event, MatrixClientPeg.get().getRoom(event.getRoomId())
|
MatrixClientPeg.get().resendEvent(event, room).done(function(res) {
|
||||||
).done(function() {
|
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'message_sent',
|
action: 'message_sent',
|
||||||
event: event
|
event: event,
|
||||||
});
|
});
|
||||||
}, function() {
|
}, function(err) {
|
||||||
|
// XXX: temporary logging to try to diagnose
|
||||||
|
// https://github.com/vector-im/riot-web/issues/3148
|
||||||
|
console.log('Resend got send failure: ' + err.name + '('+err+')');
|
||||||
|
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'message_send_failed',
|
action: 'message_send_failed',
|
||||||
event: event
|
event: event,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
removeFromQueue: function(event) {
|
removeFromQueue: function(event) {
|
||||||
MatrixClientPeg.get().cancelPendingEvent(event);
|
MatrixClientPeg.get().cancelPendingEvent(event);
|
||||||
dis.dispatch({
|
|
||||||
action: 'message_send_cancelled',
|
|
||||||
event: event
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
163
src/RichText.js
163
src/RichText.js
|
@ -12,10 +12,11 @@ import {
|
||||||
SelectionState,
|
SelectionState,
|
||||||
Entity,
|
Entity,
|
||||||
} from 'draft-js';
|
} from 'draft-js';
|
||||||
import * as sdk from './index';
|
import * as sdk from './index';
|
||||||
import * as emojione from 'emojione';
|
import * as emojione from 'emojione';
|
||||||
import {stateToHTML} from 'draft-js-export-html';
|
import {stateToHTML} from 'draft-js-export-html';
|
||||||
import {SelectionRange} from "./autocomplete/Autocompleter";
|
import {SelectionRange} from "./autocomplete/Autocompleter";
|
||||||
|
import {stateToMarkdown as __stateToMarkdown} from 'draft-js-export-markdown';
|
||||||
|
|
||||||
const MARKDOWN_REGEX = {
|
const MARKDOWN_REGEX = {
|
||||||
LINK: /(?:\[([^\]]+)\]\(([^\)]+)\))|\<(\w+:\/\/[^\>]+)\>/g,
|
LINK: /(?:\[([^\]]+)\]\(([^\)]+)\))|\<(\w+:\/\/[^\>]+)\>/g,
|
||||||
|
@ -30,17 +31,35 @@ const USERNAME_REGEX = /@\S+:\S+/g;
|
||||||
const ROOM_REGEX = /#\S+:\S+/g;
|
const ROOM_REGEX = /#\S+:\S+/g;
|
||||||
const EMOJI_REGEX = new RegExp(emojione.unicodeRegexp, 'g');
|
const EMOJI_REGEX = new RegExp(emojione.unicodeRegexp, 'g');
|
||||||
|
|
||||||
export const contentStateToHTML = stateToHTML;
|
const ZWS_CODE = 8203;
|
||||||
|
const ZWS = String.fromCharCode(ZWS_CODE); // zero width space
|
||||||
|
export function stateToMarkdown(state) {
|
||||||
|
return __stateToMarkdown(state)
|
||||||
|
.replace(
|
||||||
|
ZWS, // draft-js-export-markdown adds these
|
||||||
|
''); // this is *not* a zero width space, trust me :)
|
||||||
|
}
|
||||||
|
|
||||||
export function HTMLtoContentState(html: string): ContentState {
|
export const contentStateToHTML = (contentState: ContentState) => {
|
||||||
return ContentState.createFromBlockArray(convertFromHTML(html));
|
return stateToHTML(contentState, {
|
||||||
|
inlineStyles: {
|
||||||
|
UNDERLINE: {
|
||||||
|
element: 'u',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export function htmlToContentState(html: string): ContentState {
|
||||||
|
const blockArray = convertFromHTML(html).contentBlocks;
|
||||||
|
return ContentState.createFromBlockArray(blockArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
function unicodeToEmojiUri(str) {
|
function unicodeToEmojiUri(str) {
|
||||||
let replaceWith, unicode, alt;
|
let replaceWith, unicode, alt;
|
||||||
if ((!emojione.unicodeAlt) || (emojione.sprites)) {
|
if ((!emojione.unicodeAlt) || (emojione.sprites)) {
|
||||||
// if we are using the shortname as the alt tag then we need a reversed array to map unicode code point to shortnames
|
// if we are using the shortname as the alt tag then we need a reversed array to map unicode code point to shortnames
|
||||||
let mappedUnicode = emojione.mapUnicodeToShort();
|
const mappedUnicode = emojione.mapUnicodeToShort();
|
||||||
}
|
}
|
||||||
|
|
||||||
str = str.replace(emojione.regUnicode, function(unicodeChar) {
|
str = str.replace(emojione.regUnicode, function(unicodeChar) {
|
||||||
|
@ -48,8 +67,14 @@ function unicodeToEmojiUri(str) {
|
||||||
// if the unicodeChar doesnt exist just return the entire match
|
// if the unicodeChar doesnt exist just return the entire match
|
||||||
return unicodeChar;
|
return unicodeChar;
|
||||||
} else {
|
} else {
|
||||||
|
// Remove variant selector VS16 (explicitly emoji) as it is unnecessary and leads to an incorrect URL below
|
||||||
|
if (unicodeChar.length == 2 && unicodeChar[1] == '\ufe0f') {
|
||||||
|
unicodeChar = unicodeChar[0];
|
||||||
|
}
|
||||||
|
|
||||||
// get the unicode codepoint from the actual char
|
// get the unicode codepoint from the actual char
|
||||||
unicode = emojione.jsEscapeMap[unicodeChar];
|
unicode = emojione.jsEscapeMap[unicodeChar];
|
||||||
|
|
||||||
return emojione.imagePathSVG+unicode+'.svg'+emojione.cacheBustParam;
|
return emojione.imagePathSVG+unicode+'.svg'+emojione.cacheBustParam;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -71,14 +96,14 @@ function findWithRegex(regex, contentBlock: ContentBlock, callback: (start: numb
|
||||||
}
|
}
|
||||||
|
|
||||||
// Workaround for https://github.com/facebook/draft-js/issues/414
|
// Workaround for https://github.com/facebook/draft-js/issues/414
|
||||||
let emojiDecorator = {
|
const emojiDecorator = {
|
||||||
strategy: (contentBlock, callback) => {
|
strategy: (contentState, contentBlock, callback) => {
|
||||||
findWithRegex(EMOJI_REGEX, contentBlock, callback);
|
findWithRegex(EMOJI_REGEX, contentBlock, callback);
|
||||||
},
|
},
|
||||||
component: (props) => {
|
component: (props) => {
|
||||||
let uri = unicodeToEmojiUri(props.children[0].props.text);
|
const uri = unicodeToEmojiUri(props.children[0].props.text);
|
||||||
let shortname = emojione.toShort(props.children[0].props.text);
|
const shortname = emojione.toShort(props.children[0].props.text);
|
||||||
let style = {
|
const style = {
|
||||||
display: 'inline-block',
|
display: 'inline-block',
|
||||||
width: '1em',
|
width: '1em',
|
||||||
maxHeight: '1em',
|
maxHeight: '1em',
|
||||||
|
@ -87,7 +112,7 @@ let emojiDecorator = {
|
||||||
backgroundPosition: 'center center',
|
backgroundPosition: 'center center',
|
||||||
overflow: 'hidden',
|
overflow: 'hidden',
|
||||||
};
|
};
|
||||||
return (<span title={shortname} style={style}><span style={{opacity: 0}}>{props.children}</span></span>);
|
return (<span title={shortname} style={style}><span style={{opacity: 0}}>{ props.children }</span></span>);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -95,60 +120,35 @@ let emojiDecorator = {
|
||||||
* Returns a composite decorator which has access to provided scope.
|
* Returns a composite decorator which has access to provided scope.
|
||||||
*/
|
*/
|
||||||
export function getScopedRTDecorators(scope: any): CompositeDecorator {
|
export function getScopedRTDecorators(scope: any): CompositeDecorator {
|
||||||
let MemberAvatar = sdk.getComponent('avatars.MemberAvatar');
|
|
||||||
|
|
||||||
let usernameDecorator = {
|
|
||||||
strategy: (contentBlock, callback) => {
|
|
||||||
findWithRegex(USERNAME_REGEX, contentBlock, callback);
|
|
||||||
},
|
|
||||||
component: (props) => {
|
|
||||||
let member = scope.room.getMember(props.children[0].props.text);
|
|
||||||
// unused until we make these decorators immutable (autocomplete needed)
|
|
||||||
let name = member ? member.name : null;
|
|
||||||
let avatar = member ? <MemberAvatar member={member} width={16} height={16}/> : null;
|
|
||||||
return <span className="mx_UserPill">{avatar}{props.children}</span>;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let roomDecorator = {
|
|
||||||
strategy: (contentBlock, callback) => {
|
|
||||||
findWithRegex(ROOM_REGEX, contentBlock, callback);
|
|
||||||
},
|
|
||||||
component: (props) => {
|
|
||||||
return <span className="mx_RoomPill">{props.children}</span>;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO Re-enable usernameDecorator and roomDecorator
|
|
||||||
return [emojiDecorator];
|
return [emojiDecorator];
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getScopedMDDecorators(scope: any): CompositeDecorator {
|
export function getScopedMDDecorators(scope: any): CompositeDecorator {
|
||||||
let markdownDecorators = ['HR', 'BOLD', 'ITALIC', 'CODE', 'STRIKETHROUGH'].map(
|
const markdownDecorators = ['HR', 'BOLD', 'ITALIC', 'CODE', 'STRIKETHROUGH'].map(
|
||||||
(style) => ({
|
(style) => ({
|
||||||
strategy: (contentBlock, callback) => {
|
strategy: (contentState, contentBlock, callback) => {
|
||||||
return findWithRegex(MARKDOWN_REGEX[style], contentBlock, callback);
|
return findWithRegex(MARKDOWN_REGEX[style], contentBlock, callback);
|
||||||
},
|
},
|
||||||
component: (props) => (
|
component: (props) => (
|
||||||
<span className={"mx_MarkdownElement mx_Markdown_" + style}>
|
<span className={"mx_MarkdownElement mx_Markdown_" + style}>
|
||||||
{props.children}
|
{ props.children }
|
||||||
</span>
|
</span>
|
||||||
)
|
),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
markdownDecorators.push({
|
markdownDecorators.push({
|
||||||
strategy: (contentBlock, callback) => {
|
strategy: (contentState, contentBlock, callback) => {
|
||||||
return findWithRegex(MARKDOWN_REGEX.LINK, contentBlock, callback);
|
return findWithRegex(MARKDOWN_REGEX.LINK, contentBlock, callback);
|
||||||
},
|
},
|
||||||
component: (props) => (
|
component: (props) => (
|
||||||
<a href="#" className="mx_MarkdownElement mx_Markdown_LINK">
|
<a href="#" className="mx_MarkdownElement mx_Markdown_LINK">
|
||||||
{props.children}
|
{ props.children }
|
||||||
</a>
|
</a>
|
||||||
)
|
),
|
||||||
});
|
});
|
||||||
markdownDecorators.push(emojiDecorator);
|
// markdownDecorators.push(emojiDecorator);
|
||||||
|
// TODO Consider renabling "syntax highlighting" when we can do it properly
|
||||||
return markdownDecorators;
|
return [emojiDecorator];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -167,7 +167,7 @@ export function modifyText(contentState: ContentState, rangeToReplace: Selection
|
||||||
for (let currentKey = startKey;
|
for (let currentKey = startKey;
|
||||||
currentKey && currentKey !== endKey;
|
currentKey && currentKey !== endKey;
|
||||||
currentKey = contentState.getKeyAfter(currentKey)) {
|
currentKey = contentState.getKeyAfter(currentKey)) {
|
||||||
let blockText = getText(currentKey);
|
const blockText = getText(currentKey);
|
||||||
text += blockText.substring(startOffset, blockText.length);
|
text += blockText.substring(startOffset, blockText.length);
|
||||||
|
|
||||||
// from now on, we'll take whole blocks
|
// from now on, we'll take whole blocks
|
||||||
|
@ -188,7 +188,7 @@ export function modifyText(contentState: ContentState, rangeToReplace: Selection
|
||||||
export function selectionStateToTextOffsets(selectionState: SelectionState,
|
export function selectionStateToTextOffsets(selectionState: SelectionState,
|
||||||
contentBlocks: Array<ContentBlock>): {start: number, end: number} {
|
contentBlocks: Array<ContentBlock>): {start: number, end: number} {
|
||||||
let offset = 0, start = 0, end = 0;
|
let offset = 0, start = 0, end = 0;
|
||||||
for (let block of contentBlocks) {
|
for (const block of contentBlocks) {
|
||||||
if (selectionState.getStartKey() === block.getKey()) {
|
if (selectionState.getStartKey() === block.getKey()) {
|
||||||
start = offset + selectionState.getStartOffset();
|
start = offset + selectionState.getStartOffset();
|
||||||
}
|
}
|
||||||
|
@ -208,31 +208,36 @@ export function selectionStateToTextOffsets(selectionState: SelectionState,
|
||||||
export function textOffsetsToSelectionState({start, end}: SelectionRange,
|
export function textOffsetsToSelectionState({start, end}: SelectionRange,
|
||||||
contentBlocks: Array<ContentBlock>): SelectionState {
|
contentBlocks: Array<ContentBlock>): SelectionState {
|
||||||
let selectionState = SelectionState.createEmpty();
|
let selectionState = SelectionState.createEmpty();
|
||||||
|
// Subtract block lengths from `start` and `end` until they are less than the current
|
||||||
for (let block of contentBlocks) {
|
// block length (accounting for the NL at the end of each block). Set them to -1 to
|
||||||
let blockLength = block.getLength();
|
// indicate that the corresponding selection state has been determined.
|
||||||
|
for (const block of contentBlocks) {
|
||||||
if (start !== -1 && start < blockLength) {
|
const blockLength = block.getLength();
|
||||||
selectionState = selectionState.merge({
|
// -1 indicating that the position start position has been found
|
||||||
anchorKey: block.getKey(),
|
if (start !== -1) {
|
||||||
anchorOffset: start,
|
if (start < blockLength + 1) {
|
||||||
});
|
selectionState = selectionState.merge({
|
||||||
start = -1;
|
anchorKey: block.getKey(),
|
||||||
} else {
|
anchorOffset: start,
|
||||||
start -= blockLength;
|
});
|
||||||
|
start = -1; // selection state for the start calculated
|
||||||
|
} else {
|
||||||
|
start -= blockLength + 1; // +1 to account for newline between blocks
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
// -1 indicating that the position end position has been found
|
||||||
if (end !== -1 && end <= blockLength) {
|
if (end !== -1) {
|
||||||
selectionState = selectionState.merge({
|
if (end < blockLength + 1) {
|
||||||
focusKey: block.getKey(),
|
selectionState = selectionState.merge({
|
||||||
focusOffset: end,
|
focusKey: block.getKey(),
|
||||||
});
|
focusOffset: end,
|
||||||
end = -1;
|
});
|
||||||
} else {
|
end = -1; // selection state for the end calculated
|
||||||
end -= blockLength;
|
} else {
|
||||||
|
end -= blockLength + 1; // +1 to account for newline between blocks
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return selectionState;
|
return selectionState;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -249,7 +254,7 @@ export function attachImmutableEntitiesToEmoji(editorState: EditorState): Editor
|
||||||
const existingEntityKey = block.getEntityAt(start);
|
const existingEntityKey = block.getEntityAt(start);
|
||||||
if (existingEntityKey) {
|
if (existingEntityKey) {
|
||||||
// avoid manipulation in case the emoji already has an entity
|
// avoid manipulation in case the emoji already has an entity
|
||||||
const entity = Entity.get(existingEntityKey);
|
const entity = newContentState.getEntity(existingEntityKey);
|
||||||
if (entity && entity.get('type') === 'emoji') {
|
if (entity && entity.get('type') === 'emoji') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -259,7 +264,10 @@ export function attachImmutableEntitiesToEmoji(editorState: EditorState): Editor
|
||||||
.set('anchorOffset', start)
|
.set('anchorOffset', start)
|
||||||
.set('focusOffset', end);
|
.set('focusOffset', end);
|
||||||
const emojiText = plainText.substring(start, end);
|
const emojiText = plainText.substring(start, end);
|
||||||
const entityKey = Entity.create('emoji', 'IMMUTABLE', { emojiUnicode: emojiText });
|
newContentState = newContentState.createEntity(
|
||||||
|
'emoji', 'IMMUTABLE', { emojiUnicode: emojiText },
|
||||||
|
);
|
||||||
|
const entityKey = newContentState.getLastCreatedEntityKey();
|
||||||
newContentState = Modifier.replaceText(
|
newContentState = Modifier.replaceText(
|
||||||
newContentState,
|
newContentState,
|
||||||
selection,
|
selection,
|
||||||
|
@ -286,3 +294,14 @@ export function attachImmutableEntitiesToEmoji(editorState: EditorState): Editor
|
||||||
|
|
||||||
return editorState;
|
return editorState;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function hasMultiLineSelection(editorState: EditorState): boolean {
|
||||||
|
const selectionState = editorState.getSelection();
|
||||||
|
const anchorKey = selectionState.getAnchorKey();
|
||||||
|
const currentContent = editorState.getCurrentContent();
|
||||||
|
const currentContentBlock = currentContent.getBlockForKey(anchorKey);
|
||||||
|
const start = selectionState.getStartOffset();
|
||||||
|
const end = selectionState.getEndOffset();
|
||||||
|
const selectedText = currentContentBlock.getText().slice(start, end);
|
||||||
|
return selectedText.includes('\n');
|
||||||
|
}
|
||||||
|
|
35
src/Roles.js
Normal file
35
src/Roles.js
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Vector Creations 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 { _t } from './languageHandler';
|
||||||
|
|
||||||
|
export function levelRoleMap(usersDefault) {
|
||||||
|
return {
|
||||||
|
undefined: _t('Default'),
|
||||||
|
0: _t('Restricted'),
|
||||||
|
[usersDefault]: _t('Default'),
|
||||||
|
50: _t('Moderator'),
|
||||||
|
100: _t('Admin'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function textualPowerLevel(level, usersDefault) {
|
||||||
|
const LEVEL_ROLE_MAP = this.levelRoleMap(usersDefault);
|
||||||
|
if (LEVEL_ROLE_MAP[level]) {
|
||||||
|
return LEVEL_ROLE_MAP[level] + (level !== undefined ? ` (${level})` : ` (${usersDefault})`);
|
||||||
|
} else {
|
||||||
|
return level;
|
||||||
|
}
|
||||||
|
}
|
205
src/RoomInvite.js
Normal file
205
src/RoomInvite.js
Normal file
|
@ -0,0 +1,205 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 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 MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
import MultiInviter from './utils/MultiInviter';
|
||||||
|
import Modal from './Modal';
|
||||||
|
import { getAddressType } from './UserAddress';
|
||||||
|
import createRoom from './createRoom';
|
||||||
|
import sdk from './';
|
||||||
|
import dis from './dispatcher';
|
||||||
|
import DMRoomMap from './utils/DMRoomMap';
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
|
||||||
|
export function inviteToRoom(roomId, addr) {
|
||||||
|
const addrType = getAddressType(addr);
|
||||||
|
|
||||||
|
if (addrType == 'email') {
|
||||||
|
return MatrixClientPeg.get().inviteByEmail(roomId, addr);
|
||||||
|
} else if (addrType == 'mx-user-id') {
|
||||||
|
return MatrixClientPeg.get().invite(roomId, addr);
|
||||||
|
} else {
|
||||||
|
throw new Error('Unsupported address');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invites multiple addresses to a room
|
||||||
|
* Simpler interface to utils/MultiInviter but with
|
||||||
|
* no option to cancel.
|
||||||
|
*
|
||||||
|
* @param {string} roomId The ID of the room to invite to
|
||||||
|
* @param {string[]} addrs Array of strings of addresses to invite. May be matrix IDs or 3pids.
|
||||||
|
* @returns {Promise} Promise
|
||||||
|
*/
|
||||||
|
export function inviteMultipleToRoom(roomId, addrs) {
|
||||||
|
const inviter = new MultiInviter(roomId);
|
||||||
|
return inviter.invite(addrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function showStartChatInviteDialog() {
|
||||||
|
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
|
||||||
|
Modal.createTrackedDialog('Start a chat', '', AddressPickerDialog, {
|
||||||
|
title: _t('Start a chat'),
|
||||||
|
description: _t("Who would you like to communicate with?"),
|
||||||
|
placeholder: _t("Email, name or matrix ID"),
|
||||||
|
button: _t("Start Chat"),
|
||||||
|
onFinished: _onStartChatFinished,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function showRoomInviteDialog(roomId) {
|
||||||
|
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
|
||||||
|
Modal.createTrackedDialog('Chat Invite', '', AddressPickerDialog, {
|
||||||
|
title: _t('Invite new room members'),
|
||||||
|
description: _t('Who would you like to add to this room?'),
|
||||||
|
button: _t('Send Invites'),
|
||||||
|
placeholder: _t("Email, name or matrix ID"),
|
||||||
|
onFinished: (shouldInvite, addrs) => {
|
||||||
|
_onRoomInviteFinished(roomId, shouldInvite, addrs);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function _onStartChatFinished(shouldInvite, addrs) {
|
||||||
|
if (!shouldInvite) return;
|
||||||
|
|
||||||
|
const addrTexts = addrs.map((addr) => addr.address);
|
||||||
|
|
||||||
|
if (_isDmChat(addrTexts)) {
|
||||||
|
const rooms = _getDirectMessageRooms(addrTexts[0]);
|
||||||
|
if (rooms.length > 0) {
|
||||||
|
// A Direct Message room already exists for this user, so select a
|
||||||
|
// room from a list that is similar to the one in MemberInfo panel
|
||||||
|
const ChatCreateOrReuseDialog = sdk.getComponent("views.dialogs.ChatCreateOrReuseDialog");
|
||||||
|
const close = Modal.createTrackedDialog('Create or Reuse', '', ChatCreateOrReuseDialog, {
|
||||||
|
userId: addrTexts[0],
|
||||||
|
onNewDMClick: () => {
|
||||||
|
dis.dispatch({
|
||||||
|
action: 'start_chat',
|
||||||
|
user_id: addrTexts[0],
|
||||||
|
});
|
||||||
|
close(true);
|
||||||
|
},
|
||||||
|
onExistingRoomSelected: (roomId) => {
|
||||||
|
dis.dispatch({
|
||||||
|
action: 'view_room',
|
||||||
|
room_id: roomId,
|
||||||
|
});
|
||||||
|
close(true);
|
||||||
|
},
|
||||||
|
}).close;
|
||||||
|
} else {
|
||||||
|
// Start a new DM chat
|
||||||
|
createRoom({dmUserId: addrTexts[0]}).catch((err) => {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite user', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite user"),
|
||||||
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else if (addrTexts.length === 1) {
|
||||||
|
// Start a new DM chat
|
||||||
|
createRoom({dmUserId: addrTexts[0]}).catch((err) => {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite user', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite user"),
|
||||||
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Start multi user chat
|
||||||
|
let room;
|
||||||
|
createRoom().then((roomId) => {
|
||||||
|
room = MatrixClientPeg.get().getRoom(roomId);
|
||||||
|
return inviteMultipleToRoom(roomId, addrTexts);
|
||||||
|
}).then((addrs) => {
|
||||||
|
return _showAnyInviteErrors(addrs, room);
|
||||||
|
}).catch((err) => {
|
||||||
|
console.error(err.stack);
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite"),
|
||||||
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function _onRoomInviteFinished(roomId, shouldInvite, addrs) {
|
||||||
|
if (!shouldInvite) return;
|
||||||
|
|
||||||
|
const addrTexts = addrs.map((addr) => addr.address);
|
||||||
|
|
||||||
|
// Invite new users to a room
|
||||||
|
inviteMultipleToRoom(roomId, addrTexts).then((addrs) => {
|
||||||
|
const room = MatrixClientPeg.get().getRoom(roomId);
|
||||||
|
return _showAnyInviteErrors(addrs, room);
|
||||||
|
}).catch((err) => {
|
||||||
|
console.error(err.stack);
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite"),
|
||||||
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function _isDmChat(addrTexts) {
|
||||||
|
if (addrTexts.length === 1 && getAddressType(addrTexts[0]) === 'mx-user-id') {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function _showAnyInviteErrors(addrs, room) {
|
||||||
|
// Show user any errors
|
||||||
|
const errorList = [];
|
||||||
|
for (const addr of Object.keys(addrs)) {
|
||||||
|
if (addrs[addr] === "error") {
|
||||||
|
errorList.push(addr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errorList.length > 0) {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
Modal.createTrackedDialog('Failed to invite the following users to the room', '', ErrorDialog, {
|
||||||
|
title: _t("Failed to invite the following users to the %(roomName)s room:", {roomName: room.name}),
|
||||||
|
description: errorList.join(", "),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return addrs;
|
||||||
|
}
|
||||||
|
|
||||||
|
function _getDirectMessageRooms(addr) {
|
||||||
|
const dmRoomMap = new DMRoomMap(MatrixClientPeg.get());
|
||||||
|
const dmRooms = dmRoomMap.getDMRoomsForUserId(addr);
|
||||||
|
const rooms = [];
|
||||||
|
dmRooms.forEach((dmRoom) => {
|
||||||
|
const room = MatrixClientPeg.get().getRoom(dmRoom);
|
||||||
|
if (room) {
|
||||||
|
const me = room.getMember(MatrixClientPeg.get().credentials.userId);
|
||||||
|
if (me.membership == 'join') {
|
||||||
|
rooms.push(room);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return rooms;
|
||||||
|
}
|
||||||
|
|
|
@ -19,18 +19,17 @@ limitations under the License.
|
||||||
function tsOfNewestEvent(room) {
|
function tsOfNewestEvent(room) {
|
||||||
if (room.timeline.length) {
|
if (room.timeline.length) {
|
||||||
return room.timeline[room.timeline.length - 1].getTs();
|
return room.timeline[room.timeline.length - 1].getTs();
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
return Number.MAX_SAFE_INTEGER;
|
return Number.MAX_SAFE_INTEGER;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function mostRecentActivityFirst(roomList) {
|
function mostRecentActivityFirst(roomList) {
|
||||||
return roomList.sort(function(a,b) {
|
return roomList.sort(function(a, b) {
|
||||||
return tsOfNewestEvent(b) - tsOfNewestEvent(a);
|
return tsOfNewestEvent(b) - tsOfNewestEvent(a);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
mostRecentActivityFirst: mostRecentActivityFirst
|
mostRecentActivityFirst,
|
||||||
};
|
};
|
||||||
|
|
|
@ -16,7 +16,7 @@ limitations under the License.
|
||||||
|
|
||||||
import MatrixClientPeg from './MatrixClientPeg';
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
import PushProcessor from 'matrix-js-sdk/lib/pushprocessor';
|
import PushProcessor from 'matrix-js-sdk/lib/pushprocessor';
|
||||||
import q from 'q';
|
import Promise from 'bluebird';
|
||||||
|
|
||||||
export const ALL_MESSAGES_LOUD = 'all_messages_loud';
|
export const ALL_MESSAGES_LOUD = 'all_messages_loud';
|
||||||
export const ALL_MESSAGES = 'all_messages';
|
export const ALL_MESSAGES = 'all_messages';
|
||||||
|
@ -52,7 +52,7 @@ export function getRoomNotifsState(roomId) {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function setRoomNotifsState(roomId, newState) {
|
export function setRoomNotifsState(roomId, newState) {
|
||||||
if (newState == MUTE) {
|
if (newState === MUTE) {
|
||||||
return setRoomNotifsStateMuted(roomId);
|
return setRoomNotifsStateMuted(roomId);
|
||||||
} else {
|
} else {
|
||||||
return setRoomNotifsStateUnmuted(roomId, newState);
|
return setRoomNotifsStateUnmuted(roomId, newState);
|
||||||
|
@ -80,14 +80,14 @@ function setRoomNotifsStateMuted(roomId) {
|
||||||
kind: 'event_match',
|
kind: 'event_match',
|
||||||
key: 'room_id',
|
key: 'room_id',
|
||||||
pattern: roomId,
|
pattern: roomId,
|
||||||
}
|
},
|
||||||
],
|
],
|
||||||
actions: [
|
actions: [
|
||||||
'dont_notify',
|
'dont_notify',
|
||||||
]
|
],
|
||||||
}));
|
}));
|
||||||
|
|
||||||
return q.all(promises);
|
return Promise.all(promises);
|
||||||
}
|
}
|
||||||
|
|
||||||
function setRoomNotifsStateUnmuted(roomId, newState) {
|
function setRoomNotifsStateUnmuted(roomId, newState) {
|
||||||
|
@ -99,16 +99,16 @@ function setRoomNotifsStateUnmuted(roomId, newState) {
|
||||||
promises.push(cli.deletePushRule('global', 'override', overrideMuteRule.rule_id));
|
promises.push(cli.deletePushRule('global', 'override', overrideMuteRule.rule_id));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newState == 'all_messages') {
|
if (newState === 'all_messages') {
|
||||||
const roomRule = cli.getRoomPushRule('global', roomId);
|
const roomRule = cli.getRoomPushRule('global', roomId);
|
||||||
if (roomRule) {
|
if (roomRule) {
|
||||||
promises.push(cli.deletePushRule('global', 'room', roomRule.rule_id));
|
promises.push(cli.deletePushRule('global', 'room', roomRule.rule_id));
|
||||||
}
|
}
|
||||||
} else if (newState == 'mentions_only') {
|
} else if (newState === 'mentions_only') {
|
||||||
promises.push(cli.addPushRule('global', 'room', roomId, {
|
promises.push(cli.addPushRule('global', 'room', roomId, {
|
||||||
actions: [
|
actions: [
|
||||||
'dont_notify',
|
'dont_notify',
|
||||||
]
|
],
|
||||||
}));
|
}));
|
||||||
// https://matrix.org/jira/browse/SPEC-400
|
// https://matrix.org/jira/browse/SPEC-400
|
||||||
promises.push(cli.setPushRuleEnabled('global', 'room', roomId, true));
|
promises.push(cli.setPushRuleEnabled('global', 'room', roomId, true));
|
||||||
|
@ -119,14 +119,14 @@ function setRoomNotifsStateUnmuted(roomId, newState) {
|
||||||
{
|
{
|
||||||
set_tweak: 'sound',
|
set_tweak: 'sound',
|
||||||
value: 'default',
|
value: 'default',
|
||||||
}
|
},
|
||||||
]
|
],
|
||||||
}));
|
}));
|
||||||
// https://matrix.org/jira/browse/SPEC-400
|
// https://matrix.org/jira/browse/SPEC-400
|
||||||
promises.push(cli.setPushRuleEnabled('global', 'room', roomId, true));
|
promises.push(cli.setPushRuleEnabled('global', 'room', roomId, true));
|
||||||
}
|
}
|
||||||
|
|
||||||
return q.all(promises);
|
return Promise.all(promises);
|
||||||
}
|
}
|
||||||
|
|
||||||
function findOverrideMuteRule(roomId) {
|
function findOverrideMuteRule(roomId) {
|
||||||
|
@ -145,20 +145,10 @@ function isRuleForRoom(roomId, rule) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const cond = rule.conditions[0];
|
const cond = rule.conditions[0];
|
||||||
if (
|
return (cond.kind === 'event_match' && cond.key === 'room_id' && cond.pattern === roomId);
|
||||||
cond.kind == 'event_match' &&
|
|
||||||
cond.key == 'room_id' &&
|
|
||||||
cond.pattern == roomId
|
|
||||||
) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function isMuteRule(rule) {
|
function isMuteRule(rule) {
|
||||||
return (
|
return (rule.actions.length === 1 && rule.actions[0] === 'dont_notify');
|
||||||
rule.actions.length == 1 &&
|
|
||||||
rule.actions[0] == 'dont_notify'
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
59
src/Rooms.js
59
src/Rooms.js
|
@ -15,8 +15,7 @@ limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import MatrixClientPeg from './MatrixClientPeg';
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
import DMRoomMap from './utils/DMRoomMap';
|
import Promise from 'bluebird';
|
||||||
import q from 'q';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Given a room object, return the alias we should use for it,
|
* Given a room object, return the alias we should use for it,
|
||||||
|
@ -37,14 +36,14 @@ export function getOnlyOtherMember(room, me) {
|
||||||
|
|
||||||
if (joinedMembers.length === 2) {
|
if (joinedMembers.length === 2) {
|
||||||
return joinedMembers.filter(function(m) {
|
return joinedMembers.filter(function(m) {
|
||||||
return m.userId !== me.userId
|
return m.userId !== me.userId;
|
||||||
})[0];
|
})[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isConfCallRoom(room, me, conferenceHandler) {
|
function _isConfCallRoom(room, me, conferenceHandler) {
|
||||||
if (!conferenceHandler) return false;
|
if (!conferenceHandler) return false;
|
||||||
|
|
||||||
if (me.membership != "join") {
|
if (me.membership != "join") {
|
||||||
|
@ -59,12 +58,31 @@ export function isConfCallRoom(room, me, conferenceHandler) {
|
||||||
if (conferenceHandler.isConferenceUser(otherMember.userId)) {
|
if (conferenceHandler.isConferenceUser(otherMember.userId)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cache whether a room is a conference call. Assumes that rooms will always
|
||||||
|
// either will or will not be a conference call room.
|
||||||
|
const isConfCallRoomCache = {
|
||||||
|
// $roomId: bool
|
||||||
|
};
|
||||||
|
|
||||||
|
export function isConfCallRoom(room, me, conferenceHandler) {
|
||||||
|
if (isConfCallRoomCache[room.roomId] !== undefined) {
|
||||||
|
return isConfCallRoomCache[room.roomId];
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = _isConfCallRoom(room, me, conferenceHandler);
|
||||||
|
|
||||||
|
isConfCallRoomCache[room.roomId] = result;
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function looksLikeDirectMessageRoom(room, me) {
|
export function looksLikeDirectMessageRoom(room, me) {
|
||||||
if (me.membership == "join" || me.membership === "ban" ||
|
if (me.membership == "join" || me.membership === "ban" ||
|
||||||
(me.membership === "leave" && me.events.member.getSender() !== me.events.member.getStateKey()))
|
(me.membership === "leave" && me.events.member.getSender() !== me.events.member.getStateKey())) {
|
||||||
{
|
|
||||||
// Used to split rooms via tags
|
// Used to split rooms via tags
|
||||||
const tagNames = Object.keys(room.tags);
|
const tagNames = Object.keys(room.tags);
|
||||||
// Used for 1:1 direct chats
|
// Used for 1:1 direct chats
|
||||||
|
@ -79,6 +97,20 @@ export function looksLikeDirectMessageRoom(room, me) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function guessAndSetDMRoom(room, isDirect) {
|
||||||
|
let newTarget;
|
||||||
|
if (isDirect) {
|
||||||
|
const guessedTarget = guessDMRoomTarget(
|
||||||
|
room, room.getMember(MatrixClientPeg.get().credentials.userId),
|
||||||
|
);
|
||||||
|
newTarget = guessedTarget.userId;
|
||||||
|
} else {
|
||||||
|
newTarget = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return setDMRoom(room.roomId, newTarget);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Marks or unmarks the given room as being as a DM room.
|
* Marks or unmarks the given room as being as a DM room.
|
||||||
* @param {string} roomId The ID of the room to modify
|
* @param {string} roomId The ID of the room to modify
|
||||||
|
@ -89,7 +121,7 @@ export function looksLikeDirectMessageRoom(room, me) {
|
||||||
*/
|
*/
|
||||||
export function setDMRoom(roomId, userId) {
|
export function setDMRoom(roomId, userId) {
|
||||||
if (MatrixClientPeg.get().isGuest()) {
|
if (MatrixClientPeg.get().isGuest()) {
|
||||||
return q();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
const mDirectEvent = MatrixClientPeg.get().getAccountData('m.direct');
|
const mDirectEvent = MatrixClientPeg.get().getAccountData('m.direct');
|
||||||
|
@ -131,7 +163,18 @@ export function guessDMRoomTarget(room, me) {
|
||||||
let oldestTs;
|
let oldestTs;
|
||||||
let oldestUser;
|
let oldestUser;
|
||||||
|
|
||||||
// Pick the user who's been here longest (and isn't us)
|
// Pick the joined user who's been here longest (and isn't us),
|
||||||
|
for (const user of room.getJoinedMembers()) {
|
||||||
|
if (user.userId == me.userId) continue;
|
||||||
|
|
||||||
|
if (oldestTs === undefined || user.events.member.getTs() < oldestTs) {
|
||||||
|
oldestUser = user;
|
||||||
|
oldestTs = user.events.member.getTs();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (oldestUser) return oldestUser;
|
||||||
|
|
||||||
|
// if there are no joined members other than us, use the oldest member
|
||||||
for (const user of room.currentState.getMembers()) {
|
for (const user of room.currentState.getMembers()) {
|
||||||
if (user.userId == me.userId) continue;
|
if (user.userId == me.userId) continue;
|
||||||
|
|
||||||
|
|
104
src/RtsClient.js
Normal file
104
src/RtsClient.js
Normal file
|
@ -0,0 +1,104 @@
|
||||||
|
import 'whatwg-fetch';
|
||||||
|
|
||||||
|
let fetchFunction = fetch;
|
||||||
|
|
||||||
|
function checkStatus(response) {
|
||||||
|
if (!response.ok) {
|
||||||
|
return response.text().then((text) => {
|
||||||
|
throw new Error(text);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseJson(response) {
|
||||||
|
return response.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
function encodeQueryParams(params) {
|
||||||
|
return '?' + Object.keys(params).map((k) => {
|
||||||
|
return k + '=' + encodeURIComponent(params[k]);
|
||||||
|
}).join('&');
|
||||||
|
}
|
||||||
|
|
||||||
|
const request = (url, opts) => {
|
||||||
|
if (opts && opts.qs) {
|
||||||
|
url += encodeQueryParams(opts.qs);
|
||||||
|
delete opts.qs;
|
||||||
|
}
|
||||||
|
if (opts && opts.body) {
|
||||||
|
if (!opts.headers) {
|
||||||
|
opts.headers = {};
|
||||||
|
}
|
||||||
|
opts.body = JSON.stringify(opts.body);
|
||||||
|
opts.headers['Content-Type'] = 'application/json';
|
||||||
|
}
|
||||||
|
return fetchFunction(url, opts)
|
||||||
|
.then(checkStatus)
|
||||||
|
.then(parseJson);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
export default class RtsClient {
|
||||||
|
constructor(url) {
|
||||||
|
this._url = url;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTeamsConfig() {
|
||||||
|
return request(this._url + '/teams');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Track a referral with the Riot Team Server. This should be called once a referred
|
||||||
|
* user has been successfully registered.
|
||||||
|
* @param {string} referrer the user ID of one who referred the user to Riot.
|
||||||
|
* @param {string} sid the sign-up identity server session ID .
|
||||||
|
* @param {string} clientSecret the sign-up client secret.
|
||||||
|
* @returns {Promise} a promise that resolves to { team_token: 'sometoken' } upon
|
||||||
|
* success.
|
||||||
|
*/
|
||||||
|
trackReferral(referrer, sid, clientSecret) {
|
||||||
|
return request(this._url + '/register',
|
||||||
|
{
|
||||||
|
body: {
|
||||||
|
referrer: referrer,
|
||||||
|
session_id: sid,
|
||||||
|
client_secret: clientSecret,
|
||||||
|
},
|
||||||
|
method: 'POST',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
getTeam(teamToken) {
|
||||||
|
return request(this._url + '/teamConfiguration',
|
||||||
|
{
|
||||||
|
qs: {
|
||||||
|
team_token: teamToken,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Signal to the RTS that a login has occurred and that a user requires their team's
|
||||||
|
* token.
|
||||||
|
* @param {string} userId the user ID of the user who is a member of a team.
|
||||||
|
* @returns {Promise} a promise that resolves to { team_token: 'sometoken' } upon
|
||||||
|
* success.
|
||||||
|
*/
|
||||||
|
login(userId) {
|
||||||
|
return request(this._url + '/login',
|
||||||
|
{
|
||||||
|
qs: {
|
||||||
|
user_id: userId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// allow fetch to be replaced, for testing.
|
||||||
|
static setFetch(fn) {
|
||||||
|
fetchFunction = fn;
|
||||||
|
}
|
||||||
|
}
|
|
@ -14,11 +14,12 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var q = require("q");
|
import Promise from 'bluebird';
|
||||||
var request = require('browser-request');
|
import SettingsStore from "./settings/SettingsStore";
|
||||||
|
const request = require('browser-request');
|
||||||
|
|
||||||
var SdkConfig = require('./SdkConfig');
|
const SdkConfig = require('./SdkConfig');
|
||||||
var MatrixClientPeg = require('./MatrixClientPeg');
|
const MatrixClientPeg = require('./MatrixClientPeg');
|
||||||
|
|
||||||
class ScalarAuthClient {
|
class ScalarAuthClient {
|
||||||
|
|
||||||
|
@ -38,11 +39,53 @@ class ScalarAuthClient {
|
||||||
|
|
||||||
// Returns a scalar_token string
|
// Returns a scalar_token string
|
||||||
getScalarToken() {
|
getScalarToken() {
|
||||||
var tok = window.localStorage.getItem("mx_scalar_token");
|
const token = window.localStorage.getItem("mx_scalar_token");
|
||||||
if (tok) return q(tok);
|
|
||||||
|
|
||||||
// No saved token, so do the dance to get one. First, we
|
if (!token) {
|
||||||
// need an openid bearer token from the HS.
|
return this.registerForToken();
|
||||||
|
} else {
|
||||||
|
return this.validateToken(token).then(userId => {
|
||||||
|
const me = MatrixClientPeg.get().getUserId();
|
||||||
|
if (userId !== me) {
|
||||||
|
throw new Error("Scalar token is owned by someone else: " + me);
|
||||||
|
}
|
||||||
|
return token;
|
||||||
|
}).catch(err => {
|
||||||
|
console.error(err);
|
||||||
|
|
||||||
|
// Something went wrong - try to get a new token.
|
||||||
|
console.warn("Registering for new scalar token");
|
||||||
|
return this.registerForToken();
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
validateToken(token) {
|
||||||
|
let url = SdkConfig.get().integrations_rest_url + "/account";
|
||||||
|
|
||||||
|
const defer = Promise.defer();
|
||||||
|
request({
|
||||||
|
method: "GET",
|
||||||
|
uri: url,
|
||||||
|
qs: {scalar_token: token},
|
||||||
|
json: true,
|
||||||
|
}, (err, response, body) => {
|
||||||
|
if (err) {
|
||||||
|
defer.reject(err);
|
||||||
|
} else if (response.statusCode / 100 !== 2) {
|
||||||
|
defer.reject({statusCode: response.statusCode});
|
||||||
|
} else if (!body || !body.user_id) {
|
||||||
|
defer.reject(new Error("Missing user_id in response"));
|
||||||
|
} else {
|
||||||
|
defer.resolve(body.user_id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return defer.promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
registerForToken() {
|
||||||
|
// Get openid bearer token from the HS as the first part of our dance
|
||||||
return MatrixClientPeg.get().getOpenIdToken().then((token_object) => {
|
return MatrixClientPeg.get().getOpenIdToken().then((token_object) => {
|
||||||
// Now we can send that to scalar and exchange it for a scalar token
|
// Now we can send that to scalar and exchange it for a scalar token
|
||||||
return this.exchangeForScalarToken(token_object);
|
return this.exchangeForScalarToken(token_object);
|
||||||
|
@ -53,9 +96,9 @@ class ScalarAuthClient {
|
||||||
}
|
}
|
||||||
|
|
||||||
exchangeForScalarToken(openid_token_object) {
|
exchangeForScalarToken(openid_token_object) {
|
||||||
var defer = q.defer();
|
const defer = Promise.defer();
|
||||||
|
|
||||||
var scalar_rest_url = SdkConfig.get().integrations_rest_url;
|
const scalar_rest_url = SdkConfig.get().integrations_rest_url;
|
||||||
request({
|
request({
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
uri: scalar_rest_url+'/register',
|
uri: scalar_rest_url+'/register',
|
||||||
|
@ -76,10 +119,46 @@ class ScalarAuthClient {
|
||||||
return defer.promise;
|
return defer.promise;
|
||||||
}
|
}
|
||||||
|
|
||||||
getScalarInterfaceUrlForRoom(roomId) {
|
getScalarPageTitle(url) {
|
||||||
var url = SdkConfig.get().integrations_ui_url;
|
const defer = Promise.defer();
|
||||||
|
|
||||||
|
let scalarPageLookupUrl = SdkConfig.get().integrations_rest_url + '/widgets/title_lookup';
|
||||||
|
scalarPageLookupUrl = this.getStarterLink(scalarPageLookupUrl);
|
||||||
|
scalarPageLookupUrl += '&curl=' + encodeURIComponent(url);
|
||||||
|
request({
|
||||||
|
method: 'GET',
|
||||||
|
uri: scalarPageLookupUrl,
|
||||||
|
json: true,
|
||||||
|
}, (err, response, body) => {
|
||||||
|
if (err) {
|
||||||
|
defer.reject(err);
|
||||||
|
} else if (response.statusCode / 100 !== 2) {
|
||||||
|
defer.reject({statusCode: response.statusCode});
|
||||||
|
} else if (!body) {
|
||||||
|
defer.reject(new Error("Missing page title in response"));
|
||||||
|
} else {
|
||||||
|
let title = "";
|
||||||
|
if (body.page_title_cache_item && body.page_title_cache_item.cached_title) {
|
||||||
|
title = body.page_title_cache_item.cached_title;
|
||||||
|
}
|
||||||
|
defer.resolve(title);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return defer.promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
getScalarInterfaceUrlForRoom(roomId, screen, id) {
|
||||||
|
let url = SdkConfig.get().integrations_ui_url;
|
||||||
url += "?scalar_token=" + encodeURIComponent(this.scalarToken);
|
url += "?scalar_token=" + encodeURIComponent(this.scalarToken);
|
||||||
url += "&room_id=" + encodeURIComponent(roomId);
|
url += "&room_id=" + encodeURIComponent(roomId);
|
||||||
|
url += "&theme=" + encodeURIComponent(SettingsStore.getValue("theme"));
|
||||||
|
if (id) {
|
||||||
|
url += '&integ_id=' + encodeURIComponent(id);
|
||||||
|
}
|
||||||
|
if (screen) {
|
||||||
|
url += '&screen=' + encodeURIComponent(screen);
|
||||||
|
}
|
||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,4 +168,3 @@ class ScalarAuthClient {
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = ScalarAuthClient;
|
module.exports = ScalarAuthClient;
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2016 OpenMarket Ltd
|
Copyright 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -17,7 +18,7 @@ limitations under the License.
|
||||||
/*
|
/*
|
||||||
Listens for incoming postMessage requests from the integrations UI URL. The following API is exposed:
|
Listens for incoming postMessage requests from the integrations UI URL. The following API is exposed:
|
||||||
{
|
{
|
||||||
action: "invite" | "membership_state" | "bot_options" | "set_bot_options",
|
action: "invite" | "membership_state" | "bot_options" | "set_bot_options" | etc... ,
|
||||||
room_id: $ROOM_ID,
|
room_id: $ROOM_ID,
|
||||||
user_id: $USER_ID
|
user_id: $USER_ID
|
||||||
// additional request fields
|
// additional request fields
|
||||||
|
@ -94,6 +95,115 @@ Example:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get_membership_count
|
||||||
|
--------------------
|
||||||
|
Get the number of joined users in the room.
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- room_id is the room to get the count in.
|
||||||
|
Response:
|
||||||
|
78
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
action: "get_membership_count",
|
||||||
|
room_id: "!foo:bar",
|
||||||
|
response: 78
|
||||||
|
}
|
||||||
|
|
||||||
|
can_send_event
|
||||||
|
--------------
|
||||||
|
Check if the client can send the given event into the given room. If the client
|
||||||
|
is unable to do this, an error response is returned instead of 'response: false'.
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- room_id is the room to do the check in.
|
||||||
|
- event_type is the event type which will be sent.
|
||||||
|
- is_state is true if the event to be sent is a state event.
|
||||||
|
Response:
|
||||||
|
true
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
action: "can_send_event",
|
||||||
|
is_state: false,
|
||||||
|
event_type: "m.room.message",
|
||||||
|
room_id: "!foo:bar",
|
||||||
|
response: true
|
||||||
|
}
|
||||||
|
|
||||||
|
set_widget
|
||||||
|
----------
|
||||||
|
Set a new widget in the room. Clobbers based on the ID.
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- `room_id` (String) is the room to set the widget in.
|
||||||
|
- `widget_id` (String) is the ID of the widget to add (or replace if it already exists).
|
||||||
|
It can be an arbitrary UTF8 string and is purely for distinguishing between widgets.
|
||||||
|
- `url` (String) is the URL that clients should load in an iframe to run the widget.
|
||||||
|
All widgets must have a valid URL. If the URL is `null` (not `undefined`), the
|
||||||
|
widget will be removed from the room.
|
||||||
|
- `type` (String) is the type of widget, which is provided as a hint for matrix clients so they
|
||||||
|
can configure/lay out the widget in different ways. All widgets must have a type.
|
||||||
|
- `name` (String) is an optional human-readable string about the widget.
|
||||||
|
- `data` (Object) is some optional data about the widget, and can contain arbitrary key/value pairs.
|
||||||
|
Response:
|
||||||
|
{
|
||||||
|
success: true
|
||||||
|
}
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
action: "set_widget",
|
||||||
|
room_id: "!foo:bar",
|
||||||
|
widget_id: "abc123",
|
||||||
|
url: "http://widget.url",
|
||||||
|
type: "example",
|
||||||
|
response: {
|
||||||
|
success: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get_widgets
|
||||||
|
-----------
|
||||||
|
Get a list of all widgets in the room. The response is an array
|
||||||
|
of state events.
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- `room_id` (String) is the room to get the widgets in.
|
||||||
|
Response:
|
||||||
|
[
|
||||||
|
{
|
||||||
|
type: "im.vector.modular.widgets",
|
||||||
|
state_key: "wid1",
|
||||||
|
content: {
|
||||||
|
type: "grafana",
|
||||||
|
url: "https://grafanaurl",
|
||||||
|
name: "dashboard",
|
||||||
|
data: {key: "val"}
|
||||||
|
}
|
||||||
|
room_id: “!foo:bar”,
|
||||||
|
sender: "@alice:localhost"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
action: "get_widgets",
|
||||||
|
room_id: "!foo:bar",
|
||||||
|
response: [
|
||||||
|
{
|
||||||
|
type: "im.vector.modular.widgets",
|
||||||
|
state_key: "wid1",
|
||||||
|
content: {
|
||||||
|
type: "grafana",
|
||||||
|
url: "https://grafanaurl",
|
||||||
|
name: "dashboard",
|
||||||
|
data: {key: "val"}
|
||||||
|
}
|
||||||
|
room_id: “!foo:bar”,
|
||||||
|
sender: "@alice:localhost"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
membership_state AND bot_options
|
membership_state AND bot_options
|
||||||
--------------------------------
|
--------------------------------
|
||||||
Get the content of the "m.room.member" or "m.room.bot.options" state event respectively.
|
Get the content of the "m.room.member" or "m.room.bot.options" state event respectively.
|
||||||
|
@ -125,6 +235,7 @@ const SdkConfig = require('./SdkConfig');
|
||||||
const MatrixClientPeg = require("./MatrixClientPeg");
|
const MatrixClientPeg = require("./MatrixClientPeg");
|
||||||
const MatrixEvent = require("matrix-js-sdk").MatrixEvent;
|
const MatrixEvent = require("matrix-js-sdk").MatrixEvent;
|
||||||
const dis = require("./dispatcher");
|
const dis = require("./dispatcher");
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
|
||||||
function sendResponse(event, res) {
|
function sendResponse(event, res) {
|
||||||
const data = JSON.parse(JSON.stringify(event.data));
|
const data = JSON.parse(JSON.stringify(event.data));
|
||||||
|
@ -150,7 +261,7 @@ function inviteUser(event, roomId, userId) {
|
||||||
console.log(`Received request to invite ${userId} into room ${roomId}`);
|
console.log(`Received request to invite ${userId} into room ${roomId}`);
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
if (!client) {
|
if (!client) {
|
||||||
sendError(event, "You need to be logged in.");
|
sendError(event, _t('You need to be logged in.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const room = client.getRoom(roomId);
|
const room = client.getRoom(roomId);
|
||||||
|
@ -170,10 +281,107 @@ function inviteUser(event, roomId, userId) {
|
||||||
success: true,
|
success: true,
|
||||||
});
|
});
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
sendError(event, "You need to be able to invite users to do that.", err);
|
sendError(event, _t('You need to be able to invite users to do that.'), err);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function setWidget(event, roomId) {
|
||||||
|
const widgetId = event.data.widget_id;
|
||||||
|
const widgetType = event.data.type;
|
||||||
|
const widgetUrl = event.data.url;
|
||||||
|
const widgetName = event.data.name; // optional
|
||||||
|
const widgetData = event.data.data; // optional
|
||||||
|
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
if (!client) {
|
||||||
|
sendError(event, _t('You need to be logged in.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// both adding/removing widgets need these checks
|
||||||
|
if (!widgetId || widgetUrl === undefined) {
|
||||||
|
sendError(event, _t("Unable to create widget."), new Error("Missing required widget fields."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (widgetUrl !== null) { // if url is null it is being deleted, don't need to check name/type/etc
|
||||||
|
// check types of fields
|
||||||
|
if (widgetName !== undefined && typeof widgetName !== 'string') {
|
||||||
|
sendError(event, _t("Unable to create widget."), new Error("Optional field 'name' must be a string."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (widgetData !== undefined && !(widgetData instanceof Object)) {
|
||||||
|
sendError(event, _t("Unable to create widget."), new Error("Optional field 'data' must be an Object."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (typeof widgetType !== 'string') {
|
||||||
|
sendError(event, _t("Unable to create widget."), new Error("Field 'type' must be a string."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (typeof widgetUrl !== 'string') {
|
||||||
|
sendError(event, _t("Unable to create widget."), new Error("Field 'url' must be a string or null."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let content = {
|
||||||
|
type: widgetType,
|
||||||
|
url: widgetUrl,
|
||||||
|
name: widgetName,
|
||||||
|
data: widgetData,
|
||||||
|
};
|
||||||
|
if (widgetUrl === null) { // widget is being deleted
|
||||||
|
content = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
client.sendStateEvent(roomId, "im.vector.modular.widgets", content, widgetId).done(() => {
|
||||||
|
sendResponse(event, {
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
}, (err) => {
|
||||||
|
sendError(event, _t('Failed to send request.'), err);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getWidgets(event, roomId) {
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
if (!client) {
|
||||||
|
sendError(event, _t('You need to be logged in.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const room = client.getRoom(roomId);
|
||||||
|
if (!room) {
|
||||||
|
sendError(event, _t('This room is not recognised.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const stateEvents = room.currentState.getStateEvents("im.vector.modular.widgets");
|
||||||
|
// Only return widgets which have required fields
|
||||||
|
const widgetStateEvents = [];
|
||||||
|
stateEvents.forEach((ev) => {
|
||||||
|
if (ev.getContent().type && ev.getContent().url) {
|
||||||
|
widgetStateEvents.push(ev.event); // return the raw event
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
sendResponse(event, widgetStateEvents);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getRoomEncState(event, roomId) {
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
if (!client) {
|
||||||
|
sendError(event, _t('You need to be logged in.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const room = client.getRoom(roomId);
|
||||||
|
if (!room) {
|
||||||
|
sendError(event, _t('This room is not recognised.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const roomIsEncrypted = MatrixClientPeg.get().isRoomEncrypted(roomId);
|
||||||
|
|
||||||
|
sendResponse(event, roomIsEncrypted);
|
||||||
|
}
|
||||||
|
|
||||||
function setPlumbingState(event, roomId, status) {
|
function setPlumbingState(event, roomId, status) {
|
||||||
if (typeof status !== 'string') {
|
if (typeof status !== 'string') {
|
||||||
throw new Error('Plumbing state status should be a string');
|
throw new Error('Plumbing state status should be a string');
|
||||||
|
@ -181,15 +389,15 @@ function setPlumbingState(event, roomId, status) {
|
||||||
console.log(`Received request to set plumbing state to status "${status}" in room ${roomId}`);
|
console.log(`Received request to set plumbing state to status "${status}" in room ${roomId}`);
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
if (!client) {
|
if (!client) {
|
||||||
sendError(event, "You need to be logged in.");
|
sendError(event, _t('You need to be logged in.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
client.sendStateEvent(roomId, "m.room.plumbing", { status : status }).done(() => {
|
client.sendStateEvent(roomId, "m.room.plumbing", { status: status }).done(() => {
|
||||||
sendResponse(event, {
|
sendResponse(event, {
|
||||||
success: true,
|
success: true,
|
||||||
});
|
});
|
||||||
}, (err) => {
|
}, (err) => {
|
||||||
sendError(event, err.message ? err.message : "Failed to send request.", err);
|
sendError(event, err.message ? err.message : _t('Failed to send request.'), err);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -197,7 +405,7 @@ function setBotOptions(event, roomId, userId) {
|
||||||
console.log(`Received request to set options for bot ${userId} in room ${roomId}`);
|
console.log(`Received request to set options for bot ${userId} in room ${roomId}`);
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
if (!client) {
|
if (!client) {
|
||||||
sendError(event, "You need to be logged in.");
|
sendError(event, _t('You need to be logged in.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
client.sendStateEvent(roomId, "m.room.bot.options", event.data.content, "_" + userId).done(() => {
|
client.sendStateEvent(roomId, "m.room.bot.options", event.data.content, "_" + userId).done(() => {
|
||||||
|
@ -205,29 +413,29 @@ function setBotOptions(event, roomId, userId) {
|
||||||
success: true,
|
success: true,
|
||||||
});
|
});
|
||||||
}, (err) => {
|
}, (err) => {
|
||||||
sendError(event, err.message ? err.message : "Failed to send request.", err);
|
sendError(event, err.message ? err.message : _t('Failed to send request.'), err);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function setBotPower(event, roomId, userId, level) {
|
function setBotPower(event, roomId, userId, level) {
|
||||||
if (!(Number.isInteger(level) && level >= 0)) {
|
if (!(Number.isInteger(level) && level >= 0)) {
|
||||||
sendError(event, "Power level must be positive integer.");
|
sendError(event, _t('Power level must be positive integer.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(`Received request to set power level to ${level} for bot ${userId} in room ${roomId}.`);
|
console.log(`Received request to set power level to ${level} for bot ${userId} in room ${roomId}.`);
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
if (!client) {
|
if (!client) {
|
||||||
sendError(event, "You need to be logged in.");
|
sendError(event, _t('You need to be logged in.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
client.getStateEvent(roomId, "m.room.power_levels", "").then((powerLevels) => {
|
client.getStateEvent(roomId, "m.room.power_levels", "").then((powerLevels) => {
|
||||||
let powerEvent = new MatrixEvent(
|
const powerEvent = new MatrixEvent(
|
||||||
{
|
{
|
||||||
type: "m.room.power_levels",
|
type: "m.room.power_levels",
|
||||||
content: powerLevels,
|
content: powerLevels,
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
client.setPowerLevel(roomId, userId, level, powerEvent).done(() => {
|
client.setPowerLevel(roomId, userId, level, powerEvent).done(() => {
|
||||||
|
@ -235,7 +443,7 @@ function setBotPower(event, roomId, userId, level) {
|
||||||
success: true,
|
success: true,
|
||||||
});
|
});
|
||||||
}, (err) => {
|
}, (err) => {
|
||||||
sendError(event, err.message ? err.message : "Failed to send request.", err);
|
sendError(event, err.message ? err.message : _t('Failed to send request.'), err);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -255,15 +463,65 @@ function botOptions(event, roomId, userId) {
|
||||||
returnStateEvent(event, roomId, "m.room.bot.options", "_" + userId);
|
returnStateEvent(event, roomId, "m.room.bot.options", "_" + userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
function returnStateEvent(event, roomId, eventType, stateKey) {
|
function getMembershipCount(event, roomId) {
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
if (!client) {
|
if (!client) {
|
||||||
sendError(event, "You need to be logged in.");
|
sendError(event, _t('You need to be logged in.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const room = client.getRoom(roomId);
|
const room = client.getRoom(roomId);
|
||||||
if (!room) {
|
if (!room) {
|
||||||
sendError(event, "This room is not recognised.");
|
sendError(event, _t('This room is not recognised.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const count = room.getJoinedMembers().length;
|
||||||
|
sendResponse(event, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
function canSendEvent(event, roomId) {
|
||||||
|
const evType = "" + event.data.event_type; // force stringify
|
||||||
|
const isState = Boolean(event.data.is_state);
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
if (!client) {
|
||||||
|
sendError(event, _t('You need to be logged in.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const room = client.getRoom(roomId);
|
||||||
|
if (!room) {
|
||||||
|
sendError(event, _t('This room is not recognised.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const me = client.credentials.userId;
|
||||||
|
const member = room.getMember(me);
|
||||||
|
if (!member || member.membership !== "join") {
|
||||||
|
sendError(event, _t('You are not in this room.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let canSend = false;
|
||||||
|
if (isState) {
|
||||||
|
canSend = room.currentState.maySendStateEvent(evType, me);
|
||||||
|
} else {
|
||||||
|
canSend = room.currentState.maySendEvent(evType, me);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!canSend) {
|
||||||
|
sendError(event, _t('You do not have permission to do that in this room.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
sendResponse(event, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function returnStateEvent(event, roomId, eventType, stateKey) {
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
if (!client) {
|
||||||
|
sendError(event, _t('You need to be logged in.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const room = client.getRoom(roomId);
|
||||||
|
if (!room) {
|
||||||
|
sendError(event, _t('This room is not recognised.'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const stateEvent = room.currentState.getStateEvents(eventType, stateKey);
|
const stateEvent = room.currentState.getStateEvents(eventType, stateKey);
|
||||||
|
@ -274,8 +532,8 @@ function returnStateEvent(event, roomId, eventType, stateKey) {
|
||||||
sendResponse(event, stateEvent.getContent());
|
sendResponse(event, stateEvent.getContent());
|
||||||
}
|
}
|
||||||
|
|
||||||
var currentRoomId = null;
|
let currentRoomId = null;
|
||||||
var currentRoomAlias = null;
|
let currentRoomAlias = null;
|
||||||
|
|
||||||
// Listen for when a room is viewed
|
// Listen for when a room is viewed
|
||||||
dis.register(onAction);
|
dis.register(onAction);
|
||||||
|
@ -299,8 +557,16 @@ const onMessage = function(event) {
|
||||||
//
|
//
|
||||||
// All strings start with the empty string, so for sanity return if the length
|
// All strings start with the empty string, so for sanity return if the length
|
||||||
// of the event origin is 0.
|
// of the event origin is 0.
|
||||||
let url = SdkConfig.get().integrations_ui_url;
|
//
|
||||||
if (event.origin.length === 0 || !url.startsWith(event.origin)) {
|
// TODO -- Scalar postMessage API should be namespaced with event.data.api field
|
||||||
|
// Fix following "if" statement to respond only to specific API messages.
|
||||||
|
const url = SdkConfig.get().integrations_ui_url;
|
||||||
|
if (
|
||||||
|
event.origin.length === 0 ||
|
||||||
|
!url.startsWith(event.origin) ||
|
||||||
|
!event.data.action ||
|
||||||
|
event.data.api // Ignore messages with specific API set
|
||||||
|
) {
|
||||||
return; // don't log this - debugging APIs like to spam postMessage which floods the log otherwise
|
return; // don't log this - debugging APIs like to spam postMessage which floods the log otherwise
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -313,13 +579,13 @@ const onMessage = function(event) {
|
||||||
const roomId = event.data.room_id;
|
const roomId = event.data.room_id;
|
||||||
const userId = event.data.user_id;
|
const userId = event.data.user_id;
|
||||||
if (!roomId) {
|
if (!roomId) {
|
||||||
sendError(event, "Missing room_id in request");
|
sendError(event, _t('Missing room_id in request'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let promise = Promise.resolve(currentRoomId);
|
let promise = Promise.resolve(currentRoomId);
|
||||||
if (!currentRoomId) {
|
if (!currentRoomId) {
|
||||||
if (!currentRoomAlias) {
|
if (!currentRoomAlias) {
|
||||||
sendError(event, "Must be viewing a room");
|
sendError(event, _t('Must be viewing a room'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// no room ID but there is an alias, look it up.
|
// no room ID but there is an alias, look it up.
|
||||||
|
@ -331,21 +597,36 @@ const onMessage = function(event) {
|
||||||
|
|
||||||
promise.then((viewingRoomId) => {
|
promise.then((viewingRoomId) => {
|
||||||
if (roomId !== viewingRoomId) {
|
if (roomId !== viewingRoomId) {
|
||||||
sendError(event, "Room " + roomId + " not visible");
|
sendError(event, _t('Room %(roomId)s not visible', {roomId: roomId}));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Getting join rules does not require userId
|
// These APIs don't require userId
|
||||||
if (event.data.action === "join_rules_state") {
|
if (event.data.action === "join_rules_state") {
|
||||||
getJoinRules(event, roomId);
|
getJoinRules(event, roomId);
|
||||||
return;
|
return;
|
||||||
} else if (event.data.action === "set_plumbing_state") {
|
} else if (event.data.action === "set_plumbing_state") {
|
||||||
setPlumbingState(event, roomId, event.data.status);
|
setPlumbingState(event, roomId, event.data.status);
|
||||||
return;
|
return;
|
||||||
|
} else if (event.data.action === "get_membership_count") {
|
||||||
|
getMembershipCount(event, roomId);
|
||||||
|
return;
|
||||||
|
} else if (event.data.action === "set_widget") {
|
||||||
|
setWidget(event, roomId);
|
||||||
|
return;
|
||||||
|
} else if (event.data.action === "get_widgets") {
|
||||||
|
getWidgets(event, roomId);
|
||||||
|
return;
|
||||||
|
} else if (event.data.action === "get_room_enc_state") {
|
||||||
|
getRoomEncState(event, roomId);
|
||||||
|
return;
|
||||||
|
} else if (event.data.action === "can_send_event") {
|
||||||
|
canSendEvent(event, roomId);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!userId) {
|
if (!userId) {
|
||||||
sendError(event, "Missing user_id in request");
|
sendError(event, _t('Missing user_id in request'));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
switch (event.data.action) {
|
switch (event.data.action) {
|
||||||
|
@ -370,16 +651,31 @@ const onMessage = function(event) {
|
||||||
}
|
}
|
||||||
}, (err) => {
|
}, (err) => {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
sendError(event, "Failed to lookup current room.");
|
sendError(event, _t('Failed to lookup current room') + '.');
|
||||||
})
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let listenerCount = 0;
|
||||||
module.exports = {
|
module.exports = {
|
||||||
startListening: function() {
|
startListening: function() {
|
||||||
window.addEventListener("message", onMessage, false);
|
if (listenerCount === 0) {
|
||||||
|
window.addEventListener("message", onMessage, false);
|
||||||
|
}
|
||||||
|
listenerCount += 1;
|
||||||
},
|
},
|
||||||
|
|
||||||
stopListening: function() {
|
stopListening: function() {
|
||||||
window.removeEventListener("message", onMessage);
|
listenerCount -= 1;
|
||||||
|
if (listenerCount === 0) {
|
||||||
|
window.removeEventListener("message", onMessage);
|
||||||
|
}
|
||||||
|
if (listenerCount < 0) {
|
||||||
|
// Make an error so we get a stack trace
|
||||||
|
const e = new Error(
|
||||||
|
"ScalarMessaging: mismatched startListening / stopListening detected." +
|
||||||
|
" Negative count",
|
||||||
|
);
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -14,22 +14,31 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var DEFAULTS = {
|
const DEFAULTS = {
|
||||||
// URL to a page we show in an iframe to configure integrations
|
// URL to a page we show in an iframe to configure integrations
|
||||||
integrations_ui_url: "https://scalar.vector.im/",
|
integrations_ui_url: "https://scalar.vector.im/",
|
||||||
// Base URL to the REST interface of the integrations server
|
// Base URL to the REST interface of the integrations server
|
||||||
integrations_rest_url: "https://scalar.vector.im/api",
|
integrations_rest_url: "https://scalar.vector.im/api",
|
||||||
|
// Where to send bug reports. If not specified, bugs cannot be sent.
|
||||||
|
bug_report_endpoint_url: null,
|
||||||
|
|
||||||
|
piwik: {
|
||||||
|
url: "https://piwik.riot.im/",
|
||||||
|
whitelistedHSUrls: ["https://matrix.org"],
|
||||||
|
whitelistedISUrls: ["https://vector.im", "https://matrix.org"],
|
||||||
|
siteId: 1,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
class SdkConfig {
|
class SdkConfig {
|
||||||
|
|
||||||
static get() {
|
static get() {
|
||||||
return global.mxReactSdkConfig;
|
return global.mxReactSdkConfig || {};
|
||||||
}
|
}
|
||||||
|
|
||||||
static put(cfg) {
|
static put(cfg) {
|
||||||
var defaultKeys = Object.keys(DEFAULTS);
|
const defaultKeys = Object.keys(DEFAULTS);
|
||||||
for (var i = 0; i < defaultKeys.length; ++i) {
|
for (let i = 0; i < defaultKeys.length; ++i) {
|
||||||
if (cfg[defaultKeys[i]] === undefined) {
|
if (cfg[defaultKeys[i]] === undefined) {
|
||||||
cfg[defaultKeys[i]] = DEFAULTS[defaultKeys[i]];
|
cfg[defaultKeys[i]] = DEFAULTS[defaultKeys[i]];
|
||||||
}
|
}
|
||||||
|
@ -43,3 +52,4 @@ class SdkConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = SdkConfig;
|
module.exports = SdkConfig;
|
||||||
|
module.exports.DEFAULTS = DEFAULTS;
|
||||||
|
|
451
src/Signup.js
451
src/Signup.js
|
@ -1,451 +0,0 @@
|
||||||
"use strict";
|
|
||||||
|
|
||||||
import Matrix from "matrix-js-sdk";
|
|
||||||
|
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
|
||||||
var SignupStages = require("./SignupStages");
|
|
||||||
var dis = require("./dispatcher");
|
|
||||||
var q = require("q");
|
|
||||||
var url = require("url");
|
|
||||||
|
|
||||||
const EMAIL_STAGE_TYPE = "m.login.email.identity";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A base class for common functionality between Registration and Login e.g.
|
|
||||||
* storage of HS/IS URLs.
|
|
||||||
*/
|
|
||||||
class Signup {
|
|
||||||
constructor(hsUrl, isUrl, opts) {
|
|
||||||
this._hsUrl = hsUrl;
|
|
||||||
this._isUrl = isUrl;
|
|
||||||
this._defaultDeviceDisplayName = opts.defaultDeviceDisplayName;
|
|
||||||
}
|
|
||||||
|
|
||||||
getHomeserverUrl() {
|
|
||||||
return this._hsUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
getIdentityServerUrl() {
|
|
||||||
return this._isUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
setHomeserverUrl(hsUrl) {
|
|
||||||
this._hsUrl = hsUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
setIdentityServerUrl(isUrl) {
|
|
||||||
this._isUrl = isUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a temporary MatrixClient, which can be used for login or register
|
|
||||||
* requests.
|
|
||||||
*/
|
|
||||||
_createTemporaryClient() {
|
|
||||||
return Matrix.createClient({
|
|
||||||
baseUrl: this._hsUrl,
|
|
||||||
idBaseUrl: this._isUrl,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Registration logic class
|
|
||||||
* This exists for the lifetime of a user's attempt to register an account,
|
|
||||||
* so if their registration attempt fails for whatever reason and they
|
|
||||||
* try again, call register() on the same instance again.
|
|
||||||
*
|
|
||||||
* TODO: parts of this overlap heavily with InteractiveAuth in the js-sdk. It
|
|
||||||
* would be nice to make use of that rather than rolling our own version of it.
|
|
||||||
*/
|
|
||||||
class Register extends Signup {
|
|
||||||
constructor(hsUrl, isUrl, opts) {
|
|
||||||
super(hsUrl, isUrl, opts);
|
|
||||||
this.setStep("START");
|
|
||||||
this.data = null; // from the server
|
|
||||||
// random other stuff (e.g. query params, NOT params from the server)
|
|
||||||
this.params = {};
|
|
||||||
this.credentials = null;
|
|
||||||
this.activeStage = null;
|
|
||||||
this.registrationPromise = null;
|
|
||||||
// These values MUST be undefined else we'll send "username: null" which
|
|
||||||
// will error on Synapse rather than having the key absent.
|
|
||||||
this.username = undefined; // desired
|
|
||||||
this.email = undefined; // desired
|
|
||||||
this.password = undefined; // desired
|
|
||||||
}
|
|
||||||
|
|
||||||
setClientSecret(secret) {
|
|
||||||
this.params.clientSecret = secret;
|
|
||||||
}
|
|
||||||
|
|
||||||
setSessionId(sessionId) {
|
|
||||||
this.params.sessionId = sessionId;
|
|
||||||
}
|
|
||||||
|
|
||||||
setRegistrationUrl(regUrl) {
|
|
||||||
this.params.registrationUrl = regUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
setIdSid(idSid) {
|
|
||||||
this.params.idSid = idSid;
|
|
||||||
}
|
|
||||||
|
|
||||||
setGuestAccessToken(token) {
|
|
||||||
this.guestAccessToken = token;
|
|
||||||
}
|
|
||||||
|
|
||||||
getStep() {
|
|
||||||
return this._step;
|
|
||||||
}
|
|
||||||
|
|
||||||
getCredentials() {
|
|
||||||
return this.credentials;
|
|
||||||
}
|
|
||||||
|
|
||||||
getServerData() {
|
|
||||||
return this.data || {};
|
|
||||||
}
|
|
||||||
|
|
||||||
getPromise() {
|
|
||||||
return this.registrationPromise;
|
|
||||||
}
|
|
||||||
|
|
||||||
setStep(step) {
|
|
||||||
this._step = 'Register.' + step;
|
|
||||||
// TODO:
|
|
||||||
// It's a shame this is going to the global dispatcher, we only really
|
|
||||||
// want things which have an instance of this class to be able to add
|
|
||||||
// listeners...
|
|
||||||
console.log("Dispatching 'registration_step_update' for step %s", this._step);
|
|
||||||
dis.dispatch({
|
|
||||||
action: "registration_step_update"
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Starts the registration process from the first stage
|
|
||||||
*/
|
|
||||||
register(formVals) {
|
|
||||||
var {username, password, email} = formVals;
|
|
||||||
this.email = email;
|
|
||||||
this.username = username;
|
|
||||||
this.password = password;
|
|
||||||
const client = this._createTemporaryClient();
|
|
||||||
this.activeStage = null;
|
|
||||||
|
|
||||||
// If there hasn't been a client secret set by this point,
|
|
||||||
// generate one for this session. It will only be used if
|
|
||||||
// we do email verification, but far simpler to just make
|
|
||||||
// sure we have one.
|
|
||||||
// We re-use this same secret over multiple calls to register
|
|
||||||
// so that the identity server can honour the sendAttempt
|
|
||||||
// parameter and not re-send email unless we actually want
|
|
||||||
// another mail to be sent.
|
|
||||||
if (!this.params.clientSecret) {
|
|
||||||
this.params.clientSecret = client.generateClientSecret();
|
|
||||||
}
|
|
||||||
return this._tryRegister(client);
|
|
||||||
}
|
|
||||||
|
|
||||||
_tryRegister(client, authDict, poll_for_success) {
|
|
||||||
var self = this;
|
|
||||||
|
|
||||||
var bindEmail;
|
|
||||||
|
|
||||||
if (this.username && this.password) {
|
|
||||||
// only need to bind_email when sending u/p - sending it at other
|
|
||||||
// times clobbers the u/p resulting in M_MISSING_PARAM (password)
|
|
||||||
bindEmail = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO need to figure out how to send the device display name to /register.
|
|
||||||
return client.register(
|
|
||||||
this.username, this.password, this.params.sessionId, authDict, bindEmail,
|
|
||||||
this.guestAccessToken
|
|
||||||
).then(function(result) {
|
|
||||||
self.credentials = result;
|
|
||||||
self.setStep("COMPLETE");
|
|
||||||
return result; // contains the credentials
|
|
||||||
}, function(error) {
|
|
||||||
if (error.httpStatus === 401) {
|
|
||||||
if (error.data && error.data.flows) {
|
|
||||||
// Remember the session ID from the server:
|
|
||||||
// Either this is our first 401 in which case we need to store the
|
|
||||||
// session ID for future calls, or it isn't in which case this
|
|
||||||
// is just a no-op since it ought to be the same (or if it isn't,
|
|
||||||
// we should use the latest one from the server in any case).
|
|
||||||
self.params.sessionId = error.data.session;
|
|
||||||
self.data = error.data || {};
|
|
||||||
var flow = self.chooseFlow(error.data.flows);
|
|
||||||
|
|
||||||
if (flow) {
|
|
||||||
console.log("Active flow => %s", JSON.stringify(flow));
|
|
||||||
var flowStage = self.firstUncompletedStage(flow);
|
|
||||||
if (!self.activeStage || flowStage != self.activeStage.type) {
|
|
||||||
return self._startStage(client, flowStage).catch(function(err) {
|
|
||||||
self.setStep('START');
|
|
||||||
throw err;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (poll_for_success) {
|
|
||||||
return q.delay(5000).then(function() {
|
|
||||||
return self._tryRegister(client, authDict, poll_for_success);
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
throw new Error("Authorisation failed!");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (error.errcode === 'M_USER_IN_USE') {
|
|
||||||
throw new Error("Username in use");
|
|
||||||
} else if (error.errcode == 'M_INVALID_USERNAME') {
|
|
||||||
throw new Error("User names may only contain alphanumeric characters, underscores or dots!");
|
|
||||||
} else if (error.httpStatus >= 400 && error.httpStatus < 500) {
|
|
||||||
throw new Error(`Registration failed! (${error.httpStatus})`);
|
|
||||||
} else if (error.httpStatus >= 500 && error.httpStatus < 600) {
|
|
||||||
throw new Error(
|
|
||||||
`Server error during registration! (${error.httpStatus})`
|
|
||||||
);
|
|
||||||
} else if (error.name == "M_MISSING_PARAM") {
|
|
||||||
// The HS hasn't remembered the login params from
|
|
||||||
// the first try when the login email was sent.
|
|
||||||
throw new Error(
|
|
||||||
"This home server does not support resuming registration."
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
firstUncompletedStage(flow) {
|
|
||||||
for (var i = 0; i < flow.stages.length; ++i) {
|
|
||||||
if (!this.hasCompletedStage(flow.stages[i])) {
|
|
||||||
return flow.stages[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hasCompletedStage(stageType) {
|
|
||||||
var completed = (this.data || {}).completed || [];
|
|
||||||
return completed.indexOf(stageType) !== -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
_startStage(client, stageName) {
|
|
||||||
var self = this;
|
|
||||||
this.setStep(`STEP_${stageName}`);
|
|
||||||
var StageClass = SignupStages[stageName];
|
|
||||||
if (!StageClass) {
|
|
||||||
// no idea how to handle this!
|
|
||||||
throw new Error("Unknown stage: " + stageName);
|
|
||||||
}
|
|
||||||
|
|
||||||
var stage = new StageClass(client, this);
|
|
||||||
this.activeStage = stage;
|
|
||||||
return stage.complete().then(function(request) {
|
|
||||||
if (request.auth) {
|
|
||||||
console.log("Stage %s is returning an auth dict", stageName);
|
|
||||||
return self._tryRegister(client, request.auth, request.poll_for_success);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// never resolve the promise chain. This is for things like email auth
|
|
||||||
// which display a "check your email" message and relies on the
|
|
||||||
// link in the email to actually register you.
|
|
||||||
console.log("Waiting for external action.");
|
|
||||||
return q.defer().promise;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
chooseFlow(flows) {
|
|
||||||
// If the user gave us an email then we want to pick an email
|
|
||||||
// flow we can do, else any other flow.
|
|
||||||
var emailFlow = null;
|
|
||||||
var otherFlow = null;
|
|
||||||
flows.forEach(function(flow) {
|
|
||||||
var flowHasEmail = false;
|
|
||||||
for (var stageI = 0; stageI < flow.stages.length; ++stageI) {
|
|
||||||
var stage = flow.stages[stageI];
|
|
||||||
|
|
||||||
if (!SignupStages[stage]) {
|
|
||||||
// we can't do this flow, don't have a Stage impl.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (stage === EMAIL_STAGE_TYPE) {
|
|
||||||
flowHasEmail = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (flowHasEmail) {
|
|
||||||
emailFlow = flow;
|
|
||||||
} else {
|
|
||||||
otherFlow = flow;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (this.email || this.hasCompletedStage(EMAIL_STAGE_TYPE)) {
|
|
||||||
// we've been given an email or we've already done an email part
|
|
||||||
return emailFlow;
|
|
||||||
} else {
|
|
||||||
return otherFlow;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
recheckState() {
|
|
||||||
// We've been given a bunch of data from a previous register step,
|
|
||||||
// this only happens for email auth currently. It's kinda ming we need
|
|
||||||
// to know this though. A better solution would be to ask the stages if
|
|
||||||
// they are ready to do something rather than accepting that we know about
|
|
||||||
// email auth and its internals.
|
|
||||||
this.params.hasEmailInfo = (
|
|
||||||
this.params.clientSecret && this.params.sessionId && this.params.idSid
|
|
||||||
);
|
|
||||||
|
|
||||||
if (this.params.hasEmailInfo) {
|
|
||||||
const client = this._createTemporaryClient();
|
|
||||||
this.registrationPromise = this._startStage(client, EMAIL_STAGE_TYPE);
|
|
||||||
}
|
|
||||||
return this.registrationPromise;
|
|
||||||
}
|
|
||||||
|
|
||||||
tellStage(stageName, data) {
|
|
||||||
if (this.activeStage && this.activeStage.type === stageName) {
|
|
||||||
console.log("Telling stage %s about something..", stageName);
|
|
||||||
this.activeStage.onReceiveData(data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class Login extends Signup {
|
|
||||||
constructor(hsUrl, isUrl, fallbackHsUrl, opts) {
|
|
||||||
super(hsUrl, isUrl, opts);
|
|
||||||
this._fallbackHsUrl = fallbackHsUrl;
|
|
||||||
this._currentFlowIndex = 0;
|
|
||||||
this._flows = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
getFlows() {
|
|
||||||
var self = this;
|
|
||||||
var client = this._createTemporaryClient();
|
|
||||||
return client.loginFlows().then(function(result) {
|
|
||||||
self._flows = result.flows;
|
|
||||||
self._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 self._flows;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
chooseFlow(flowIndex) {
|
|
||||||
this._currentFlowIndex = flowIndex;
|
|
||||||
}
|
|
||||||
|
|
||||||
getCurrentFlowStep() {
|
|
||||||
// technically the flow can have multiple steps, but no one does this
|
|
||||||
// for login so we can ignore it.
|
|
||||||
var flowStep = this._flows[this._currentFlowIndex];
|
|
||||||
return flowStep ? flowStep.type : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
loginAsGuest() {
|
|
||||||
var client = this._createTemporaryClient();
|
|
||||||
return client.registerGuest({
|
|
||||||
body: {
|
|
||||||
initial_device_display_name: this._defaultDeviceDisplayName,
|
|
||||||
},
|
|
||||||
}).then((creds) => {
|
|
||||||
return {
|
|
||||||
userId: creds.user_id,
|
|
||||||
deviceId: creds.device_id,
|
|
||||||
accessToken: creds.access_token,
|
|
||||||
homeserverUrl: this._hsUrl,
|
|
||||||
identityServerUrl: this._isUrl,
|
|
||||||
guest: true
|
|
||||||
};
|
|
||||||
}, (error) => {
|
|
||||||
if (error.httpStatus === 403) {
|
|
||||||
error.friendlyText = "Guest access is disabled on this Home Server.";
|
|
||||||
} else {
|
|
||||||
error.friendlyText = "Failed to register as guest: " + error.data;
|
|
||||||
}
|
|
||||||
throw error;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
loginViaPassword(username, pass) {
|
|
||||||
var self = this;
|
|
||||||
var isEmail = username.indexOf("@") > 0;
|
|
||||||
var loginParams = {
|
|
||||||
password: pass,
|
|
||||||
initial_device_display_name: this._defaultDeviceDisplayName,
|
|
||||||
};
|
|
||||||
if (isEmail) {
|
|
||||||
loginParams.medium = 'email';
|
|
||||||
loginParams.address = username;
|
|
||||||
} else {
|
|
||||||
loginParams.user = username;
|
|
||||||
}
|
|
||||||
|
|
||||||
var client = this._createTemporaryClient();
|
|
||||||
return client.login('m.login.password', loginParams).then(function(data) {
|
|
||||||
return q({
|
|
||||||
homeserverUrl: self._hsUrl,
|
|
||||||
identityServerUrl: self._isUrl,
|
|
||||||
userId: data.user_id,
|
|
||||||
deviceId: data.device_id,
|
|
||||||
accessToken: data.access_token
|
|
||||||
});
|
|
||||||
}, function(error) {
|
|
||||||
if (error.httpStatus == 400 && loginParams.medium) {
|
|
||||||
error.friendlyText = (
|
|
||||||
'This Home Server does not support login using email address.'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else if (error.httpStatus === 403) {
|
|
||||||
error.friendlyText = (
|
|
||||||
'Incorrect username and/or password.'
|
|
||||||
);
|
|
||||||
if (self._fallbackHsUrl) {
|
|
||||||
var fbClient = Matrix.createClient({
|
|
||||||
baseUrl: self._fallbackHsUrl,
|
|
||||||
idBaseUrl: this._isUrl,
|
|
||||||
});
|
|
||||||
|
|
||||||
return fbClient.login('m.login.password', loginParams).then(function(data) {
|
|
||||||
return q({
|
|
||||||
homeserverUrl: self._fallbackHsUrl,
|
|
||||||
identityServerUrl: self._isUrl,
|
|
||||||
userId: data.user_id,
|
|
||||||
deviceId: data.device_id,
|
|
||||||
accessToken: data.access_token
|
|
||||||
});
|
|
||||||
}, function(fallback_error) {
|
|
||||||
// throw the original error
|
|
||||||
throw error;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
error.friendlyText = (
|
|
||||||
'There was a problem logging in. (HTTP ' + error.httpStatus + ")"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
throw error;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
redirectToCas() {
|
|
||||||
var client = this._createTemporaryClient();
|
|
||||||
var parsedUrl = url.parse(window.location.href, true);
|
|
||||||
parsedUrl.query["homeserver"] = client.getHomeserverUrl();
|
|
||||||
parsedUrl.query["identityServer"] = client.getIdentityServerUrl();
|
|
||||||
var casUrl = client.getCasLoginUrl(url.format(parsedUrl));
|
|
||||||
window.location.href = casUrl;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports.Register = Register;
|
|
||||||
module.exports.Login = Login;
|
|
|
@ -1,166 +0,0 @@
|
||||||
"use strict";
|
|
||||||
var q = require("q");
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An interface class which login types should abide by.
|
|
||||||
*/
|
|
||||||
class Stage {
|
|
||||||
constructor(type, matrixClient, signupInstance) {
|
|
||||||
this.type = type;
|
|
||||||
this.client = matrixClient;
|
|
||||||
this.signupInstance = signupInstance;
|
|
||||||
}
|
|
||||||
|
|
||||||
complete() {
|
|
||||||
// Return a promise which is:
|
|
||||||
// RESOLVED => With an Object which has an 'auth' key which is the auth dict
|
|
||||||
// to submit.
|
|
||||||
// REJECTED => With an Error if there was a problem with this stage.
|
|
||||||
// Has a "message" string and an "isFatal" flag.
|
|
||||||
return q.reject("NOT IMPLEMENTED");
|
|
||||||
}
|
|
||||||
|
|
||||||
onReceiveData() {
|
|
||||||
// NOP
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Stage.TYPE = "NOT IMPLEMENTED";
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This stage requires no auth.
|
|
||||||
*/
|
|
||||||
class DummyStage extends Stage {
|
|
||||||
constructor(matrixClient, signupInstance) {
|
|
||||||
super(DummyStage.TYPE, matrixClient, signupInstance);
|
|
||||||
}
|
|
||||||
|
|
||||||
complete() {
|
|
||||||
return q({
|
|
||||||
auth: {
|
|
||||||
type: DummyStage.TYPE
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DummyStage.TYPE = "m.login.dummy";
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This stage uses Google's Recaptcha to do auth.
|
|
||||||
*/
|
|
||||||
class RecaptchaStage extends Stage {
|
|
||||||
constructor(matrixClient, signupInstance) {
|
|
||||||
super(RecaptchaStage.TYPE, matrixClient, signupInstance);
|
|
||||||
this.defer = q.defer(); // resolved with the captcha response
|
|
||||||
}
|
|
||||||
|
|
||||||
// called when the recaptcha has been completed.
|
|
||||||
onReceiveData(data) {
|
|
||||||
if (!data || !data.response) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this.defer.resolve({
|
|
||||||
auth: {
|
|
||||||
type: 'm.login.recaptcha',
|
|
||||||
response: data.response,
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
complete() {
|
|
||||||
return this.defer.promise;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
RecaptchaStage.TYPE = "m.login.recaptcha";
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This state uses the IS to verify email addresses.
|
|
||||||
*/
|
|
||||||
class EmailIdentityStage extends Stage {
|
|
||||||
constructor(matrixClient, signupInstance) {
|
|
||||||
super(EmailIdentityStage.TYPE, matrixClient, signupInstance);
|
|
||||||
}
|
|
||||||
|
|
||||||
_completeVerify() {
|
|
||||||
// pull out the host of the IS URL by creating an anchor element
|
|
||||||
var isLocation = document.createElement('a');
|
|
||||||
isLocation.href = this.signupInstance.getIdentityServerUrl();
|
|
||||||
|
|
||||||
var clientSecret = this.clientSecret || this.signupInstance.params.clientSecret;
|
|
||||||
var sid = this.sid || this.signupInstance.params.idSid;
|
|
||||||
|
|
||||||
return q({
|
|
||||||
auth: {
|
|
||||||
type: 'm.login.email.identity',
|
|
||||||
threepid_creds: {
|
|
||||||
sid: sid,
|
|
||||||
client_secret: clientSecret,
|
|
||||||
id_server: isLocation.host
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Complete the email stage.
|
|
||||||
*
|
|
||||||
* This is called twice under different circumstances:
|
|
||||||
* 1) When requesting an email token from the IS
|
|
||||||
* 2) When validating query parameters received from the link in the email
|
|
||||||
*/
|
|
||||||
complete() {
|
|
||||||
// TODO: The Registration class shouldn't really know this info.
|
|
||||||
if (this.signupInstance.params.hasEmailInfo) {
|
|
||||||
return this._completeVerify();
|
|
||||||
}
|
|
||||||
|
|
||||||
this.clientSecret = this.signupInstance.params.clientSecret;
|
|
||||||
if (!this.clientSecret) {
|
|
||||||
return q.reject(new Error("No client secret specified by Signup class!"));
|
|
||||||
}
|
|
||||||
|
|
||||||
var nextLink = this.signupInstance.params.registrationUrl +
|
|
||||||
'?client_secret=' +
|
|
||||||
encodeURIComponent(this.clientSecret) +
|
|
||||||
"&hs_url=" +
|
|
||||||
encodeURIComponent(this.signupInstance.getHomeserverUrl()) +
|
|
||||||
"&is_url=" +
|
|
||||||
encodeURIComponent(this.signupInstance.getIdentityServerUrl()) +
|
|
||||||
"&session_id=" +
|
|
||||||
encodeURIComponent(this.signupInstance.getServerData().session);
|
|
||||||
|
|
||||||
var self = this;
|
|
||||||
return this.client.requestRegisterEmailToken(
|
|
||||||
this.signupInstance.email,
|
|
||||||
this.clientSecret,
|
|
||||||
1, // TODO: Multiple send attempts?
|
|
||||||
nextLink
|
|
||||||
).then(function(response) {
|
|
||||||
self.sid = response.sid;
|
|
||||||
return self._completeVerify();
|
|
||||||
}).then(function(request) {
|
|
||||||
request.poll_for_success = true;
|
|
||||||
return request;
|
|
||||||
}, function(error) {
|
|
||||||
console.error(error);
|
|
||||||
var e = {
|
|
||||||
isFatal: true
|
|
||||||
};
|
|
||||||
if (error.errcode == 'M_THREEPID_IN_USE') {
|
|
||||||
e.message = "This email address is already registered";
|
|
||||||
} else {
|
|
||||||
e.message = 'Unable to contact the given identity server';
|
|
||||||
}
|
|
||||||
throw e;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
EmailIdentityStage.TYPE = "m.login.email.identity";
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
[DummyStage.TYPE]: DummyStage,
|
|
||||||
[RecaptchaStage.TYPE]: RecaptchaStage,
|
|
||||||
[EmailIdentityStage.TYPE]: EmailIdentityStage
|
|
||||||
};
|
|
|
@ -23,41 +23,46 @@ class Skinner {
|
||||||
if (this.components === null) {
|
if (this.components === null) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
"Attempted to get a component before a skin has been loaded."+
|
"Attempted to get a component before a skin has been loaded."+
|
||||||
"This is probably because either:"+
|
" This is probably because either:"+
|
||||||
" a) Your app has not called sdk.loadSkin(), or"+
|
" a) Your app has not called sdk.loadSkin(), or"+
|
||||||
" b) A component has called getComponent at the root level"
|
" b) A component has called getComponent at the root level",
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
var comp = this.components[name];
|
let comp = this.components[name];
|
||||||
if (comp) {
|
|
||||||
return comp;
|
|
||||||
}
|
|
||||||
// XXX: Temporarily also try 'views.' as we're currently
|
// XXX: Temporarily also try 'views.' as we're currently
|
||||||
// leaving the 'views.' off views.
|
// leaving the 'views.' off views.
|
||||||
var comp = this.components['views.'+name];
|
if (!comp) {
|
||||||
if (comp) {
|
comp = this.components['views.'+name];
|
||||||
return comp;
|
|
||||||
}
|
}
|
||||||
throw new Error("No such component: "+name);
|
|
||||||
|
if (!comp) {
|
||||||
|
throw new Error("No such component: "+name);
|
||||||
|
}
|
||||||
|
|
||||||
|
// components have to be functions.
|
||||||
|
const validType = typeof comp === 'function';
|
||||||
|
if (!validType) {
|
||||||
|
throw new Error(`Not a valid component: ${name}.`);
|
||||||
|
}
|
||||||
|
return comp;
|
||||||
}
|
}
|
||||||
|
|
||||||
load(skinObject) {
|
load(skinObject) {
|
||||||
if (this.components !== null) {
|
if (this.components !== null) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
"Attempted to load a skin while a skin is already loaded"+
|
"Attempted to load a skin while a skin is already loaded"+
|
||||||
"If you want to change the active skin, call resetSkin first"
|
"If you want to change the active skin, call resetSkin first");
|
||||||
);
|
|
||||||
}
|
}
|
||||||
this.components = {};
|
this.components = {};
|
||||||
var compKeys = Object.keys(skinObject.components);
|
const compKeys = Object.keys(skinObject.components);
|
||||||
for (var i = 0; i < compKeys.length; ++i) {
|
for (let i = 0; i < compKeys.length; ++i) {
|
||||||
var comp = skinObject.components[compKeys[i]];
|
const comp = skinObject.components[compKeys[i]];
|
||||||
this.addComponent(compKeys[i], comp);
|
this.addComponent(compKeys[i], comp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
addComponent(name, comp) {
|
addComponent(name, comp) {
|
||||||
var slot = name;
|
let slot = name;
|
||||||
if (comp.replaces !== undefined) {
|
if (comp.replaces !== undefined) {
|
||||||
if (comp.replaces.indexOf('.') > -1) {
|
if (comp.replaces.indexOf('.') > -1) {
|
||||||
slot = comp.replaces;
|
slot = comp.replaces;
|
||||||
|
@ -79,6 +84,9 @@ class Skinner {
|
||||||
// behaviour with multiple copies of files etc. is erratic at best.
|
// behaviour with multiple copies of files etc. is erratic at best.
|
||||||
// XXX: We can still end up with the same file twice in the resulting
|
// XXX: We can still end up with the same file twice in the resulting
|
||||||
// JS bundle which is nonideal.
|
// JS bundle which is nonideal.
|
||||||
|
// See https://derickbailey.com/2016/03/09/creating-a-true-singleton-in-node-js-with-es6-symbols/
|
||||||
|
// or https://nodejs.org/api/modules.html#modules_module_caching_caveats
|
||||||
|
// ("Modules are cached based on their resolved filename")
|
||||||
if (global.mxSkinner === undefined) {
|
if (global.mxSkinner === undefined) {
|
||||||
global.mxSkinner = new Skinner();
|
global.mxSkinner = new Skinner();
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,11 +14,13 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
import MatrixClientPeg from "./MatrixClientPeg";
|
||||||
var dis = require("./dispatcher");
|
import dis from "./dispatcher";
|
||||||
var Tinter = require("./Tinter");
|
import Tinter from "./Tinter";
|
||||||
import sdk from './index';
|
import sdk from './index';
|
||||||
|
import { _t } from './languageHandler';
|
||||||
import Modal from './Modal';
|
import Modal from './Modal';
|
||||||
|
import SettingsStore, {SettingLevel} from "./settings/SettingsStore";
|
||||||
|
|
||||||
|
|
||||||
class Command {
|
class Command {
|
||||||
|
@ -41,58 +43,64 @@ class Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
getUsage() {
|
getUsage() {
|
||||||
return "Usage: " + this.getCommandWithArgs()
|
return _t('Usage') + ': ' + this.getCommandWithArgs();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var reject = function(msg) {
|
function reject(msg) {
|
||||||
return {
|
return {
|
||||||
error: msg
|
error: msg,
|
||||||
};
|
};
|
||||||
};
|
}
|
||||||
|
|
||||||
var success = function(promise) {
|
function success(promise) {
|
||||||
return {
|
return {
|
||||||
promise: promise
|
promise: promise,
|
||||||
};
|
};
|
||||||
};
|
}
|
||||||
|
|
||||||
var commands = {
|
/* Disable the "unexpected this" error for these commands - all of the run
|
||||||
|
* functions are called with `this` bound to the Command instance.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* eslint-disable babel/no-invalid-this */
|
||||||
|
|
||||||
|
const commands = {
|
||||||
ddg: new Command("ddg", "<query>", function(roomId, args) {
|
ddg: new Command("ddg", "<query>", function(roomId, args) {
|
||||||
const ErrorDialog = sdk.getComponent('dialogs.ErrorDialog');
|
const ErrorDialog = sdk.getComponent('dialogs.ErrorDialog');
|
||||||
// TODO Don't explain this away, actually show a search UI here.
|
// TODO Don't explain this away, actually show a search UI here.
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createTrackedDialog('Slash Commands', '/ddg is not a command', ErrorDialog, {
|
||||||
title: "/ddg is not a command",
|
title: _t('/ddg is not a command'),
|
||||||
description: "To use it, just wait for autocomplete results to load and tab through them.",
|
description: _t('To use it, just wait for autocomplete results to load and tab through them.'),
|
||||||
});
|
});
|
||||||
return success();
|
return success();
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Change your nickname
|
// Change your nickname
|
||||||
nick: new Command("nick", "<display_name>", function(room_id, args) {
|
nick: new Command("nick", "<display_name>", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().setDisplayName(args)
|
MatrixClientPeg.get().setDisplayName(args),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return reject(this.getUsage());
|
return reject(this.getUsage());
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Changes the colorscheme of your current room
|
// Changes the colorscheme of your current room
|
||||||
tint: new Command("tint", "<color1> [<color2>]", function(room_id, args) {
|
tint: new Command("tint", "<color1> [<color2>]", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(#([0-9a-fA-F]{3}|[0-9a-fA-F]{6}))( +(#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})))?$/);
|
const matches = args.match(/^(#([0-9a-fA-F]{3}|[0-9a-fA-F]{6}))( +(#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})))?$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
Tinter.tint(matches[1], matches[4]);
|
Tinter.tint(matches[1], matches[4]);
|
||||||
var colorScheme = {}
|
const colorScheme = {};
|
||||||
colorScheme.primary_color = matches[1];
|
colorScheme.primary_color = matches[1];
|
||||||
if (matches[4]) {
|
if (matches[4]) {
|
||||||
colorScheme.secondary_color = matches[4];
|
colorScheme.secondary_color = matches[4];
|
||||||
|
} else {
|
||||||
|
colorScheme.secondary_color = colorScheme.primary_color;
|
||||||
}
|
}
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().setRoomAccountData(
|
SettingsStore.setValue("roomColor", roomId, SettingLevel.ROOM_ACCOUNT, colorScheme),
|
||||||
room_id, "org.matrix.room.color_scheme", colorScheme
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -100,22 +108,22 @@ var commands = {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Change the room topic
|
// Change the room topic
|
||||||
topic: new Command("topic", "<topic>", function(room_id, args) {
|
topic: new Command("topic", "<topic>", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().setRoomTopic(room_id, args)
|
MatrixClientPeg.get().setRoomTopic(roomId, args),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return reject(this.getUsage());
|
return reject(this.getUsage());
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Invite a user
|
// Invite a user
|
||||||
invite: new Command("invite", "<userId>", function(room_id, args) {
|
invite: new Command("invite", "<userId>", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+)$/);
|
const matches = args.match(/^(\S+)$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().invite(room_id, matches[1])
|
MatrixClientPeg.get().invite(roomId, matches[1]),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -123,21 +131,21 @@ var commands = {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Join a room
|
// Join a room
|
||||||
join: new Command("join", "#alias:domain", function(room_id, args) {
|
join: new Command("join", "#alias:domain", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+)$/);
|
const matches = args.match(/^(\S+)$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
var room_alias = matches[1];
|
let roomAlias = matches[1];
|
||||||
if (room_alias[0] !== '#') {
|
if (roomAlias[0] !== '#') {
|
||||||
return reject(this.getUsage());
|
return reject(this.getUsage());
|
||||||
}
|
}
|
||||||
if (!room_alias.match(/:/)) {
|
if (!roomAlias.match(/:/)) {
|
||||||
room_alias += ':' + MatrixClientPeg.get().getDomain();
|
roomAlias += ':' + MatrixClientPeg.get().getDomain();
|
||||||
}
|
}
|
||||||
|
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'view_room',
|
action: 'view_room',
|
||||||
room_alias: room_alias,
|
room_alias: roomAlias,
|
||||||
auto_join: true,
|
auto_join: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -147,29 +155,29 @@ var commands = {
|
||||||
return reject(this.getUsage());
|
return reject(this.getUsage());
|
||||||
}),
|
}),
|
||||||
|
|
||||||
part: new Command("part", "[#alias:domain]", function(room_id, args) {
|
part: new Command("part", "[#alias:domain]", function(roomId, args) {
|
||||||
var targetRoomId;
|
let targetRoomId;
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+)$/);
|
const matches = args.match(/^(\S+)$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
var room_alias = matches[1];
|
let roomAlias = matches[1];
|
||||||
if (room_alias[0] !== '#') {
|
if (roomAlias[0] !== '#') {
|
||||||
return reject(this.getUsage());
|
return reject(this.getUsage());
|
||||||
}
|
}
|
||||||
if (!room_alias.match(/:/)) {
|
if (!roomAlias.match(/:/)) {
|
||||||
room_alias += ':' + MatrixClientPeg.get().getDomain();
|
roomAlias += ':' + MatrixClientPeg.get().getDomain();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to find a room with this alias
|
// Try to find a room with this alias
|
||||||
var rooms = MatrixClientPeg.get().getRooms();
|
const rooms = MatrixClientPeg.get().getRooms();
|
||||||
for (var i = 0; i < rooms.length; i++) {
|
for (let i = 0; i < rooms.length; i++) {
|
||||||
var aliasEvents = rooms[i].currentState.getStateEvents(
|
const aliasEvents = rooms[i].currentState.getStateEvents(
|
||||||
"m.room.aliases"
|
"m.room.aliases",
|
||||||
);
|
);
|
||||||
for (var j = 0; j < aliasEvents.length; j++) {
|
for (let j = 0; j < aliasEvents.length; j++) {
|
||||||
var aliases = aliasEvents[j].getContent().aliases || [];
|
const aliases = aliasEvents[j].getContent().aliases || [];
|
||||||
for (var k = 0; k < aliases.length; k++) {
|
for (let k = 0; k < aliases.length; k++) {
|
||||||
if (aliases[k] === room_alias) {
|
if (aliases[k] === roomAlias) {
|
||||||
targetRoomId = rooms[i].roomId;
|
targetRoomId = rooms[i].roomId;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -178,27 +186,28 @@ var commands = {
|
||||||
}
|
}
|
||||||
if (targetRoomId) { break; }
|
if (targetRoomId) { break; }
|
||||||
}
|
}
|
||||||
}
|
if (!targetRoomId) {
|
||||||
if (!targetRoomId) {
|
return reject(_t("Unrecognised room alias:") + ' ' + roomAlias);
|
||||||
return reject("Unrecognised room alias: " + room_alias);
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!targetRoomId) targetRoomId = room_id;
|
if (!targetRoomId) targetRoomId = roomId;
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().leave(targetRoomId).then(
|
MatrixClientPeg.get().leave(targetRoomId).then(
|
||||||
function() {
|
function() {
|
||||||
dis.dispatch({action: 'view_next_room'});
|
dis.dispatch({action: 'view_next_room'});
|
||||||
})
|
},
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Kick a user from the room with an optional reason
|
// Kick a user from the room with an optional reason
|
||||||
kick: new Command("kick", "<userId> [<reason>]", function(room_id, args) {
|
kick: new Command("kick", "<userId> [<reason>]", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+?)( +(.*))?$/);
|
const matches = args.match(/^(\S+?)( +(.*))?$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().kick(room_id, matches[1], matches[3])
|
MatrixClientPeg.get().kick(roomId, matches[1], matches[3]),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -206,12 +215,12 @@ var commands = {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Ban a user from the room with an optional reason
|
// Ban a user from the room with an optional reason
|
||||||
ban: new Command("ban", "<userId> [<reason>]", function(room_id, args) {
|
ban: new Command("ban", "<userId> [<reason>]", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+?)( +(.*))?$/);
|
const matches = args.match(/^(\S+?)( +(.*))?$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().ban(room_id, matches[1], matches[3])
|
MatrixClientPeg.get().ban(roomId, matches[1], matches[3]),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -219,13 +228,66 @@ var commands = {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Unban a user from the room
|
// Unban a user from the room
|
||||||
unban: new Command("unban", "<userId>", function(room_id, args) {
|
unban: new Command("unban", "<userId>", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+)$/);
|
const matches = args.match(/^(\S+)$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
// Reset the user membership to "leave" to unban him
|
// Reset the user membership to "leave" to unban him
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().unban(room_id, matches[1])
|
MatrixClientPeg.get().unban(roomId, matches[1]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return reject(this.getUsage());
|
||||||
|
}),
|
||||||
|
|
||||||
|
ignore: new Command("ignore", "<userId>", function(roomId, args) {
|
||||||
|
if (args) {
|
||||||
|
const matches = args.match(/^(\S+)$/);
|
||||||
|
if (matches) {
|
||||||
|
const userId = matches[1];
|
||||||
|
const ignoredUsers = MatrixClientPeg.get().getIgnoredUsers();
|
||||||
|
ignoredUsers.push(userId); // de-duped internally in the js-sdk
|
||||||
|
return success(
|
||||||
|
MatrixClientPeg.get().setIgnoredUsers(ignoredUsers).then(() => {
|
||||||
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
||||||
|
Modal.createTrackedDialog('Slash Commands', 'User ignored', QuestionDialog, {
|
||||||
|
title: _t("Ignored user"),
|
||||||
|
description: (
|
||||||
|
<div>
|
||||||
|
<p>{ _t("You are now ignoring %(userId)s", {userId: userId}) }</p>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
hasCancelButton: false,
|
||||||
|
});
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return reject(this.getUsage());
|
||||||
|
}),
|
||||||
|
|
||||||
|
unignore: new Command("unignore", "<userId>", function(roomId, args) {
|
||||||
|
if (args) {
|
||||||
|
const matches = args.match(/^(\S+)$/);
|
||||||
|
if (matches) {
|
||||||
|
const userId = matches[1];
|
||||||
|
const ignoredUsers = MatrixClientPeg.get().getIgnoredUsers();
|
||||||
|
const index = ignoredUsers.indexOf(userId);
|
||||||
|
if (index !== -1) ignoredUsers.splice(index, 1);
|
||||||
|
return success(
|
||||||
|
MatrixClientPeg.get().setIgnoredUsers(ignoredUsers).then(() => {
|
||||||
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
||||||
|
Modal.createTrackedDialog('Slash Commands', 'User unignored', QuestionDialog, {
|
||||||
|
title: _t("Unignored user"),
|
||||||
|
description: (
|
||||||
|
<div>
|
||||||
|
<p>{ _t("You are no longer ignoring %(userId)s", {userId: userId}) }</p>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
hasCancelButton: false,
|
||||||
|
});
|
||||||
|
}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -233,27 +295,27 @@ var commands = {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Define the power level of a user
|
// Define the power level of a user
|
||||||
op: new Command("op", "<userId> [<power level>]", function(room_id, args) {
|
op: new Command("op", "<userId> [<power level>]", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+?)( +(\d+))?$/);
|
const matches = args.match(/^(\S+?)( +(-?\d+))?$/);
|
||||||
var powerLevel = 50; // default power level for op
|
let powerLevel = 50; // default power level for op
|
||||||
if (matches) {
|
if (matches) {
|
||||||
var user_id = matches[1];
|
const userId = matches[1];
|
||||||
if (matches.length === 4 && undefined !== matches[3]) {
|
if (matches.length === 4 && undefined !== matches[3]) {
|
||||||
powerLevel = parseInt(matches[3]);
|
powerLevel = parseInt(matches[3]);
|
||||||
}
|
}
|
||||||
if (powerLevel !== NaN) {
|
if (!isNaN(powerLevel)) {
|
||||||
var room = MatrixClientPeg.get().getRoom(room_id);
|
const room = MatrixClientPeg.get().getRoom(roomId);
|
||||||
if (!room) {
|
if (!room) {
|
||||||
return reject("Bad room ID: " + room_id);
|
return reject("Bad room ID: " + roomId);
|
||||||
}
|
}
|
||||||
var powerLevelEvent = room.currentState.getStateEvents(
|
const powerLevelEvent = room.currentState.getStateEvents(
|
||||||
"m.room.power_levels", ""
|
"m.room.power_levels", "",
|
||||||
);
|
);
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().setPowerLevel(
|
MatrixClientPeg.get().setPowerLevel(
|
||||||
room_id, user_id, powerLevel, powerLevelEvent
|
roomId, userId, powerLevel, powerLevelEvent,
|
||||||
)
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -262,33 +324,102 @@ var commands = {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Reset the power level of a user
|
// Reset the power level of a user
|
||||||
deop: new Command("deop", "<userId>", function(room_id, args) {
|
deop: new Command("deop", "<userId>", function(roomId, args) {
|
||||||
if (args) {
|
if (args) {
|
||||||
var matches = args.match(/^(\S+)$/);
|
const matches = args.match(/^(\S+)$/);
|
||||||
if (matches) {
|
if (matches) {
|
||||||
var room = MatrixClientPeg.get().getRoom(room_id);
|
const room = MatrixClientPeg.get().getRoom(roomId);
|
||||||
if (!room) {
|
if (!room) {
|
||||||
return reject("Bad room ID: " + room_id);
|
return reject("Bad room ID: " + roomId);
|
||||||
}
|
}
|
||||||
|
|
||||||
var powerLevelEvent = room.currentState.getStateEvents(
|
const powerLevelEvent = room.currentState.getStateEvents(
|
||||||
"m.room.power_levels", ""
|
"m.room.power_levels", "",
|
||||||
);
|
);
|
||||||
return success(
|
return success(
|
||||||
MatrixClientPeg.get().setPowerLevel(
|
MatrixClientPeg.get().setPowerLevel(
|
||||||
room_id, args, undefined, powerLevelEvent
|
roomId, args, undefined, powerLevelEvent,
|
||||||
)
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return reject(this.getUsage());
|
return reject(this.getUsage());
|
||||||
})
|
}),
|
||||||
|
|
||||||
|
// Open developer tools
|
||||||
|
devtools: new Command("devtools", "", function(roomId) {
|
||||||
|
const DevtoolsDialog = sdk.getComponent("dialogs.DevtoolsDialog");
|
||||||
|
Modal.createDialog(DevtoolsDialog, { roomId });
|
||||||
|
return success();
|
||||||
|
}),
|
||||||
|
|
||||||
|
// Verify a user, device, and pubkey tuple
|
||||||
|
verify: new Command("verify", "<userId> <deviceId> <deviceSigningKey>", function(roomId, args) {
|
||||||
|
if (args) {
|
||||||
|
const matches = args.match(/^(\S+) +(\S+) +(\S+)$/);
|
||||||
|
if (matches) {
|
||||||
|
const userId = matches[1];
|
||||||
|
const deviceId = matches[2];
|
||||||
|
const fingerprint = matches[3];
|
||||||
|
|
||||||
|
return success(
|
||||||
|
// Promise.resolve to handle transition from static result to promise; can be removed
|
||||||
|
// in future
|
||||||
|
Promise.resolve(MatrixClientPeg.get().getStoredDevice(userId, deviceId)).then((device) => {
|
||||||
|
if (!device) {
|
||||||
|
throw new Error(_t(`Unknown (user, device) pair:`) + ` (${userId}, ${deviceId})`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (device.isVerified()) {
|
||||||
|
if (device.getFingerprint() === fingerprint) {
|
||||||
|
throw new Error(_t(`Device already verified!`));
|
||||||
|
} else {
|
||||||
|
throw new Error(_t(`WARNING: Device already verified, but keys do NOT MATCH!`));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (device.getFingerprint() !== fingerprint) {
|
||||||
|
const fprint = device.getFingerprint();
|
||||||
|
throw new Error(
|
||||||
|
_t('WARNING: KEY VERIFICATION FAILED! The signing key for %(userId)s and device' +
|
||||||
|
' %(deviceId)s is "%(fprint)s" which does not match the provided key' +
|
||||||
|
' "%(fingerprint)s". This could mean your communications are being intercepted!',
|
||||||
|
{deviceId: deviceId, fprint: fprint, userId: userId, fingerprint: fingerprint}));
|
||||||
|
}
|
||||||
|
|
||||||
|
return MatrixClientPeg.get().setDeviceVerified(userId, deviceId, true);
|
||||||
|
}).then(() => {
|
||||||
|
// Tell the user we verified everything
|
||||||
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
||||||
|
Modal.createTrackedDialog('Slash Commands', 'Verified key', QuestionDialog, {
|
||||||
|
title: _t("Verified key"),
|
||||||
|
description: (
|
||||||
|
<div>
|
||||||
|
<p>
|
||||||
|
{
|
||||||
|
_t("The signing key you provided matches the signing key you received " +
|
||||||
|
"from %(userId)s's device %(deviceId)s. Device marked as verified.",
|
||||||
|
{userId: userId, deviceId: deviceId})
|
||||||
|
}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
hasCancelButton: false,
|
||||||
|
});
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return reject(this.getUsage());
|
||||||
|
}),
|
||||||
};
|
};
|
||||||
|
/* eslint-enable babel/no-invalid-this */
|
||||||
|
|
||||||
|
|
||||||
// helpful aliases
|
// helpful aliases
|
||||||
var aliases = {
|
const aliases = {
|
||||||
j: "join"
|
j: "join",
|
||||||
}
|
};
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
/**
|
/**
|
||||||
|
@ -304,13 +435,13 @@ module.exports = {
|
||||||
// IRC-style commands
|
// IRC-style commands
|
||||||
input = input.replace(/\s+$/, "");
|
input = input.replace(/\s+$/, "");
|
||||||
if (input[0] === "/" && input[1] !== "/") {
|
if (input[0] === "/" && input[1] !== "/") {
|
||||||
var bits = input.match(/^(\S+?)( +((.|\n)*))?$/);
|
const bits = input.match(/^(\S+?)( +((.|\n)*))?$/);
|
||||||
var cmd, args;
|
let cmd;
|
||||||
|
let args;
|
||||||
if (bits) {
|
if (bits) {
|
||||||
cmd = bits[1].substring(1).toLowerCase();
|
cmd = bits[1].substring(1).toLowerCase();
|
||||||
args = bits[3];
|
args = bits[3];
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
cmd = input;
|
cmd = input;
|
||||||
}
|
}
|
||||||
if (cmd === "me") return null;
|
if (cmd === "me") return null;
|
||||||
|
@ -319,9 +450,8 @@ module.exports = {
|
||||||
}
|
}
|
||||||
if (commands[cmd]) {
|
if (commands[cmd]) {
|
||||||
return commands[cmd].run(roomId, args);
|
return commands[cmd].run(roomId, args);
|
||||||
}
|
} else {
|
||||||
else {
|
return reject(_t("Unrecognised command:") + ' ' + input);
|
||||||
return reject("Unrecognised command: " + input);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null; // not a command
|
return null; // not a command
|
||||||
|
@ -329,12 +459,12 @@ module.exports = {
|
||||||
|
|
||||||
getCommandList: function() {
|
getCommandList: function() {
|
||||||
// Return all the commands plus /me and /markdown which aren't handled like normal commands
|
// Return all the commands plus /me and /markdown which aren't handled like normal commands
|
||||||
var cmds = Object.keys(commands).sort().map(function(cmdKey) {
|
const cmds = Object.keys(commands).sort().map(function(cmdKey) {
|
||||||
return commands[cmdKey];
|
return commands[cmdKey];
|
||||||
})
|
});
|
||||||
cmds.push(new Command("me", "<action>", function(){}));
|
cmds.push(new Command("me", "<action>", function() {}));
|
||||||
cmds.push(new Command("markdown", "<on|off>", function(){}));
|
cmds.push(new Command("markdown", "<on|off>", function() {}));
|
||||||
|
|
||||||
return cmds;
|
return cmds;
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,391 +0,0 @@
|
||||||
/*
|
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { Entry, MemberEntry, CommandEntry } from './TabCompleteEntries';
|
|
||||||
import SlashCommands from './SlashCommands';
|
|
||||||
import MatrixClientPeg from './MatrixClientPeg';
|
|
||||||
|
|
||||||
const DELAY_TIME_MS = 1000;
|
|
||||||
const KEY_TAB = 9;
|
|
||||||
const KEY_SHIFT = 16;
|
|
||||||
const KEY_WINDOWS = 91;
|
|
||||||
|
|
||||||
// NB: DO NOT USE \b its "words" are roman alphabet only!
|
|
||||||
//
|
|
||||||
// Capturing group containing the start
|
|
||||||
// of line or a whitespace char
|
|
||||||
// \_______________ __________Capturing group of 0 or more non-whitespace chars
|
|
||||||
// _|__ _|_ followed by the end of line
|
|
||||||
// / \/ \
|
|
||||||
const MATCH_REGEX = /(^|\s)(\S*)$/;
|
|
||||||
|
|
||||||
class TabComplete {
|
|
||||||
|
|
||||||
constructor(opts) {
|
|
||||||
opts.allowLooping = opts.allowLooping || false;
|
|
||||||
opts.autoEnterTabComplete = opts.autoEnterTabComplete || false;
|
|
||||||
opts.onClickCompletes = opts.onClickCompletes || false;
|
|
||||||
this.opts = opts;
|
|
||||||
this.completing = false;
|
|
||||||
this.list = []; // full set of tab-completable things
|
|
||||||
this.matchedList = []; // subset of completable things to loop over
|
|
||||||
this.currentIndex = 0; // index in matchedList currently
|
|
||||||
this.originalText = null; // original input text when tab was first hit
|
|
||||||
this.textArea = opts.textArea; // DOMElement
|
|
||||||
this.isFirstWord = false; // true if you tab-complete on the first word
|
|
||||||
this.enterTabCompleteTimerId = null;
|
|
||||||
this.inPassiveMode = false;
|
|
||||||
|
|
||||||
// Map tracking ordering of the room members.
|
|
||||||
// userId: integer, highest comes first.
|
|
||||||
this.memberTabOrder = {};
|
|
||||||
|
|
||||||
// monotonically increasing counter used for tracking ordering of members
|
|
||||||
this.memberOrderSeq = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Call this when a a UI element representing a tab complete entry has been clicked
|
|
||||||
* @param {entry} The entry that was clicked
|
|
||||||
*/
|
|
||||||
onEntryClick(entry) {
|
|
||||||
if (this.opts.onClickCompletes) {
|
|
||||||
this.completeTo(entry);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
loadEntries(room) {
|
|
||||||
this._makeEntries(room);
|
|
||||||
this._initSorting(room);
|
|
||||||
this._sortEntries();
|
|
||||||
}
|
|
||||||
|
|
||||||
onMemberSpoke(member) {
|
|
||||||
if (this.memberTabOrder[member.userId] === undefined) {
|
|
||||||
this.list.push(new MemberEntry(member));
|
|
||||||
}
|
|
||||||
this.memberTabOrder[member.userId] = this.memberOrderSeq++;
|
|
||||||
this._sortEntries();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param {DOMElement}
|
|
||||||
*/
|
|
||||||
setTextArea(textArea) {
|
|
||||||
this.textArea = textArea;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {Boolean}
|
|
||||||
*/
|
|
||||||
isTabCompleting() {
|
|
||||||
// actually have things to tab over
|
|
||||||
return this.completing && this.matchedList.length > 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
stopTabCompleting() {
|
|
||||||
this.completing = false;
|
|
||||||
this.currentIndex = 0;
|
|
||||||
this._notifyStateChange();
|
|
||||||
}
|
|
||||||
|
|
||||||
startTabCompleting(passive) {
|
|
||||||
this.originalText = this.textArea.value; // cache starting text
|
|
||||||
|
|
||||||
// grab the partial word from the text which we'll be tab-completing
|
|
||||||
var res = MATCH_REGEX.exec(this.originalText);
|
|
||||||
if (!res) {
|
|
||||||
this.matchedList = [];
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// ES6 destructuring; ignore first element (the complete match)
|
|
||||||
var [ , boundaryGroup, partialGroup] = res;
|
|
||||||
|
|
||||||
if (partialGroup.length === 0 && passive) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.isFirstWord = partialGroup.length === this.originalText.length;
|
|
||||||
|
|
||||||
this.completing = true;
|
|
||||||
this.currentIndex = 0;
|
|
||||||
|
|
||||||
this.matchedList = [
|
|
||||||
new Entry(partialGroup) // first entry is always the original partial
|
|
||||||
];
|
|
||||||
|
|
||||||
// find matching entries in the set of entries given to us
|
|
||||||
this.list.forEach((entry) => {
|
|
||||||
if (entry.text.toLowerCase().indexOf(partialGroup.toLowerCase()) === 0) {
|
|
||||||
this.matchedList.push(entry);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// console.log("calculated completions => %s", JSON.stringify(this.matchedList));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Do an auto-complete with the given word. This terminates the tab-complete.
|
|
||||||
* @param {Entry} entry The tab-complete entry to complete to.
|
|
||||||
*/
|
|
||||||
completeTo(entry) {
|
|
||||||
this.textArea.value = this._replaceWith(
|
|
||||||
entry.getFillText(), true, entry.getSuffix(this.isFirstWord)
|
|
||||||
);
|
|
||||||
this.stopTabCompleting();
|
|
||||||
// keep focus on the text area
|
|
||||||
this.textArea.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param {Number} numAheadToPeek Return *up to* this many elements.
|
|
||||||
* @return {Entry[]}
|
|
||||||
*/
|
|
||||||
peek(numAheadToPeek) {
|
|
||||||
if (this.matchedList.length === 0) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
var peekList = [];
|
|
||||||
|
|
||||||
// return the current match item and then one with an index higher, and
|
|
||||||
// so on until we've reached the requested limit. If we hit the end of
|
|
||||||
// the list of options we're done.
|
|
||||||
for (var i = 0; i < numAheadToPeek; i++) {
|
|
||||||
var nextIndex;
|
|
||||||
if (this.opts.allowLooping) {
|
|
||||||
nextIndex = (this.currentIndex + i) % this.matchedList.length;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
nextIndex = this.currentIndex + i;
|
|
||||||
if (nextIndex === this.matchedList.length) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
peekList.push(this.matchedList[nextIndex]);
|
|
||||||
}
|
|
||||||
// console.log("Peek list(%s): %s", numAheadToPeek, JSON.stringify(peekList));
|
|
||||||
return peekList;
|
|
||||||
}
|
|
||||||
|
|
||||||
handleTabPress(passive, shiftKey) {
|
|
||||||
var wasInPassiveMode = this.inPassiveMode && !passive;
|
|
||||||
this.inPassiveMode = passive;
|
|
||||||
|
|
||||||
if (!this.completing) {
|
|
||||||
this.startTabCompleting(passive);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (shiftKey) {
|
|
||||||
this.nextMatchedEntry(-1);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// if we were in passive mode we got out of sync by incrementing the
|
|
||||||
// index to show the peek view but not set the text area. Therefore,
|
|
||||||
// we want to set the *current* index rather than the *next* index.
|
|
||||||
this.nextMatchedEntry(wasInPassiveMode ? 0 : 1);
|
|
||||||
}
|
|
||||||
this._notifyStateChange();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param {DOMEvent} e
|
|
||||||
*/
|
|
||||||
onKeyDown(ev) {
|
|
||||||
if (!this.textArea) {
|
|
||||||
console.error("onKeyDown called before a <textarea> was set!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ev.keyCode !== KEY_TAB) {
|
|
||||||
// pressing any key (except shift, windows, cmd (OSX) and ctrl/alt combinations)
|
|
||||||
// aborts the current tab completion
|
|
||||||
if (this.completing && ev.keyCode !== KEY_SHIFT &&
|
|
||||||
!ev.metaKey && !ev.ctrlKey && !ev.altKey && ev.keyCode !== KEY_WINDOWS) {
|
|
||||||
// they're resuming typing; reset tab complete state vars.
|
|
||||||
this.stopTabCompleting();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// explicitly pressing any key except tab removes passive mode. Tab doesn't remove
|
|
||||||
// passive mode because handleTabPress needs to know when passive mode is toggling
|
|
||||||
// off so it can resync the textarea/peek list. If tab did remove passive mode then
|
|
||||||
// handleTabPress would never be able to tell when passive mode toggled off.
|
|
||||||
this.inPassiveMode = false;
|
|
||||||
|
|
||||||
// pressing any key at all (except tab) restarts the automatic tab-complete timer
|
|
||||||
if (this.opts.autoEnterTabComplete) {
|
|
||||||
const cachedText = ev.target.value;
|
|
||||||
clearTimeout(this.enterTabCompleteTimerId);
|
|
||||||
this.enterTabCompleteTimerId = setTimeout(() => {
|
|
||||||
if (this.completing) {
|
|
||||||
// If you highlight text and CTRL+X it, tab-completing will not be reset.
|
|
||||||
// This check makes sure that if something like a cut operation has been
|
|
||||||
// done, that we correctly refresh the tab-complete list. Normal backspace
|
|
||||||
// operations get caught by the stopTabCompleting() section above, but
|
|
||||||
// because the CTRL key is held, this does not execute for CTRL+X.
|
|
||||||
if (cachedText !== this.textArea.value) {
|
|
||||||
this.stopTabCompleting();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!this.completing) {
|
|
||||||
this.handleTabPress(true, false);
|
|
||||||
}
|
|
||||||
}, DELAY_TIME_MS);
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ctrl-tab/alt-tab etc shouldn't trigger a complete
|
|
||||||
if (ev.ctrlKey || ev.metaKey || ev.altKey) return;
|
|
||||||
|
|
||||||
// tab key has been pressed at this point
|
|
||||||
this.handleTabPress(false, ev.shiftKey)
|
|
||||||
|
|
||||||
// prevent the default TAB operation (typically focus shifting)
|
|
||||||
ev.preventDefault();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the textarea to the next value in the matched list.
|
|
||||||
* @param {Number} offset Offset to apply *before* setting the next value.
|
|
||||||
*/
|
|
||||||
nextMatchedEntry(offset) {
|
|
||||||
if (this.matchedList.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// work out the new index, wrapping if necessary.
|
|
||||||
this.currentIndex += offset;
|
|
||||||
if (this.currentIndex >= this.matchedList.length) {
|
|
||||||
this.currentIndex = 0;
|
|
||||||
}
|
|
||||||
else if (this.currentIndex < 0) {
|
|
||||||
this.currentIndex = this.matchedList.length - 1;
|
|
||||||
}
|
|
||||||
var isTransitioningToOriginalText = (
|
|
||||||
// impossible to transition if they've never hit tab
|
|
||||||
!this.inPassiveMode && this.currentIndex === 0
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!this.inPassiveMode) {
|
|
||||||
// set textarea to this new value
|
|
||||||
this.textArea.value = this._replaceWith(
|
|
||||||
this.matchedList[this.currentIndex].getFillText(),
|
|
||||||
this.currentIndex !== 0, // don't suffix the original text!
|
|
||||||
this.matchedList[this.currentIndex].getSuffix(this.isFirstWord)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// visual display to the user that we looped - TODO: This should be configurable
|
|
||||||
if (isTransitioningToOriginalText) {
|
|
||||||
this.textArea.style["background-color"] = "#faa";
|
|
||||||
setTimeout(() => { // yay for lexical 'this'!
|
|
||||||
this.textArea.style["background-color"] = "";
|
|
||||||
}, 150);
|
|
||||||
|
|
||||||
if (!this.opts.allowLooping) {
|
|
||||||
this.stopTabCompleting();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
this.textArea.style["background-color"] = ""; // cancel blinks TODO: required?
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_replaceWith(newVal, includeSuffix, suffix) {
|
|
||||||
// The regex to replace the input matches a character of whitespace AND
|
|
||||||
// the partial word. If we just use string.replace() with the regex it will
|
|
||||||
// replace the partial word AND the character of whitespace. We want to
|
|
||||||
// preserve whatever that character is (\n, \t, etc) so find out what it is now.
|
|
||||||
var boundaryChar;
|
|
||||||
var res = MATCH_REGEX.exec(this.originalText);
|
|
||||||
if (res) {
|
|
||||||
boundaryChar = res[1]; // the first captured group
|
|
||||||
}
|
|
||||||
if (boundaryChar === undefined) {
|
|
||||||
console.warn("Failed to find boundary char on text: '%s'", this.originalText);
|
|
||||||
boundaryChar = "";
|
|
||||||
}
|
|
||||||
|
|
||||||
suffix = suffix || "";
|
|
||||||
if (!includeSuffix) {
|
|
||||||
suffix = "";
|
|
||||||
}
|
|
||||||
|
|
||||||
var replacementText = boundaryChar + newVal + suffix;
|
|
||||||
return this.originalText.replace(MATCH_REGEX, function() {
|
|
||||||
return replacementText; // function form to avoid `$` special-casing
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_notifyStateChange() {
|
|
||||||
if (this.opts.onStateChange) {
|
|
||||||
this.opts.onStateChange(this.completing);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_sortEntries() {
|
|
||||||
// largest comes first
|
|
||||||
const KIND_ORDER = {
|
|
||||||
command: 1,
|
|
||||||
member: 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
this.list.sort((a, b) => {
|
|
||||||
const kindOrderDifference = KIND_ORDER[b.kind] - KIND_ORDER[a.kind];
|
|
||||||
if (kindOrderDifference != 0) {
|
|
||||||
return kindOrderDifference;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (a.kind == 'member') {
|
|
||||||
let orderA = this.memberTabOrder[a.member.userId];
|
|
||||||
let orderB = this.memberTabOrder[b.member.userId];
|
|
||||||
if (orderA === undefined) orderA = -1;
|
|
||||||
if (orderB === undefined) orderB = -1;
|
|
||||||
|
|
||||||
return orderB - orderA;
|
|
||||||
}
|
|
||||||
|
|
||||||
// anything else we have no ordering for
|
|
||||||
return 0;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_makeEntries(room) {
|
|
||||||
const myUserId = MatrixClientPeg.get().credentials.userId;
|
|
||||||
|
|
||||||
const members = room.getJoinedMembers().filter(function(member) {
|
|
||||||
if (member.userId !== myUserId) return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.list = MemberEntry.fromMemberList(members).concat(
|
|
||||||
CommandEntry.fromCommands(SlashCommands.getCommandList())
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
_initSorting(room) {
|
|
||||||
this.memberTabOrder = {};
|
|
||||||
this.memberOrderSeq = 0;
|
|
||||||
|
|
||||||
for (const ev of room.getLiveTimeline().getEvents()) {
|
|
||||||
this.memberTabOrder[ev.getSender()] = this.memberOrderSeq++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = TabComplete;
|
|
|
@ -1,126 +0,0 @@
|
||||||
/*
|
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
*/
|
|
||||||
var React = require("react");
|
|
||||||
var sdk = require("./index");
|
|
||||||
|
|
||||||
class Entry {
|
|
||||||
constructor(text) {
|
|
||||||
this.text = text;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {string} The text to display in this entry.
|
|
||||||
*/
|
|
||||||
getText() {
|
|
||||||
return this.text;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {string} The text to insert into the input box. Most of the time
|
|
||||||
* this is the same as getText().
|
|
||||||
*/
|
|
||||||
getFillText() {
|
|
||||||
return this.text;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {ReactClass} Raw JSX
|
|
||||||
*/
|
|
||||||
getImageJsx() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {?string} The unique key= prop for React dedupe
|
|
||||||
*/
|
|
||||||
getKey() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {?string} The suffix to append to the tab-complete, or null to
|
|
||||||
* not do this.
|
|
||||||
*/
|
|
||||||
getSuffix(isFirstWord) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Called when this entry is clicked.
|
|
||||||
*/
|
|
||||||
onClick() {
|
|
||||||
// NOP
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class CommandEntry extends Entry {
|
|
||||||
constructor(cmd, cmdWithArgs) {
|
|
||||||
super(cmdWithArgs);
|
|
||||||
this.kind = 'command';
|
|
||||||
this.cmd = cmd;
|
|
||||||
}
|
|
||||||
|
|
||||||
getFillText() {
|
|
||||||
return this.cmd;
|
|
||||||
}
|
|
||||||
|
|
||||||
getKey() {
|
|
||||||
return this.getFillText();
|
|
||||||
}
|
|
||||||
|
|
||||||
getSuffix(isFirstWord) {
|
|
||||||
return " "; // force a space after the command.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CommandEntry.fromCommands = function(commandArray) {
|
|
||||||
return commandArray.map(function(cmd) {
|
|
||||||
return new CommandEntry(cmd.getCommand(), cmd.getCommandWithArgs());
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
class MemberEntry extends Entry {
|
|
||||||
constructor(member) {
|
|
||||||
super((member.name || member.userId).replace(' (IRC)', ''));
|
|
||||||
this.member = member;
|
|
||||||
this.kind = 'member';
|
|
||||||
}
|
|
||||||
|
|
||||||
getImageJsx() {
|
|
||||||
var MemberAvatar = sdk.getComponent("views.avatars.MemberAvatar");
|
|
||||||
return (
|
|
||||||
<MemberAvatar member={this.member} width={24} height={24} />
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
getKey() {
|
|
||||||
return this.member.userId;
|
|
||||||
}
|
|
||||||
|
|
||||||
getSuffix(isFirstWord) {
|
|
||||||
return isFirstWord ? ": " : " ";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
MemberEntry.fromMemberList = function(members) {
|
|
||||||
return members.map(function(m) {
|
|
||||||
return new MemberEntry(m);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports.Entry = Entry;
|
|
||||||
module.exports.MemberEntry = MemberEntry;
|
|
||||||
module.exports.CommandEntry = CommandEntry;
|
|
|
@ -13,194 +13,310 @@ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
See the License for the specific language governing permissions and
|
See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
import CallHandler from './CallHandler';
|
||||||
var CallHandler = require("./CallHandler");
|
import { _t } from './languageHandler';
|
||||||
|
import * as Roles from './Roles';
|
||||||
|
|
||||||
function textForMemberEvent(ev) {
|
function textForMemberEvent(ev) {
|
||||||
// XXX: SYJS-16 "sender is sometimes null for join messages"
|
// XXX: SYJS-16 "sender is sometimes null for join messages"
|
||||||
var senderName = ev.sender ? ev.sender.name : ev.getSender();
|
const senderName = ev.sender ? ev.sender.name : ev.getSender();
|
||||||
var targetName = ev.target ? ev.target.name : ev.getStateKey();
|
const targetName = ev.target ? ev.target.name : ev.getStateKey();
|
||||||
var ConferenceHandler = CallHandler.getConferenceHandler();
|
const prevContent = ev.getPrevContent();
|
||||||
var reason = ev.getContent().reason ? (
|
const content = ev.getContent();
|
||||||
" Reason: " + ev.getContent().reason
|
|
||||||
) : "";
|
const ConferenceHandler = CallHandler.getConferenceHandler();
|
||||||
switch (ev.getContent().membership) {
|
const reason = content.reason ? (_t('Reason') + ': ' + content.reason) : '';
|
||||||
case 'invite':
|
switch (content.membership) {
|
||||||
var threePidContent = ev.getContent().third_party_invite;
|
case 'invite': {
|
||||||
|
const threePidContent = content.third_party_invite;
|
||||||
if (threePidContent) {
|
if (threePidContent) {
|
||||||
if (threePidContent.display_name) {
|
if (threePidContent.display_name) {
|
||||||
return targetName + " accepted the invitation for " +
|
return _t('%(targetName)s accepted the invitation for %(displayName)s.', {
|
||||||
threePidContent.display_name + ".";
|
targetName,
|
||||||
|
displayName: threePidContent.display_name,
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
return targetName + " accepted an invitation.";
|
return _t('%(targetName)s accepted an invitation.', {targetName});
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
if (ConferenceHandler && ConferenceHandler.isConferenceUser(ev.getStateKey())) {
|
if (ConferenceHandler && ConferenceHandler.isConferenceUser(ev.getStateKey())) {
|
||||||
return senderName + " requested a VoIP conference";
|
return _t('%(senderName)s requested a VoIP conference.', {senderName});
|
||||||
}
|
} else {
|
||||||
else {
|
return _t('%(senderName)s invited %(targetName)s.', {senderName, targetName});
|
||||||
return senderName + " invited " + targetName + ".";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
case 'ban':
|
case 'ban':
|
||||||
return senderName + " banned " + targetName + "." + reason;
|
return _t('%(senderName)s banned %(targetName)s.', {senderName, targetName}) + ' ' + reason;
|
||||||
case 'join':
|
case 'join':
|
||||||
if (ev.getPrevContent() && ev.getPrevContent().membership == 'join') {
|
if (prevContent && prevContent.membership === 'join') {
|
||||||
if (ev.getPrevContent().displayname && ev.getContent().displayname && ev.getPrevContent().displayname != ev.getContent().displayname) {
|
if (prevContent.displayname && content.displayname && prevContent.displayname !== content.displayname) {
|
||||||
return ev.getSender() + " changed their display name from " +
|
return _t('%(oldDisplayName)s changed their display name to %(displayName)s.', {
|
||||||
ev.getPrevContent().displayname + " to " +
|
oldDisplayName: prevContent.displayname,
|
||||||
ev.getContent().displayname;
|
displayName: content.displayname,
|
||||||
} else if (!ev.getPrevContent().displayname && ev.getContent().displayname) {
|
});
|
||||||
return ev.getSender() + " set their display name to " + ev.getContent().displayname;
|
} else if (!prevContent.displayname && content.displayname) {
|
||||||
} else if (ev.getPrevContent().displayname && !ev.getContent().displayname) {
|
return _t('%(senderName)s set their display name to %(displayName)s.', {
|
||||||
return ev.getSender() + " removed their display name (" + ev.getPrevContent().displayname + ")";
|
senderName,
|
||||||
} else if (ev.getPrevContent().avatar_url && !ev.getContent().avatar_url) {
|
displayName: content.displayname,
|
||||||
return senderName + " removed their profile picture";
|
});
|
||||||
} else if (ev.getPrevContent().avatar_url && ev.getContent().avatar_url && ev.getPrevContent().avatar_url != ev.getContent().avatar_url) {
|
} else if (prevContent.displayname && !content.displayname) {
|
||||||
return senderName + " changed their profile picture";
|
return _t('%(senderName)s removed their display name (%(oldDisplayName)s).', {
|
||||||
} else if (!ev.getPrevContent().avatar_url && ev.getContent().avatar_url) {
|
senderName,
|
||||||
return senderName + " set a profile picture";
|
oldDisplayName: prevContent.displayname,
|
||||||
|
});
|
||||||
|
} else if (prevContent.avatar_url && !content.avatar_url) {
|
||||||
|
return _t('%(senderName)s removed their profile picture.', {senderName});
|
||||||
|
} else if (prevContent.avatar_url && content.avatar_url &&
|
||||||
|
prevContent.avatar_url !== content.avatar_url) {
|
||||||
|
return _t('%(senderName)s changed their profile picture.', {senderName});
|
||||||
|
} else if (!prevContent.avatar_url && content.avatar_url) {
|
||||||
|
return _t('%(senderName)s set a profile picture.', {senderName});
|
||||||
} else {
|
} else {
|
||||||
// hacky hack for https://github.com/vector-im/vector-web/issues/2020
|
// suppress null rejoins
|
||||||
return senderName + " rejoined the room.";
|
return '';
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (!ev.target) console.warn("Join message has no target! -- " + ev.getContent().state_key);
|
if (!ev.target) console.warn("Join message has no target! -- " + ev.getContent().state_key);
|
||||||
if (ConferenceHandler && ConferenceHandler.isConferenceUser(ev.getStateKey())) {
|
if (ConferenceHandler && ConferenceHandler.isConferenceUser(ev.getStateKey())) {
|
||||||
return "VoIP conference started";
|
return _t('VoIP conference started.');
|
||||||
}
|
} else {
|
||||||
else {
|
return _t('%(targetName)s joined the room.', {targetName});
|
||||||
return targetName + " joined the room.";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return '';
|
|
||||||
case 'leave':
|
case 'leave':
|
||||||
if (ev.getSender() === ev.getStateKey()) {
|
if (ev.getSender() === ev.getStateKey()) {
|
||||||
if (ConferenceHandler && ConferenceHandler.isConferenceUser(ev.getStateKey())) {
|
if (ConferenceHandler && ConferenceHandler.isConferenceUser(ev.getStateKey())) {
|
||||||
return "VoIP conference finished";
|
return _t('VoIP conference finished.');
|
||||||
|
} else if (prevContent.membership === "invite") {
|
||||||
|
return _t('%(targetName)s rejected the invitation.', {targetName});
|
||||||
|
} else {
|
||||||
|
return _t('%(targetName)s left the room.', {targetName});
|
||||||
}
|
}
|
||||||
else if (ev.getPrevContent().membership === "invite") {
|
} else if (prevContent.membership === "ban") {
|
||||||
return targetName + " rejected the invitation.";
|
return _t('%(senderName)s unbanned %(targetName)s.', {senderName, targetName});
|
||||||
}
|
} else if (prevContent.membership === "join") {
|
||||||
else {
|
return _t('%(senderName)s kicked %(targetName)s.', {senderName, targetName}) + ' ' + reason;
|
||||||
return targetName + " left the room.";
|
} else if (prevContent.membership === "invite") {
|
||||||
}
|
return _t('%(senderName)s withdrew %(targetName)s\'s invitation.', {
|
||||||
}
|
senderName,
|
||||||
else if (ev.getPrevContent().membership === "ban") {
|
targetName,
|
||||||
return senderName + " unbanned " + targetName + ".";
|
}) + ' ' + reason;
|
||||||
}
|
} else {
|
||||||
else if (ev.getPrevContent().membership === "join") {
|
return _t('%(targetName)s left the room.', {targetName});
|
||||||
return senderName + " kicked " + targetName + "." + reason;
|
|
||||||
}
|
|
||||||
else if (ev.getPrevContent().membership === "invite") {
|
|
||||||
return senderName + " withdrew " + targetName + "'s invitation." + reason;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return targetName + " left the room.";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForTopicEvent(ev) {
|
function textForTopicEvent(ev) {
|
||||||
var senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
||||||
|
return _t('%(senderDisplayName)s changed the topic to "%(topic)s".', {
|
||||||
return senderDisplayName + ' changed the topic to "' + ev.getContent().topic + '"';
|
senderDisplayName,
|
||||||
|
topic: ev.getContent().topic,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForRoomNameEvent(ev) {
|
function textForRoomNameEvent(ev) {
|
||||||
var senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
||||||
|
|
||||||
return senderDisplayName + ' changed the room name to "' + ev.getContent().name + '"';
|
if (!ev.getContent().name || ev.getContent().name.trim().length === 0) {
|
||||||
|
return _t('%(senderDisplayName)s removed the room name.', {senderDisplayName});
|
||||||
|
}
|
||||||
|
return _t('%(senderDisplayName)s changed the room name to %(roomName)s.', {
|
||||||
|
senderDisplayName,
|
||||||
|
roomName: ev.getContent().name,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForMessageEvent(ev) {
|
function textForMessageEvent(ev) {
|
||||||
var senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
||||||
|
let message = senderDisplayName + ': ' + ev.getContent().body;
|
||||||
var message = senderDisplayName + ': ' + ev.getContent().body;
|
|
||||||
if (ev.getContent().msgtype === "m.emote") {
|
if (ev.getContent().msgtype === "m.emote") {
|
||||||
message = "* " + senderDisplayName + " " + message;
|
message = "* " + senderDisplayName + " " + message;
|
||||||
} else if (ev.getContent().msgtype === "m.image") {
|
} else if (ev.getContent().msgtype === "m.image") {
|
||||||
message = senderDisplayName + " sent an image.";
|
message = _t('%(senderDisplayName)s sent an image.', {senderDisplayName});
|
||||||
}
|
}
|
||||||
return message;
|
return message;
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForCallAnswerEvent(event) {
|
function textForCallAnswerEvent(event) {
|
||||||
var senderName = event.sender ? event.sender.name : "Someone";
|
const senderName = event.sender ? event.sender.name : _t('Someone');
|
||||||
var supported = MatrixClientPeg.get().supportsVoip() ? "" : " (not supported by this browser)";
|
const supported = MatrixClientPeg.get().supportsVoip() ? '' : _t('(not supported by this browser)');
|
||||||
return senderName + " answered the call." + supported;
|
return _t('%(senderName)s answered the call.', {senderName}) + ' ' + supported;
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForCallHangupEvent(event) {
|
function textForCallHangupEvent(event) {
|
||||||
var senderName = event.sender ? event.sender.name : "Someone";
|
const senderName = event.sender ? event.sender.name : _t('Someone');
|
||||||
var supported = MatrixClientPeg.get().supportsVoip() ? "" : " (not supported by this browser)";
|
const eventContent = event.getContent();
|
||||||
return senderName + " ended the call." + supported;
|
let reason = "";
|
||||||
|
if (!MatrixClientPeg.get().supportsVoip()) {
|
||||||
|
reason = _t('(not supported by this browser)');
|
||||||
|
} else if (eventContent.reason) {
|
||||||
|
if (eventContent.reason === "ice_failed") {
|
||||||
|
reason = _t('(could not connect media)');
|
||||||
|
} else if (eventContent.reason === "invite_timeout") {
|
||||||
|
reason = _t('(no answer)');
|
||||||
|
} else {
|
||||||
|
reason = _t('(unknown failure: %(reason)s)', {reason: eventContent.reason});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return _t('%(senderName)s ended the call.', {senderName}) + ' ' + reason;
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForCallInviteEvent(event) {
|
function textForCallInviteEvent(event) {
|
||||||
var senderName = event.sender ? event.sender.name : "Someone";
|
const senderName = event.sender ? event.sender.name : _t('Someone');
|
||||||
// FIXME: Find a better way to determine this from the event?
|
// FIXME: Find a better way to determine this from the event?
|
||||||
var type = "voice";
|
let callType = "voice";
|
||||||
if (event.getContent().offer && event.getContent().offer.sdp &&
|
if (event.getContent().offer && event.getContent().offer.sdp &&
|
||||||
event.getContent().offer.sdp.indexOf('m=video') !== -1) {
|
event.getContent().offer.sdp.indexOf('m=video') !== -1) {
|
||||||
type = "video";
|
callType = "video";
|
||||||
}
|
}
|
||||||
var supported = MatrixClientPeg.get().supportsVoip() ? "" : " (not supported by this browser)";
|
const supported = MatrixClientPeg.get().supportsVoip() ? "" : _t('(not supported by this browser)');
|
||||||
return senderName + " placed a " + type + " call." + supported;
|
return _t('%(senderName)s placed a %(callType)s call.', {senderName, callType}) + ' ' + supported;
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForThreePidInviteEvent(event) {
|
function textForThreePidInviteEvent(event) {
|
||||||
var senderName = event.sender ? event.sender.name : event.getSender();
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
||||||
return senderName + " sent an invitation to " + event.getContent().display_name +
|
return _t('%(senderName)s sent an invitation to %(targetDisplayName)s to join the room.', {
|
||||||
" to join the room.";
|
senderName,
|
||||||
|
targetDisplayName: event.getContent().display_name,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForHistoryVisibilityEvent(event) {
|
function textForHistoryVisibilityEvent(event) {
|
||||||
var senderName = event.sender ? event.sender.name : event.getSender();
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
||||||
var vis = event.getContent().history_visibility;
|
switch (event.getContent().history_visibility) {
|
||||||
var text = senderName + " made future room history visible to ";
|
case 'invited':
|
||||||
if (vis === "invited") {
|
return _t('%(senderName)s made future room history visible to all room members, '
|
||||||
text += "all room members, from the point they are invited.";
|
+ 'from the point they are invited.', {senderName});
|
||||||
|
case 'joined':
|
||||||
|
return _t('%(senderName)s made future room history visible to all room members, '
|
||||||
|
+ 'from the point they joined.', {senderName});
|
||||||
|
case 'shared':
|
||||||
|
return _t('%(senderName)s made future room history visible to all room members.', {senderName});
|
||||||
|
case 'world_readable':
|
||||||
|
return _t('%(senderName)s made future room history visible to anyone.', {senderName});
|
||||||
|
default:
|
||||||
|
return _t('%(senderName)s made future room history visible to unknown (%(visibility)s).', {
|
||||||
|
senderName,
|
||||||
|
visibility: event.getContent().history_visibility,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
else if (vis === "joined") {
|
|
||||||
text += "all room members, from the point they joined.";
|
|
||||||
}
|
|
||||||
else if (vis === "shared") {
|
|
||||||
text += "all room members.";
|
|
||||||
}
|
|
||||||
else if (vis === "world_readable") {
|
|
||||||
text += "anyone.";
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
text += " unknown (" + vis + ")";
|
|
||||||
}
|
|
||||||
return text;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function textForEncryptionEvent(event) {
|
function textForEncryptionEvent(event) {
|
||||||
var senderName = event.sender ? event.sender.name : event.getSender();
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
||||||
return senderName + " turned on end-to-end encryption (algorithm " + event.getContent().algorithm + ")";
|
return _t('%(senderName)s turned on end-to-end encryption (algorithm %(algorithm)s).', {
|
||||||
|
senderName,
|
||||||
|
algorithm: event.getContent().algorithm,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
var handlers = {
|
// Currently will only display a change if a user's power level is changed
|
||||||
|
function textForPowerEvent(event) {
|
||||||
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
||||||
|
if (!event.getPrevContent() || !event.getPrevContent().users) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
const userDefault = event.getContent().users_default || 0;
|
||||||
|
// Construct set of userIds
|
||||||
|
const users = [];
|
||||||
|
Object.keys(event.getContent().users).forEach(
|
||||||
|
(userId) => {
|
||||||
|
if (users.indexOf(userId) === -1) users.push(userId);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
Object.keys(event.getPrevContent().users).forEach(
|
||||||
|
(userId) => {
|
||||||
|
if (users.indexOf(userId) === -1) users.push(userId);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
const diff = [];
|
||||||
|
// XXX: This is also surely broken for i18n
|
||||||
|
users.forEach((userId) => {
|
||||||
|
// Previous power level
|
||||||
|
const from = event.getPrevContent().users[userId];
|
||||||
|
// Current power level
|
||||||
|
const to = event.getContent().users[userId];
|
||||||
|
if (to !== from) {
|
||||||
|
diff.push(
|
||||||
|
_t('%(userId)s from %(fromPowerLevel)s to %(toPowerLevel)s', {
|
||||||
|
userId,
|
||||||
|
fromPowerLevel: Roles.textualPowerLevel(from, userDefault),
|
||||||
|
toPowerLevel: Roles.textualPowerLevel(to, userDefault),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (!diff.length) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
return _t('%(senderName)s changed the power level of %(powerLevelDiffText)s.', {
|
||||||
|
senderName,
|
||||||
|
powerLevelDiffText: diff.join(", "),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function textForPinnedEvent(event) {
|
||||||
|
const senderName = event.getSender();
|
||||||
|
return _t("%(senderName)s changed the pinned messages for the room.", {senderName});
|
||||||
|
}
|
||||||
|
|
||||||
|
function textForWidgetEvent(event) {
|
||||||
|
const senderName = event.getSender();
|
||||||
|
const {name: prevName, type: prevType, url: prevUrl} = event.getPrevContent();
|
||||||
|
const {name, type, url} = event.getContent() || {};
|
||||||
|
|
||||||
|
let widgetName = name || prevName || type || prevType || '';
|
||||||
|
// Apply sentence case to widget name
|
||||||
|
if (widgetName && widgetName.length > 0) {
|
||||||
|
widgetName = widgetName[0].toUpperCase() + widgetName.slice(1) + ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the widget was removed, its content should be {}, but this is sufficiently
|
||||||
|
// equivalent to that condition.
|
||||||
|
if (url) {
|
||||||
|
if (prevUrl) {
|
||||||
|
return _t('%(widgetName)s widget modified by %(senderName)s', {
|
||||||
|
widgetName, senderName,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
return _t('%(widgetName)s widget added by %(senderName)s', {
|
||||||
|
widgetName, senderName,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return _t('%(widgetName)s widget removed by %(senderName)s', {
|
||||||
|
widgetName, senderName,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlers = {
|
||||||
'm.room.message': textForMessageEvent,
|
'm.room.message': textForMessageEvent,
|
||||||
'm.room.name': textForRoomNameEvent,
|
'm.call.invite': textForCallInviteEvent,
|
||||||
'm.room.topic': textForTopicEvent,
|
'm.call.answer': textForCallAnswerEvent,
|
||||||
'm.room.member': textForMemberEvent,
|
'm.call.hangup': textForCallHangupEvent,
|
||||||
'm.call.invite': textForCallInviteEvent,
|
};
|
||||||
'm.call.answer': textForCallAnswerEvent,
|
|
||||||
'm.call.hangup': textForCallHangupEvent,
|
const stateHandlers = {
|
||||||
|
'm.room.name': textForRoomNameEvent,
|
||||||
|
'm.room.topic': textForTopicEvent,
|
||||||
|
'm.room.member': textForMemberEvent,
|
||||||
'm.room.third_party_invite': textForThreePidInviteEvent,
|
'm.room.third_party_invite': textForThreePidInviteEvent,
|
||||||
'm.room.history_visibility': textForHistoryVisibilityEvent,
|
'm.room.history_visibility': textForHistoryVisibilityEvent,
|
||||||
'm.room.encryption': textForEncryptionEvent,
|
'm.room.encryption': textForEncryptionEvent,
|
||||||
|
'm.room.power_levels': textForPowerEvent,
|
||||||
|
'm.room.pinned_events': textForPinnedEvent,
|
||||||
|
|
||||||
|
'im.vector.modular.widgets': textForWidgetEvent,
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
textForEvent: function(ev) {
|
textForEvent: function(ev) {
|
||||||
var hdlr = handlers[ev.getType()];
|
const handler = (ev.isState() ? stateHandlers : handlers)[ev.getType()];
|
||||||
if (!hdlr) return "";
|
if (handler) return handler(ev);
|
||||||
return hdlr(ev);
|
return '';
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
|
|
521
src/Tinter.js
521
src/Tinter.js
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015 OpenMarket Ltd
|
Copyright 2015 OpenMarket Ltd
|
||||||
|
Copyright 2017 New Vector Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -14,149 +15,125 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var dis = require("./dispatcher");
|
const DEBUG = 0;
|
||||||
var sdk = require("./index");
|
|
||||||
|
|
||||||
// FIXME: these vars should be bundled up and attached to
|
// utility to turn #rrggbb or rgb(r,g,b) into [red,green,blue]
|
||||||
// module.exports otherwise this will break when included by both
|
function colorToRgb(color) {
|
||||||
// react-sdk and apps layered on top.
|
if (!color) {
|
||||||
|
return [0, 0, 0];
|
||||||
|
}
|
||||||
|
|
||||||
var DEBUG = 0;
|
if (color[0] === '#') {
|
||||||
|
color = color.slice(1);
|
||||||
// The colour keys to be replaced as referred to in CSS
|
if (color.length === 3) {
|
||||||
var keyRgb = [
|
color = color[0] + color[0] +
|
||||||
"rgb(118, 207, 166)", // Vector Green
|
color[1] + color[1] +
|
||||||
"rgb(234, 245, 240)", // Vector Light Green
|
color[2] + color[2];
|
||||||
"rgb(211, 239, 225)", // BottomLeftMenu overlay (20% Vector Green)
|
}
|
||||||
];
|
const val = parseInt(color, 16);
|
||||||
|
const r = (val >> 16) & 255;
|
||||||
// Some algebra workings for calculating the tint % of Vector Green & Light Green
|
const g = (val >> 8) & 255;
|
||||||
// x * 118 + (1 - x) * 255 = 234
|
const b = val & 255;
|
||||||
// x * 118 + 255 - 255 * x = 234
|
return [r, g, b];
|
||||||
// x * 118 - x * 255 = 234 - 255
|
} else {
|
||||||
// (255 - 118) x = 255 - 234
|
const match = color.match(/rgb\((.*?),(.*?),(.*?)\)/);
|
||||||
// x = (255 - 234) / (255 - 118) = 0.16
|
if (match) {
|
||||||
|
return [
|
||||||
// The colour keys to be replaced as referred to in SVGs
|
parseInt(match[1]),
|
||||||
var keyHex = [
|
parseInt(match[2]),
|
||||||
"#76CFA6", // Vector Green
|
parseInt(match[3]),
|
||||||
"#EAF5F0", // Vector Light Green
|
];
|
||||||
"#D3EFE1", // BottomLeftMenu overlay (20% Vector Green overlaid on Vector Light Green)
|
|
||||||
];
|
|
||||||
|
|
||||||
// cache of our replacement colours
|
|
||||||
// defaults to our keys.
|
|
||||||
var colors = [
|
|
||||||
keyHex[0],
|
|
||||||
keyHex[1],
|
|
||||||
keyHex[2],
|
|
||||||
];
|
|
||||||
|
|
||||||
var cssFixups = [
|
|
||||||
// {
|
|
||||||
// style: a style object that should be fixed up taken from a stylesheet
|
|
||||||
// attr: name of the attribute to be clobbered, e.g. 'color'
|
|
||||||
// index: ordinal of primary, secondary or tertiary
|
|
||||||
// }
|
|
||||||
];
|
|
||||||
|
|
||||||
// CSS attributes to be fixed up
|
|
||||||
var cssAttrs = [
|
|
||||||
"color",
|
|
||||||
"backgroundColor",
|
|
||||||
"borderColor",
|
|
||||||
"borderTopColor",
|
|
||||||
"borderBottomColor",
|
|
||||||
"borderLeftColor",
|
|
||||||
];
|
|
||||||
|
|
||||||
var svgAttrs = [
|
|
||||||
"fill",
|
|
||||||
"stroke",
|
|
||||||
];
|
|
||||||
|
|
||||||
var cached = false;
|
|
||||||
|
|
||||||
function calcCssFixups() {
|
|
||||||
if (DEBUG) console.log("calcSvgFixups start");
|
|
||||||
for (var i = 0; i < document.styleSheets.length; i++) {
|
|
||||||
var ss = document.styleSheets[i];
|
|
||||||
if (!ss) continue; // well done safari >:(
|
|
||||||
// Chromium apparently sometimes returns null here; unsure why.
|
|
||||||
// see $14534907369972FRXBx:matrix.org in HQ
|
|
||||||
// ...ah, it's because there's a third party extension like
|
|
||||||
// privacybadger inserting its own stylesheet in there with a
|
|
||||||
// resource:// URI or something which results in a XSS error.
|
|
||||||
// See also #vector:matrix.org/$145357669685386ebCfr:matrix.org
|
|
||||||
// ...except some browsers apparently return stylesheets without
|
|
||||||
// hrefs, which we have no choice but ignore right now
|
|
||||||
|
|
||||||
// XXX seriously? we are hardcoding the name of vector's CSS file in
|
|
||||||
// here?
|
|
||||||
//
|
|
||||||
// Why do we need to limit it to vector's CSS file anyway - if there
|
|
||||||
// are other CSS files affecting the doc don't we want to apply the
|
|
||||||
// same transformations to them?
|
|
||||||
//
|
|
||||||
// Iterating through the CSS looking for matches to hack on feels
|
|
||||||
// pretty horrible anyway. And what if the application skin doesn't use
|
|
||||||
// Vector Green as its primary color?
|
|
||||||
|
|
||||||
if (ss.href && !ss.href.match(/\/bundle.*\.css$/)) continue;
|
|
||||||
|
|
||||||
if (!ss.cssRules) continue;
|
|
||||||
for (var j = 0; j < ss.cssRules.length; j++) {
|
|
||||||
var rule = ss.cssRules[j];
|
|
||||||
if (!rule.style) continue;
|
|
||||||
for (var k = 0; k < cssAttrs.length; k++) {
|
|
||||||
var attr = cssAttrs[k];
|
|
||||||
for (var l = 0; l < keyRgb.length; l++) {
|
|
||||||
if (rule.style[attr] === keyRgb[l]) {
|
|
||||||
cssFixups.push({
|
|
||||||
style: rule.style,
|
|
||||||
attr: attr,
|
|
||||||
index: l,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (DEBUG) console.log("calcSvgFixups end");
|
return [0, 0, 0];
|
||||||
}
|
}
|
||||||
|
|
||||||
function applyCssFixups() {
|
// utility to turn [red,green,blue] into #rrggbb
|
||||||
if (DEBUG) console.log("applyCssFixups start");
|
function rgbToColor(rgb) {
|
||||||
for (var i = 0; i < cssFixups.length; i++) {
|
const val = (rgb[0] << 16) | (rgb[1] << 8) | rgb[2];
|
||||||
var cssFixup = cssFixups[i];
|
return '#' + (0x1000000 + val).toString(16).slice(1);
|
||||||
cssFixup.style[cssFixup.attr] = colors[cssFixup.index];
|
}
|
||||||
|
|
||||||
|
class Tinter {
|
||||||
|
constructor() {
|
||||||
|
// The default colour keys to be replaced as referred to in CSS
|
||||||
|
// (should be overridden by .mx_theme_accentColor and .mx_theme_secondaryAccentColor)
|
||||||
|
this.keyRgb = [
|
||||||
|
"rgb(118, 207, 166)", // Vector Green
|
||||||
|
"rgb(234, 245, 240)", // Vector Light Green
|
||||||
|
"rgb(211, 239, 225)", // roomsublist-label-bg-color (20% Green overlaid on Light Green)
|
||||||
|
];
|
||||||
|
|
||||||
|
// Some algebra workings for calculating the tint % of Vector Green & Light Green
|
||||||
|
// x * 118 + (1 - x) * 255 = 234
|
||||||
|
// x * 118 + 255 - 255 * x = 234
|
||||||
|
// x * 118 - x * 255 = 234 - 255
|
||||||
|
// (255 - 118) x = 255 - 234
|
||||||
|
// x = (255 - 234) / (255 - 118) = 0.16
|
||||||
|
|
||||||
|
// The colour keys to be replaced as referred to in SVGs
|
||||||
|
this.keyHex = [
|
||||||
|
"#76CFA6", // Vector Green
|
||||||
|
"#EAF5F0", // Vector Light Green
|
||||||
|
"#D3EFE1", // roomsublist-label-bg-color (20% Green overlaid on Light Green)
|
||||||
|
"#FFFFFF", // white highlights of the SVGs (for switching to dark theme)
|
||||||
|
"#000000", // black lowlights of the SVGs (for switching to dark theme)
|
||||||
|
];
|
||||||
|
|
||||||
|
// track the replacement colours actually being used
|
||||||
|
// defaults to our keys.
|
||||||
|
this.colors = [
|
||||||
|
this.keyHex[0],
|
||||||
|
this.keyHex[1],
|
||||||
|
this.keyHex[2],
|
||||||
|
this.keyHex[3],
|
||||||
|
this.keyHex[4],
|
||||||
|
];
|
||||||
|
|
||||||
|
// track the most current tint request inputs (which may differ from the
|
||||||
|
// end result stored in this.colors
|
||||||
|
this.currentTint = [
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
];
|
||||||
|
|
||||||
|
this.cssFixups = [
|
||||||
|
// { theme: {
|
||||||
|
// style: a style object that should be fixed up taken from a stylesheet
|
||||||
|
// attr: name of the attribute to be clobbered, e.g. 'color'
|
||||||
|
// index: ordinal of primary, secondary or tertiary
|
||||||
|
// },
|
||||||
|
// }
|
||||||
|
];
|
||||||
|
|
||||||
|
// CSS attributes to be fixed up
|
||||||
|
this.cssAttrs = [
|
||||||
|
"color",
|
||||||
|
"backgroundColor",
|
||||||
|
"borderColor",
|
||||||
|
"borderTopColor",
|
||||||
|
"borderBottomColor",
|
||||||
|
"borderLeftColor",
|
||||||
|
];
|
||||||
|
|
||||||
|
this.svgAttrs = [
|
||||||
|
"fill",
|
||||||
|
"stroke",
|
||||||
|
];
|
||||||
|
|
||||||
|
// List of functions to call when the tint changes.
|
||||||
|
this.tintables = [];
|
||||||
|
|
||||||
|
// the currently loaded theme (if any)
|
||||||
|
this.theme = undefined;
|
||||||
|
|
||||||
|
// whether to force a tint (e.g. after changing theme)
|
||||||
|
this.forceTint = false;
|
||||||
}
|
}
|
||||||
if (DEBUG) console.log("applyCssFixups end");
|
|
||||||
}
|
|
||||||
|
|
||||||
function hexToRgb(color) {
|
|
||||||
if (color[0] === '#') color = color.slice(1);
|
|
||||||
if (color.length === 3) {
|
|
||||||
color = color[0] + color[0] +
|
|
||||||
color[1] + color[1] +
|
|
||||||
color[2] + color[2];
|
|
||||||
}
|
|
||||||
var val = parseInt(color, 16);
|
|
||||||
var r = (val >> 16) & 255;
|
|
||||||
var g = (val >> 8) & 255;
|
|
||||||
var b = val & 255;
|
|
||||||
return [r, g, b];
|
|
||||||
}
|
|
||||||
|
|
||||||
function rgbToHex(rgb) {
|
|
||||||
var val = (rgb[0] << 16) | (rgb[1] << 8) | rgb[2];
|
|
||||||
return '#' + (0x1000000 + val).toString(16).slice(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
// List of functions to call when the tint changes.
|
|
||||||
const tintables = [];
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
/**
|
/**
|
||||||
* Register a callback to fire when the tint changes.
|
* Register a callback to fire when the tint changes.
|
||||||
* This is used to rewrite the tintable SVGs with the new tint.
|
* This is used to rewrite the tintable SVGs with the new tint.
|
||||||
|
@ -168,96 +145,273 @@ module.exports = {
|
||||||
*
|
*
|
||||||
* @param {Function} tintable Function to call when the tint changes.
|
* @param {Function} tintable Function to call when the tint changes.
|
||||||
*/
|
*/
|
||||||
registerTintable : function(tintable) {
|
registerTintable(tintable) {
|
||||||
tintables.push(tintable);
|
this.tintables.push(tintable);
|
||||||
},
|
}
|
||||||
|
|
||||||
tint: function(primaryColor, secondaryColor, tertiaryColor) {
|
getKeyRgb() {
|
||||||
|
return this.keyRgb;
|
||||||
|
}
|
||||||
|
|
||||||
if (!cached) {
|
tint(primaryColor, secondaryColor, tertiaryColor) {
|
||||||
calcCssFixups();
|
this.currentTint[0] = primaryColor;
|
||||||
cached = true;
|
this.currentTint[1] = secondaryColor;
|
||||||
|
this.currentTint[2] = tertiaryColor;
|
||||||
|
|
||||||
|
this.calcCssFixups();
|
||||||
|
|
||||||
|
if (DEBUG) {
|
||||||
|
console.log("Tinter.tint(" + primaryColor + ", " +
|
||||||
|
secondaryColor + ", " +
|
||||||
|
tertiaryColor + ")");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!primaryColor) {
|
if (!primaryColor) {
|
||||||
primaryColor = "#76CFA6"; // Vector green
|
primaryColor = this.keyRgb[0];
|
||||||
secondaryColor = "#EAF5F0"; // Vector light green
|
secondaryColor = this.keyRgb[1];
|
||||||
|
tertiaryColor = this.keyRgb[2];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!secondaryColor) {
|
if (!secondaryColor) {
|
||||||
var x = 0.16; // average weighting factor calculated from vector green & light green
|
const x = 0.16; // average weighting factor calculated from vector green & light green
|
||||||
var rgb = hexToRgb(primaryColor);
|
const rgb = colorToRgb(primaryColor);
|
||||||
rgb[0] = x * rgb[0] + (1 - x) * 255;
|
rgb[0] = x * rgb[0] + (1 - x) * 255;
|
||||||
rgb[1] = x * rgb[1] + (1 - x) * 255;
|
rgb[1] = x * rgb[1] + (1 - x) * 255;
|
||||||
rgb[2] = x * rgb[2] + (1 - x) * 255;
|
rgb[2] = x * rgb[2] + (1 - x) * 255;
|
||||||
secondaryColor = rgbToHex(rgb);
|
secondaryColor = rgbToColor(rgb);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!tertiaryColor) {
|
if (!tertiaryColor) {
|
||||||
var x = 0.19;
|
const x = 0.19;
|
||||||
var rgb1 = hexToRgb(primaryColor);
|
const rgb1 = colorToRgb(primaryColor);
|
||||||
var rgb2 = hexToRgb(secondaryColor);
|
const rgb2 = colorToRgb(secondaryColor);
|
||||||
rgb1[0] = x * rgb1[0] + (1 - x) * rgb2[0];
|
rgb1[0] = x * rgb1[0] + (1 - x) * rgb2[0];
|
||||||
rgb1[1] = x * rgb1[1] + (1 - x) * rgb2[1];
|
rgb1[1] = x * rgb1[1] + (1 - x) * rgb2[1];
|
||||||
rgb1[2] = x * rgb1[2] + (1 - x) * rgb2[2];
|
rgb1[2] = x * rgb1[2] + (1 - x) * rgb2[2];
|
||||||
tertiaryColor = rgbToHex(rgb1);
|
tertiaryColor = rgbToColor(rgb1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (colors[0] === primaryColor &&
|
if (this.forceTint == false &&
|
||||||
colors[1] === secondaryColor &&
|
this.colors[0] === primaryColor &&
|
||||||
colors[2] === tertiaryColor)
|
this.colors[1] === secondaryColor &&
|
||||||
{
|
this.colors[2] === tertiaryColor) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
colors = [primaryColor, secondaryColor, tertiaryColor];
|
this.forceTint = false;
|
||||||
|
|
||||||
if (DEBUG) console.log("Tinter.tint");
|
this.colors[0] = primaryColor;
|
||||||
|
this.colors[1] = secondaryColor;
|
||||||
|
this.colors[2] = tertiaryColor;
|
||||||
|
|
||||||
|
if (DEBUG) {
|
||||||
|
console.log("Tinter.tint final: (" + primaryColor + ", " +
|
||||||
|
secondaryColor + ", " +
|
||||||
|
tertiaryColor + ")");
|
||||||
|
}
|
||||||
|
|
||||||
// go through manually fixing up the stylesheets.
|
// go through manually fixing up the stylesheets.
|
||||||
applyCssFixups();
|
this.applyCssFixups();
|
||||||
|
|
||||||
// tell all the SVGs to go fix themselves up
|
// tell all the SVGs to go fix themselves up
|
||||||
// we don't do this as a dispatch otherwise it will visually lag
|
// we don't do this as a dispatch otherwise it will visually lag
|
||||||
tintables.forEach(function(tintable) {
|
this.tintables.forEach(function(tintable) {
|
||||||
tintable();
|
tintable();
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
|
|
||||||
|
tintSvgWhite(whiteColor) {
|
||||||
|
this.currentTint[3] = whiteColor;
|
||||||
|
|
||||||
|
if (!whiteColor) {
|
||||||
|
whiteColor = this.colors[3];
|
||||||
|
}
|
||||||
|
if (this.colors[3] === whiteColor) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.colors[3] = whiteColor;
|
||||||
|
this.tintables.forEach(function(tintable) {
|
||||||
|
tintable();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
tintSvgBlack(blackColor) {
|
||||||
|
this.currentTint[4] = blackColor;
|
||||||
|
|
||||||
|
if (!blackColor) {
|
||||||
|
blackColor = this.colors[4];
|
||||||
|
}
|
||||||
|
if (this.colors[4] === blackColor) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.colors[4] = blackColor;
|
||||||
|
this.tintables.forEach(function(tintable) {
|
||||||
|
tintable();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
setTheme(theme) {
|
||||||
|
console.trace("setTheme " + theme);
|
||||||
|
this.theme = theme;
|
||||||
|
|
||||||
|
// update keyRgb from the current theme CSS itself, if it defines it
|
||||||
|
if (document.getElementById('mx_theme_accentColor')) {
|
||||||
|
this.keyRgb[0] = window.getComputedStyle(
|
||||||
|
document.getElementById('mx_theme_accentColor')).color;
|
||||||
|
}
|
||||||
|
if (document.getElementById('mx_theme_secondaryAccentColor')) {
|
||||||
|
this.keyRgb[1] = window.getComputedStyle(
|
||||||
|
document.getElementById('mx_theme_secondaryAccentColor')).color;
|
||||||
|
}
|
||||||
|
if (document.getElementById('mx_theme_tertiaryAccentColor')) {
|
||||||
|
this.keyRgb[2] = window.getComputedStyle(
|
||||||
|
document.getElementById('mx_theme_tertiaryAccentColor')).color;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.calcCssFixups();
|
||||||
|
this.forceTint = true;
|
||||||
|
|
||||||
|
this.tint(this.currentTint[0], this.currentTint[1], this.currentTint[2]);
|
||||||
|
|
||||||
|
if (theme === 'dark') {
|
||||||
|
// abuse the tinter to change all the SVG's #fff to #2d2d2d
|
||||||
|
// XXX: obviously this shouldn't be hardcoded here.
|
||||||
|
this.tintSvgWhite('#2d2d2d');
|
||||||
|
this.tintSvgBlack('#dddddd');
|
||||||
|
} else {
|
||||||
|
this.tintSvgWhite('#ffffff');
|
||||||
|
this.tintSvgBlack('#000000');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
calcCssFixups() {
|
||||||
|
// cache our fixups
|
||||||
|
if (this.cssFixups[this.theme]) return;
|
||||||
|
|
||||||
|
if (DEBUG) {
|
||||||
|
console.debug("calcCssFixups start for " + this.theme + " (checking " +
|
||||||
|
document.styleSheets.length +
|
||||||
|
" stylesheets)");
|
||||||
|
}
|
||||||
|
|
||||||
|
this.cssFixups[this.theme] = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < document.styleSheets.length; i++) {
|
||||||
|
const ss = document.styleSheets[i];
|
||||||
|
if (!ss) continue; // well done safari >:(
|
||||||
|
// Chromium apparently sometimes returns null here; unsure why.
|
||||||
|
// see $14534907369972FRXBx:matrix.org in HQ
|
||||||
|
// ...ah, it's because there's a third party extension like
|
||||||
|
// privacybadger inserting its own stylesheet in there with a
|
||||||
|
// resource:// URI or something which results in a XSS error.
|
||||||
|
// See also #vector:matrix.org/$145357669685386ebCfr:matrix.org
|
||||||
|
// ...except some browsers apparently return stylesheets without
|
||||||
|
// hrefs, which we have no choice but ignore right now
|
||||||
|
|
||||||
|
// XXX seriously? we are hardcoding the name of vector's CSS file in
|
||||||
|
// here?
|
||||||
|
//
|
||||||
|
// Why do we need to limit it to vector's CSS file anyway - if there
|
||||||
|
// are other CSS files affecting the doc don't we want to apply the
|
||||||
|
// same transformations to them?
|
||||||
|
//
|
||||||
|
// Iterating through the CSS looking for matches to hack on feels
|
||||||
|
// pretty horrible anyway. And what if the application skin doesn't use
|
||||||
|
// Vector Green as its primary color?
|
||||||
|
// --richvdh
|
||||||
|
|
||||||
|
// Yes, tinting assumes that you are using the Riot skin for now.
|
||||||
|
// The right solution will be to move the CSS over to react-sdk.
|
||||||
|
// And yes, the default assets for the base skin might as well use
|
||||||
|
// Vector Green as any other colour.
|
||||||
|
// --matthew
|
||||||
|
|
||||||
|
if (ss.href && !ss.href.match(new RegExp('/theme-' + this.theme + '.css$'))) continue;
|
||||||
|
if (ss.disabled) continue;
|
||||||
|
if (!ss.cssRules) continue;
|
||||||
|
|
||||||
|
if (DEBUG) console.debug("calcCssFixups checking " + ss.cssRules.length + " rules for " + ss.href);
|
||||||
|
|
||||||
|
for (let j = 0; j < ss.cssRules.length; j++) {
|
||||||
|
const rule = ss.cssRules[j];
|
||||||
|
if (!rule.style) continue;
|
||||||
|
if (rule.selectorText && rule.selectorText.match(/#mx_theme/)) continue;
|
||||||
|
for (let k = 0; k < this.cssAttrs.length; k++) {
|
||||||
|
const attr = this.cssAttrs[k];
|
||||||
|
for (let l = 0; l < this.keyRgb.length; l++) {
|
||||||
|
if (rule.style[attr] === this.keyRgb[l]) {
|
||||||
|
this.cssFixups[this.theme].push({
|
||||||
|
style: rule.style,
|
||||||
|
attr: attr,
|
||||||
|
index: l,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (DEBUG) {
|
||||||
|
console.log("calcCssFixups end (" +
|
||||||
|
this.cssFixups[this.theme].length +
|
||||||
|
" fixups)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
applyCssFixups() {
|
||||||
|
if (DEBUG) {
|
||||||
|
console.log("applyCssFixups start (" +
|
||||||
|
this.cssFixups[this.theme].length +
|
||||||
|
" fixups)");
|
||||||
|
}
|
||||||
|
for (let i = 0; i < this.cssFixups[this.theme].length; i++) {
|
||||||
|
const cssFixup = this.cssFixups[this.theme][i];
|
||||||
|
try {
|
||||||
|
cssFixup.style[cssFixup.attr] = this.colors[cssFixup.index];
|
||||||
|
} catch (e) {
|
||||||
|
// Firefox Quantum explodes if you manually edit the CSS in the
|
||||||
|
// inspector and then try to do a tint, as apparently all the
|
||||||
|
// fixups are then stale.
|
||||||
|
console.error("Failed to apply cssFixup in Tinter! ", e.name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (DEBUG) console.log("applyCssFixups end");
|
||||||
|
}
|
||||||
|
|
||||||
// XXX: we could just move this all into TintableSvg, but as it's so similar
|
// XXX: we could just move this all into TintableSvg, but as it's so similar
|
||||||
// to the CSS fixup stuff in Tinter (just that the fixups are stored in TintableSvg)
|
// to the CSS fixup stuff in Tinter (just that the fixups are stored in TintableSvg)
|
||||||
// keeping it here for now.
|
// keeping it here for now.
|
||||||
calcSvgFixups: function(svgs) {
|
calcSvgFixups(svgs) {
|
||||||
// go through manually fixing up SVG colours.
|
// go through manually fixing up SVG colours.
|
||||||
// we could do this by stylesheets, but keeping the stylesheets
|
// we could do this by stylesheets, but keeping the stylesheets
|
||||||
// updated would be a PITA, so just brute-force search for the
|
// updated would be a PITA, so just brute-force search for the
|
||||||
// key colour; cache the element and apply.
|
// key colour; cache the element and apply.
|
||||||
|
|
||||||
if (DEBUG) console.log("calcSvgFixups start for " + svgs);
|
if (DEBUG) console.log("calcSvgFixups start for " + svgs);
|
||||||
var fixups = [];
|
const fixups = [];
|
||||||
for (var i = 0; i < svgs.length; i++) {
|
for (let i = 0; i < svgs.length; i++) {
|
||||||
var svgDoc;
|
let svgDoc;
|
||||||
try {
|
try {
|
||||||
svgDoc = svgs[i].contentDocument;
|
svgDoc = svgs[i].contentDocument;
|
||||||
}
|
} catch (e) {
|
||||||
catch(e) {
|
let msg = 'Failed to get svg.contentDocument of ' + svgs[i].toString();
|
||||||
var msg = 'Failed to get svg.contentDocument of ' + svgs[i].toString();
|
|
||||||
if (e.message) {
|
if (e.message) {
|
||||||
msg += e.message;
|
msg += e.message;
|
||||||
}
|
}
|
||||||
if (e.stack) {
|
if (e.stack) {
|
||||||
msg += ' | stack: ' + e.stack;
|
msg += ' | stack: ' + e.stack;
|
||||||
}
|
}
|
||||||
console.error(e);
|
console.error(msg);
|
||||||
}
|
}
|
||||||
if (!svgDoc) continue;
|
if (!svgDoc) continue;
|
||||||
var tags = svgDoc.getElementsByTagName("*");
|
const tags = svgDoc.getElementsByTagName("*");
|
||||||
for (var j = 0; j < tags.length; j++) {
|
for (let j = 0; j < tags.length; j++) {
|
||||||
var tag = tags[j];
|
const tag = tags[j];
|
||||||
for (var k = 0; k < svgAttrs.length; k++) {
|
for (let k = 0; k < this.svgAttrs.length; k++) {
|
||||||
var attr = svgAttrs[k];
|
const attr = this.svgAttrs[k];
|
||||||
for (var l = 0; l < keyHex.length; l++) {
|
for (let l = 0; l < this.keyHex.length; l++) {
|
||||||
if (tag.getAttribute(attr) && tag.getAttribute(attr).toUpperCase() === keyHex[l]) {
|
if (tag.getAttribute(attr) &&
|
||||||
|
tag.getAttribute(attr).toUpperCase() === this.keyHex[l]) {
|
||||||
fixups.push({
|
fixups.push({
|
||||||
node: tag,
|
node: tag,
|
||||||
attr: attr,
|
attr: attr,
|
||||||
|
@ -271,14 +425,19 @@ module.exports = {
|
||||||
if (DEBUG) console.log("calcSvgFixups end");
|
if (DEBUG) console.log("calcSvgFixups end");
|
||||||
|
|
||||||
return fixups;
|
return fixups;
|
||||||
},
|
}
|
||||||
|
|
||||||
applySvgFixups: function(fixups) {
|
applySvgFixups(fixups) {
|
||||||
if (DEBUG) console.log("applySvgFixups start for " + fixups);
|
if (DEBUG) console.log("applySvgFixups start for " + fixups);
|
||||||
for (var i = 0; i < fixups.length; i++) {
|
for (let i = 0; i < fixups.length; i++) {
|
||||||
var svgFixup = fixups[i];
|
const svgFixup = fixups[i];
|
||||||
svgFixup.node.setAttribute(svgFixup.attr, colors[svgFixup.index]);
|
svgFixup.node.setAttribute(svgFixup.attr, this.colors[svgFixup.index]);
|
||||||
}
|
}
|
||||||
if (DEBUG) console.log("applySvgFixups end");
|
if (DEBUG) console.log("applySvgFixups end");
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
|
if (global.singletonTinter === undefined) {
|
||||||
|
global.singletonTinter = new Tinter();
|
||||||
|
}
|
||||||
|
export default global.singletonTinter;
|
||||||
|
|
|
@ -14,8 +14,9 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var MatrixClientPeg = require('./MatrixClientPeg');
|
const MatrixClientPeg = require('./MatrixClientPeg');
|
||||||
var sdk = require('./index');
|
import shouldHideEvent from './shouldHideEvent';
|
||||||
|
const sdk = require('./index');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
/**
|
/**
|
||||||
|
@ -25,17 +26,37 @@ module.exports = {
|
||||||
eventTriggersUnreadCount: function(ev) {
|
eventTriggersUnreadCount: function(ev) {
|
||||||
if (ev.sender && ev.sender.userId == MatrixClientPeg.get().credentials.userId) {
|
if (ev.sender && ev.sender.userId == MatrixClientPeg.get().credentials.userId) {
|
||||||
return false;
|
return false;
|
||||||
} else if (ev.getType() == "m.room.member") {
|
} else if (ev.getType() == 'm.room.member') {
|
||||||
|
return false;
|
||||||
|
} else if (ev.getType() == 'm.call.answer' || ev.getType() == 'm.call.hangup') {
|
||||||
return false;
|
return false;
|
||||||
} else if (ev.getType == 'm.room.message' && ev.getContent().msgtype == 'm.notify') {
|
} else if (ev.getType == 'm.room.message' && ev.getContent().msgtype == 'm.notify') {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
var EventTile = sdk.getComponent('rooms.EventTile');
|
const EventTile = sdk.getComponent('rooms.EventTile');
|
||||||
return EventTile.haveTileForEvent(ev);
|
return EventTile.haveTileForEvent(ev);
|
||||||
},
|
},
|
||||||
|
|
||||||
doesRoomHaveUnreadMessages: function(room) {
|
doesRoomHaveUnreadMessages: function(room) {
|
||||||
var readUpToId = room.getEventReadUpTo(MatrixClientPeg.get().credentials.userId);
|
const myUserId = MatrixClientPeg.get().credentials.userId;
|
||||||
|
|
||||||
|
// get the most recent read receipt sent by our account.
|
||||||
|
// N.B. this is NOT a read marker (RM, aka "read up to marker"),
|
||||||
|
// despite the name of the method :((
|
||||||
|
const readUpToId = room.getEventReadUpTo(myUserId);
|
||||||
|
|
||||||
|
// as we don't send RRs for our own messages, make sure we special case that
|
||||||
|
// if *we* sent the last message into the room, we consider it not unread!
|
||||||
|
// Should fix: https://github.com/vector-im/riot-web/issues/3263
|
||||||
|
// https://github.com/vector-im/riot-web/issues/2427
|
||||||
|
// ...and possibly some of the others at
|
||||||
|
// https://github.com/vector-im/riot-web/issues/3363
|
||||||
|
if (room.timeline.length &&
|
||||||
|
room.timeline[room.timeline.length - 1].sender &&
|
||||||
|
room.timeline[room.timeline.length - 1].sender.userId === myUserId) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// this just looks at whatever history we have, which if we've only just started
|
// this just looks at whatever history we have, which if we've only just started
|
||||||
// up probably won't be very much, so if the last couple of events are ones that
|
// up probably won't be very much, so if the last couple of events are ones that
|
||||||
// don't count, we don't know if there are any events that do count between where
|
// don't count, we don't know if there are any events that do count between where
|
||||||
|
@ -43,15 +64,15 @@ module.exports = {
|
||||||
// but currently we just guess.
|
// but currently we just guess.
|
||||||
|
|
||||||
// Loop through messages, starting with the most recent...
|
// Loop through messages, starting with the most recent...
|
||||||
for (var i = room.timeline.length - 1; i >= 0; --i) {
|
for (let i = room.timeline.length - 1; i >= 0; --i) {
|
||||||
var ev = room.timeline[i];
|
const ev = room.timeline[i];
|
||||||
|
|
||||||
if (ev.getId() == readUpToId) {
|
if (ev.getId() == readUpToId) {
|
||||||
// If we've read up to this event, there's nothing more recents
|
// If we've read up to this event, there's nothing more recents
|
||||||
// that counts and we can stop looking because the user's read
|
// that counts and we can stop looking because the user's read
|
||||||
// this and everything before.
|
// this and everything before.
|
||||||
return false;
|
return false;
|
||||||
} else if (this.eventTriggersUnreadCount(ev)) {
|
} else if (!shouldHideEvent(ev) && this.eventTriggersUnreadCount(ev)) {
|
||||||
// We've found a message that counts before we hit
|
// We've found a message that counts before we hit
|
||||||
// the read marker, so this room is definitely unread.
|
// the read marker, so this room is definitely unread.
|
||||||
return true;
|
return true;
|
||||||
|
@ -62,5 +83,5 @@ module.exports = {
|
||||||
// is unread on the theory that false positives are better than
|
// is unread on the theory that false positives are better than
|
||||||
// false negatives here.
|
// false negatives here.
|
||||||
return true;
|
return true;
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -14,10 +14,10 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var dis = require("./dispatcher");
|
import dis from './dispatcher';
|
||||||
|
|
||||||
var MIN_DISPATCH_INTERVAL_MS = 500;
|
const MIN_DISPATCH_INTERVAL_MS = 500;
|
||||||
var CURRENTLY_ACTIVE_THRESHOLD_MS = 2000;
|
const CURRENTLY_ACTIVE_THRESHOLD_MS = 2000;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class watches for user activity (moving the mouse or pressing a key)
|
* This class watches for user activity (moving the mouse or pressing a key)
|
||||||
|
@ -32,7 +32,7 @@ class UserActivity {
|
||||||
start() {
|
start() {
|
||||||
document.onmousedown = this._onUserActivity.bind(this);
|
document.onmousedown = this._onUserActivity.bind(this);
|
||||||
document.onmousemove = this._onUserActivity.bind(this);
|
document.onmousemove = this._onUserActivity.bind(this);
|
||||||
document.onkeypress = this._onUserActivity.bind(this);
|
document.onkeydown = this._onUserActivity.bind(this);
|
||||||
// can't use document.scroll here because that's only the document
|
// can't use document.scroll here because that's only the document
|
||||||
// itself being scrolled. Need to use addEventListener's useCapture.
|
// itself being scrolled. Need to use addEventListener's useCapture.
|
||||||
// also this needs to be the wheel event, not scroll, as scroll is
|
// also this needs to be the wheel event, not scroll, as scroll is
|
||||||
|
@ -50,7 +50,7 @@ class UserActivity {
|
||||||
stop() {
|
stop() {
|
||||||
document.onmousedown = undefined;
|
document.onmousedown = undefined;
|
||||||
document.onmousemove = undefined;
|
document.onmousemove = undefined;
|
||||||
document.onkeypress = undefined;
|
document.onkeydown = undefined;
|
||||||
window.removeEventListener('wheel', this._onUserActivity.bind(this),
|
window.removeEventListener('wheel', this._onUserActivity.bind(this),
|
||||||
{ passive: true, capture: true });
|
{ passive: true, capture: true });
|
||||||
}
|
}
|
||||||
|
@ -58,16 +58,15 @@ class UserActivity {
|
||||||
/**
|
/**
|
||||||
* Return true if there has been user activity very recently
|
* Return true if there has been user activity very recently
|
||||||
* (ie. within a few seconds)
|
* (ie. within a few seconds)
|
||||||
|
* @returns {boolean} true if user is currently/very recently active
|
||||||
*/
|
*/
|
||||||
userCurrentlyActive() {
|
userCurrentlyActive() {
|
||||||
return this.lastActivityAtTs > new Date().getTime() - CURRENTLY_ACTIVE_THRESHOLD_MS;
|
return this.lastActivityAtTs > new Date().getTime() - CURRENTLY_ACTIVE_THRESHOLD_MS;
|
||||||
}
|
}
|
||||||
|
|
||||||
_onUserActivity(event) {
|
_onUserActivity(event) {
|
||||||
if (event.screenX && event.type == "mousemove") {
|
if (event.screenX && event.type === "mousemove") {
|
||||||
if (event.screenX === this.lastScreenX &&
|
if (event.screenX === this.lastScreenX && event.screenY === this.lastScreenY) {
|
||||||
event.screenY === this.lastScreenY)
|
|
||||||
{
|
|
||||||
// mouse hasn't actually moved
|
// mouse hasn't actually moved
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -79,28 +78,24 @@ class UserActivity {
|
||||||
if (this.lastDispatchAtTs < this.lastActivityAtTs - MIN_DISPATCH_INTERVAL_MS) {
|
if (this.lastDispatchAtTs < this.lastActivityAtTs - MIN_DISPATCH_INTERVAL_MS) {
|
||||||
this.lastDispatchAtTs = this.lastActivityAtTs;
|
this.lastDispatchAtTs = this.lastActivityAtTs;
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'user_activity'
|
action: 'user_activity',
|
||||||
});
|
});
|
||||||
if (!this.activityEndTimer) {
|
if (!this.activityEndTimer) {
|
||||||
this.activityEndTimer = setTimeout(
|
this.activityEndTimer = setTimeout(this._onActivityEndTimer.bind(this), MIN_DISPATCH_INTERVAL_MS);
|
||||||
this._onActivityEndTimer.bind(this), MIN_DISPATCH_INTERVAL_MS
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_onActivityEndTimer() {
|
_onActivityEndTimer() {
|
||||||
var now = new Date().getTime();
|
const now = new Date().getTime();
|
||||||
var targetTime = this.lastActivityAtTs + MIN_DISPATCH_INTERVAL_MS;
|
const targetTime = this.lastActivityAtTs + MIN_DISPATCH_INTERVAL_MS;
|
||||||
if (now >= targetTime) {
|
if (now >= targetTime) {
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'user_activity_end'
|
action: 'user_activity_end',
|
||||||
});
|
});
|
||||||
this.activityEndTimer = undefined;
|
this.activityEndTimer = undefined;
|
||||||
} else {
|
} else {
|
||||||
this.activityEndTimer = setTimeout(
|
this.activityEndTimer = setTimeout(this._onActivityEndTimer.bind(this), targetTime - now);
|
||||||
this._onActivityEndTimer.bind(this), targetTime - now
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
58
src/UserAddress.js
Normal file
58
src/UserAddress.js
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const emailRegex = /^\S+@\S+\.\S+$/;
|
||||||
|
|
||||||
|
const mxUserIdRegex = /^@\S+:\S+$/;
|
||||||
|
const mxRoomIdRegex = /^!\S+:\S+$/;
|
||||||
|
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
export const addressTypes = [
|
||||||
|
'mx-user-id', 'mx-room-id', 'email',
|
||||||
|
];
|
||||||
|
|
||||||
|
// PropType definition for an object describing
|
||||||
|
// an address that can be invited to a room (which
|
||||||
|
// could be a third party identifier or a matrix ID)
|
||||||
|
// along with some additional information about the
|
||||||
|
// address / target.
|
||||||
|
export const UserAddressType = PropTypes.shape({
|
||||||
|
addressType: PropTypes.oneOf(addressTypes).isRequired,
|
||||||
|
address: PropTypes.string.isRequired,
|
||||||
|
displayName: PropTypes.string,
|
||||||
|
avatarMxc: PropTypes.string,
|
||||||
|
// true if the address is known to be a valid address (eg. is a real
|
||||||
|
// user we've seen) or false otherwise (eg. is just an address the
|
||||||
|
// user has entered)
|
||||||
|
isKnown: PropTypes.bool,
|
||||||
|
});
|
||||||
|
|
||||||
|
export function getAddressType(inputText) {
|
||||||
|
const isEmailAddress = emailRegex.test(inputText);
|
||||||
|
const isUserId = mxUserIdRegex.test(inputText);
|
||||||
|
const isRoomId = mxRoomIdRegex.test(inputText);
|
||||||
|
|
||||||
|
// sanity check the input for user IDs
|
||||||
|
if (isEmailAddress) {
|
||||||
|
return 'email';
|
||||||
|
} else if (isUserId) {
|
||||||
|
return 'mx-user-id';
|
||||||
|
} else if (isRoomId) {
|
||||||
|
return 'mx-room-id';
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 New Vector Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -14,26 +15,15 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'use strict';
|
import Promise from 'bluebird';
|
||||||
var q = require("q");
|
import MatrixClientPeg from './MatrixClientPeg';
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
|
||||||
var Notifier = require("./Notifier");
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* TODO: Make things use this. This is all WIP - see UserSettings.js for usage.
|
* TODO: Make things use this. This is all WIP - see UserSettings.js for usage.
|
||||||
*/
|
*/
|
||||||
|
export default {
|
||||||
module.exports = {
|
|
||||||
LABS_FEATURES: [
|
|
||||||
{
|
|
||||||
name: 'Rich Text Editor',
|
|
||||||
id: 'rich_text_editor',
|
|
||||||
default: false,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
loadProfileInfo: function() {
|
loadProfileInfo: function() {
|
||||||
var cli = MatrixClientPeg.get();
|
const cli = MatrixClientPeg.get();
|
||||||
return cli.getProfileInfo(cli.credentials.userId);
|
return cli.getProfileInfo(cli.credentials.userId);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -43,8 +33,8 @@ module.exports = {
|
||||||
|
|
||||||
loadThreePids: function() {
|
loadThreePids: function() {
|
||||||
if (MatrixClientPeg.get().isGuest()) {
|
if (MatrixClientPeg.get().isGuest()) {
|
||||||
return q({
|
return Promise.resolve({
|
||||||
threepids: []
|
threepids: [],
|
||||||
}); // guests can't poke 3pid endpoint
|
}); // guests can't poke 3pid endpoint
|
||||||
}
|
}
|
||||||
return MatrixClientPeg.get().getThreePids();
|
return MatrixClientPeg.get().getThreePids();
|
||||||
|
@ -54,38 +44,19 @@ module.exports = {
|
||||||
// TODO
|
// TODO
|
||||||
},
|
},
|
||||||
|
|
||||||
getEnableNotifications: function() {
|
changePassword: function(oldPassword, newPassword) {
|
||||||
return Notifier.isEnabled();
|
const cli = MatrixClientPeg.get();
|
||||||
},
|
|
||||||
|
|
||||||
setEnableNotifications: function(enable) {
|
const authDict = {
|
||||||
if (!Notifier.supportsDesktopNotifications()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
Notifier.setEnabled(enable);
|
|
||||||
},
|
|
||||||
|
|
||||||
getEnableAudioNotifications: function() {
|
|
||||||
return Notifier.isAudioEnabled();
|
|
||||||
},
|
|
||||||
|
|
||||||
setEnableAudioNotifications: function(enable) {
|
|
||||||
Notifier.setAudioEnabled(enable);
|
|
||||||
},
|
|
||||||
|
|
||||||
changePassword: function(old_password, new_password) {
|
|
||||||
var cli = MatrixClientPeg.get();
|
|
||||||
|
|
||||||
var authDict = {
|
|
||||||
type: 'm.login.password',
|
type: 'm.login.password',
|
||||||
user: cli.credentials.userId,
|
user: cli.credentials.userId,
|
||||||
password: old_password
|
password: oldPassword,
|
||||||
};
|
};
|
||||||
|
|
||||||
return cli.setPassword(authDict, new_password);
|
return cli.setPassword(authDict, newPassword);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/*
|
||||||
* Returns the email pusher (pusher of type 'email') for a given
|
* Returns the email pusher (pusher of type 'email') for a given
|
||||||
* email address. Email pushers all have the same app ID, so since
|
* email address. Email pushers all have the same app ID, so since
|
||||||
* pushers are unique over (app ID, pushkey), there will be at most
|
* pushers are unique over (app ID, pushkey), there will be at most
|
||||||
|
@ -95,8 +66,8 @@ module.exports = {
|
||||||
if (pushers === undefined) {
|
if (pushers === undefined) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
for (var i = 0; i < pushers.length; ++i) {
|
for (let i = 0; i < pushers.length; ++i) {
|
||||||
if (pushers[i].kind == 'email' && pushers[i].pushkey == address) {
|
if (pushers[i].kind === 'email' && pushers[i].pushkey === address) {
|
||||||
return pushers[i];
|
return pushers[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -110,7 +81,7 @@ module.exports = {
|
||||||
addEmailPusher: function(address, data) {
|
addEmailPusher: function(address, data) {
|
||||||
return MatrixClientPeg.get().setPusher({
|
return MatrixClientPeg.get().setPusher({
|
||||||
kind: 'email',
|
kind: 'email',
|
||||||
app_id: "m.email",
|
app_id: 'm.email',
|
||||||
pushkey: address,
|
pushkey: address,
|
||||||
app_display_name: 'Email Notifications',
|
app_display_name: 'Email Notifications',
|
||||||
device_display_name: address,
|
device_display_name: address,
|
||||||
|
@ -119,52 +90,4 @@ module.exports = {
|
||||||
append: true, // We always append for email pushers since we don't want to stop other accounts notifying to the same email address
|
append: true, // We always append for email pushers since we don't want to stop other accounts notifying to the same email address
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
getUrlPreviewsDisabled: function() {
|
|
||||||
var event = MatrixClientPeg.get().getAccountData("org.matrix.preview_urls");
|
|
||||||
return (event && event.getContent().disable);
|
|
||||||
},
|
|
||||||
|
|
||||||
setUrlPreviewsDisabled: function(disabled) {
|
|
||||||
// FIXME: handle errors
|
|
||||||
return MatrixClientPeg.get().setAccountData("org.matrix.preview_urls", {
|
|
||||||
disable: disabled
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
getSyncedSettings: function() {
|
|
||||||
var event = MatrixClientPeg.get().getAccountData("im.vector.web.settings");
|
|
||||||
return event ? event.getContent() : {};
|
|
||||||
},
|
|
||||||
|
|
||||||
getSyncedSetting: function(type, defaultValue = null) {
|
|
||||||
var settings = this.getSyncedSettings();
|
|
||||||
return settings.hasOwnProperty(type) ? settings[type] : null;
|
|
||||||
},
|
|
||||||
|
|
||||||
setSyncedSetting: function(type, value) {
|
|
||||||
var settings = this.getSyncedSettings();
|
|
||||||
settings[type] = value;
|
|
||||||
// FIXME: handle errors
|
|
||||||
return MatrixClientPeg.get().setAccountData("im.vector.web.settings", settings);
|
|
||||||
},
|
|
||||||
|
|
||||||
isFeatureEnabled: function(feature: string): boolean {
|
|
||||||
// Disable labs for guests.
|
|
||||||
if (MatrixClientPeg.get().isGuest()) return false;
|
|
||||||
|
|
||||||
if (localStorage.getItem(`mx_labs_feature_${feature}`) === null) {
|
|
||||||
for (var i = 0; i < this.LABS_FEATURES.length; i++) {
|
|
||||||
var f = this.LABS_FEATURES[i];
|
|
||||||
if (f.id === feature) {
|
|
||||||
return f.default;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return localStorage.getItem(`mx_labs_feature_${feature}`) === 'true';
|
|
||||||
},
|
|
||||||
|
|
||||||
setFeatureEnabled: function(feature: string, enabled: boolean) {
|
|
||||||
localStorage.setItem(`mx_labs_feature_${feature}`, enabled);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
var React = require('react');
|
const React = require('react');
|
||||||
var ReactDom = require('react-dom');
|
const ReactDom = require('react-dom');
|
||||||
var Velocity = require('velocity-vector');
|
import PropTypes from 'prop-types';
|
||||||
|
const Velocity = require('velocity-vector');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The Velociraptor contains components and animates transitions with velocity.
|
* The Velociraptor contains components and animates transitions with velocity.
|
||||||
|
@ -14,16 +15,16 @@ module.exports = React.createClass({
|
||||||
|
|
||||||
propTypes: {
|
propTypes: {
|
||||||
// either a list of child nodes, or a single child.
|
// either a list of child nodes, or a single child.
|
||||||
children: React.PropTypes.any,
|
children: PropTypes.any,
|
||||||
|
|
||||||
// optional transition information for changing existing children
|
// optional transition information for changing existing children
|
||||||
transition: React.PropTypes.object,
|
transition: PropTypes.object,
|
||||||
|
|
||||||
// a list of state objects to apply to each child node in turn
|
// a list of state objects to apply to each child node in turn
|
||||||
startStyles: React.PropTypes.array,
|
startStyles: PropTypes.array,
|
||||||
|
|
||||||
// a list of transition options from the corresponding startStyle
|
// a list of transition options from the corresponding startStyle
|
||||||
enterTransitionOpts: React.PropTypes.array,
|
enterTransitionOpts: PropTypes.array,
|
||||||
},
|
},
|
||||||
|
|
||||||
getDefaultProps: function() {
|
getDefaultProps: function() {
|
||||||
|
@ -46,13 +47,13 @@ module.exports = React.createClass({
|
||||||
* update `this.children` according to the new list of children given
|
* update `this.children` according to the new list of children given
|
||||||
*/
|
*/
|
||||||
_updateChildren: function(newChildren) {
|
_updateChildren: function(newChildren) {
|
||||||
var self = this;
|
const self = this;
|
||||||
var oldChildren = this.children || {};
|
const oldChildren = this.children || {};
|
||||||
this.children = {};
|
this.children = {};
|
||||||
React.Children.toArray(newChildren).forEach(function(c) {
|
React.Children.toArray(newChildren).forEach(function(c) {
|
||||||
if (oldChildren[c.key]) {
|
if (oldChildren[c.key]) {
|
||||||
var old = oldChildren[c.key];
|
const old = oldChildren[c.key];
|
||||||
var oldNode = ReactDom.findDOMNode(self.nodes[old.key]);
|
const oldNode = ReactDom.findDOMNode(self.nodes[old.key]);
|
||||||
|
|
||||||
if (oldNode && oldNode.style.left != c.props.style.left) {
|
if (oldNode && oldNode.style.left != c.props.style.left) {
|
||||||
Velocity(oldNode, { left: c.props.style.left }, self.props.transition).then(function() {
|
Velocity(oldNode, { left: c.props.style.left }, self.props.transition).then(function() {
|
||||||
|
@ -62,27 +63,27 @@ module.exports = React.createClass({
|
||||||
oldNode.style.visibility = c.props.style.visibility;
|
oldNode.style.visibility = c.props.style.visibility;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
if (oldNode.style.visibility == 'hidden' && c.props.style.visibility == 'visible') {
|
|
||||||
oldNode.style.visibility = c.props.style.visibility;
|
|
||||||
}
|
|
||||||
//console.log("translation: "+oldNode.style.left+" -> "+c.props.style.left);
|
//console.log("translation: "+oldNode.style.left+" -> "+c.props.style.left);
|
||||||
}
|
}
|
||||||
|
if (oldNode && oldNode.style.visibility == 'hidden' && c.props.style.visibility == 'visible') {
|
||||||
|
oldNode.style.visibility = c.props.style.visibility;
|
||||||
|
}
|
||||||
self.children[c.key] = old;
|
self.children[c.key] = old;
|
||||||
} else {
|
} else {
|
||||||
// new element. If we have a startStyle, use that as the style and go through
|
// new element. If we have a startStyle, use that as the style and go through
|
||||||
// the enter animations
|
// the enter animations
|
||||||
var newProps = {};
|
const newProps = {};
|
||||||
var restingStyle = c.props.style;
|
const restingStyle = c.props.style;
|
||||||
|
|
||||||
var startStyles = self.props.startStyles;
|
const startStyles = self.props.startStyles;
|
||||||
if (startStyles.length > 0) {
|
if (startStyles.length > 0) {
|
||||||
var startStyle = startStyles[0]
|
const startStyle = startStyles[0];
|
||||||
newProps.style = startStyle;
|
newProps.style = startStyle;
|
||||||
// console.log("mounted@startstyle0: "+JSON.stringify(startStyle));
|
// console.log("mounted@startstyle0: "+JSON.stringify(startStyle));
|
||||||
}
|
}
|
||||||
|
|
||||||
newProps.ref = (n => self._collectNode(
|
newProps.ref = ((n) => self._collectNode(
|
||||||
c.key, n, restingStyle
|
c.key, n, restingStyle,
|
||||||
));
|
));
|
||||||
|
|
||||||
self.children[c.key] = React.cloneElement(c, newProps);
|
self.children[c.key] = React.cloneElement(c, newProps);
|
||||||
|
@ -103,9 +104,9 @@ module.exports = React.createClass({
|
||||||
this.nodes[k] === undefined &&
|
this.nodes[k] === undefined &&
|
||||||
this.props.startStyles.length > 0
|
this.props.startStyles.length > 0
|
||||||
) {
|
) {
|
||||||
var startStyles = this.props.startStyles;
|
const startStyles = this.props.startStyles;
|
||||||
var transitionOpts = this.props.enterTransitionOpts;
|
const transitionOpts = this.props.enterTransitionOpts;
|
||||||
var domNode = ReactDom.findDOMNode(node);
|
const domNode = ReactDom.findDOMNode(node);
|
||||||
// start from startStyle 1: 0 is the one we gave it
|
// start from startStyle 1: 0 is the one we gave it
|
||||||
// to start with, so now we animate 1 etc.
|
// to start with, so now we animate 1 etc.
|
||||||
for (var i = 1; i < startStyles.length; ++i) {
|
for (var i = 1; i < startStyles.length; ++i) {
|
||||||
|
@ -145,7 +146,7 @@ module.exports = React.createClass({
|
||||||
// and the FAQ entry, "Preventing memory leaks when
|
// and the FAQ entry, "Preventing memory leaks when
|
||||||
// creating/destroying large numbers of elements"
|
// creating/destroying large numbers of elements"
|
||||||
// (https://github.com/julianshapiro/velocity/issues/47)
|
// (https://github.com/julianshapiro/velocity/issues/47)
|
||||||
var domNode = ReactDom.findDOMNode(this.nodes[k]);
|
const domNode = ReactDom.findDOMNode(this.nodes[k]);
|
||||||
Velocity.Utilities.removeData(domNode);
|
Velocity.Utilities.removeData(domNode);
|
||||||
}
|
}
|
||||||
this.nodes[k] = node;
|
this.nodes[k] = node;
|
||||||
|
@ -154,7 +155,7 @@ module.exports = React.createClass({
|
||||||
render: function() {
|
render: function() {
|
||||||
return (
|
return (
|
||||||
<span>
|
<span>
|
||||||
{Object.values(this.children)}
|
{ Object.values(this.children) }
|
||||||
</span>
|
</span>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,15 +1,17 @@
|
||||||
var Velocity = require('velocity-vector');
|
const Velocity = require('velocity-vector');
|
||||||
|
|
||||||
// courtesy of https://github.com/julianshapiro/velocity/issues/283
|
// courtesy of https://github.com/julianshapiro/velocity/issues/283
|
||||||
// We only use easeOutBounce (easeInBounce is just sort of nonsensical)
|
// We only use easeOutBounce (easeInBounce is just sort of nonsensical)
|
||||||
function bounce( p ) {
|
function bounce( p ) {
|
||||||
var pow2,
|
let pow2,
|
||||||
bounce = 4;
|
bounce = 4;
|
||||||
|
|
||||||
while ( p < ( ( pow2 = Math.pow( 2, --bounce ) ) - 1 ) / 11 ) {}
|
while ( p < ( ( pow2 = Math.pow( 2, --bounce ) ) - 1 ) / 11 ) {
|
||||||
|
// just sets pow2
|
||||||
|
}
|
||||||
return 1 / Math.pow( 4, 3 - bounce ) - 7.5625 * Math.pow( ( pow2 * 3 - 2 ) / 22 - p, 2 );
|
return 1 / Math.pow( 4, 3 - bounce ) - 7.5625 * Math.pow( ( pow2 * 3 - 2 ) / 22 - p, 2 );
|
||||||
}
|
}
|
||||||
|
|
||||||
Velocity.Easings.easeOutBounce = function(p) {
|
Velocity.Easings.easeOutBounce = function(p) {
|
||||||
return 1 - bounce(1 - p);
|
return 1 - bounce(1 - p);
|
||||||
}
|
};
|
||||||
|
|
|
@ -1,9 +1,32 @@
|
||||||
var MatrixClientPeg = require("./MatrixClientPeg");
|
/*
|
||||||
|
Copyright 2017 Vector Creations 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const MatrixClientPeg = require("./MatrixClientPeg");
|
||||||
|
import { _t } from './languageHandler';
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
|
usersTypingApartFromMeAndIgnored: function(room) {
|
||||||
|
return this.usersTyping(
|
||||||
|
room, [MatrixClientPeg.get().credentials.userId].concat(MatrixClientPeg.get().getIgnoredUsers()),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
usersTypingApartFromMe: function(room) {
|
usersTypingApartFromMe: function(room) {
|
||||||
return this.usersTyping(
|
return this.usersTyping(
|
||||||
room, [MatrixClientPeg.get().credentials.userId]
|
room, [MatrixClientPeg.get().credentials.userId],
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -12,15 +35,15 @@ module.exports = {
|
||||||
* to exclude, return a list of user objects who are typing.
|
* to exclude, return a list of user objects who are typing.
|
||||||
*/
|
*/
|
||||||
usersTyping: function(room, exclude) {
|
usersTyping: function(room, exclude) {
|
||||||
var whoIsTyping = [];
|
const whoIsTyping = [];
|
||||||
|
|
||||||
if (exclude === undefined) {
|
if (exclude === undefined) {
|
||||||
exclude = [];
|
exclude = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
var memberKeys = Object.keys(room.currentState.members);
|
const memberKeys = Object.keys(room.currentState.members);
|
||||||
for (var i = 0; i < memberKeys.length; ++i) {
|
for (let i = 0; i < memberKeys.length; ++i) {
|
||||||
var userId = memberKeys[i];
|
const userId = memberKeys[i];
|
||||||
|
|
||||||
if (room.currentState.members[userId].typing) {
|
if (room.currentState.members[userId].typing) {
|
||||||
if (exclude.indexOf(userId) == -1) {
|
if (exclude.indexOf(userId) == -1) {
|
||||||
|
@ -32,18 +55,24 @@ module.exports = {
|
||||||
return whoIsTyping;
|
return whoIsTyping;
|
||||||
},
|
},
|
||||||
|
|
||||||
whoIsTypingString: function(room) {
|
whoIsTypingString: function(whoIsTyping, limit) {
|
||||||
var whoIsTyping = this.usersTypingApartFromMe(room);
|
let othersCount = 0;
|
||||||
if (whoIsTyping.length == 0) {
|
if (whoIsTyping.length > limit) {
|
||||||
return null;
|
othersCount = whoIsTyping.length - limit + 1;
|
||||||
} else if (whoIsTyping.length == 1) {
|
|
||||||
return whoIsTyping[0].name + ' is typing';
|
|
||||||
} else {
|
|
||||||
var names = whoIsTyping.map(function(m) {
|
|
||||||
return m.name;
|
|
||||||
});
|
|
||||||
var lastPerson = names.shift();
|
|
||||||
return names.join(', ') + ' and ' + lastPerson + ' are typing';
|
|
||||||
}
|
}
|
||||||
}
|
if (whoIsTyping.length == 0) {
|
||||||
}
|
return '';
|
||||||
|
} else if (whoIsTyping.length == 1) {
|
||||||
|
return _t('%(displayName)s is typing', {displayName: whoIsTyping[0].name});
|
||||||
|
}
|
||||||
|
const names = whoIsTyping.map(function(m) {
|
||||||
|
return m.name;
|
||||||
|
});
|
||||||
|
if (othersCount>=1) {
|
||||||
|
return _t('%(names)s and %(count)s others are typing', {names: names.slice(0, limit - 1).join(', '), count: othersCount});
|
||||||
|
} else {
|
||||||
|
const lastPerson = names.pop();
|
||||||
|
return _t('%(names)s and %(lastPerson)s are typing', {names: names.join(', '), lastPerson: lastPerson});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
326
src/WidgetMessaging.js
Normal file
326
src/WidgetMessaging.js
Normal file
|
@ -0,0 +1,326 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Listens for incoming postMessage requests from embedded widgets. The following API is exposed:
|
||||||
|
{
|
||||||
|
api: "widget",
|
||||||
|
action: "content_loaded",
|
||||||
|
widgetId: $WIDGET_ID,
|
||||||
|
data: {}
|
||||||
|
// additional request fields
|
||||||
|
}
|
||||||
|
|
||||||
|
The complete request object is returned to the caller with an additional "response" key like so:
|
||||||
|
{
|
||||||
|
api: "widget",
|
||||||
|
action: "content_loaded",
|
||||||
|
widgetId: $WIDGET_ID,
|
||||||
|
data: {},
|
||||||
|
// additional request fields
|
||||||
|
response: { ... }
|
||||||
|
}
|
||||||
|
|
||||||
|
The "api" field is required to use this API, and must be set to "widget" in all requests.
|
||||||
|
|
||||||
|
The "action" determines the format of the request and response. All actions can return an error response.
|
||||||
|
|
||||||
|
Additional data can be sent as additional, abritrary fields. However, typically the data object should be used.
|
||||||
|
|
||||||
|
A success response is an object with zero or more keys.
|
||||||
|
|
||||||
|
An error response is a "response" object which consists of a sole "error" key to indicate an error.
|
||||||
|
They look like:
|
||||||
|
{
|
||||||
|
error: {
|
||||||
|
message: "Unable to invite user into room.",
|
||||||
|
_error: <Original Error Object>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
The "message" key should be a human-friendly string.
|
||||||
|
|
||||||
|
ACTIONS
|
||||||
|
=======
|
||||||
|
** All actions must include an "api" field with valie "widget".**
|
||||||
|
All actions can return an error response instead of the response outlined below.
|
||||||
|
|
||||||
|
content_loaded
|
||||||
|
--------------
|
||||||
|
Indicates that widget contet has fully loaded
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- widgetId is the unique ID of the widget instance in riot / matrix state.
|
||||||
|
- No additional fields.
|
||||||
|
Response:
|
||||||
|
{
|
||||||
|
success: true
|
||||||
|
}
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
api: "widget",
|
||||||
|
action: "content_loaded",
|
||||||
|
widgetId: $WIDGET_ID
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
api_version
|
||||||
|
-----------
|
||||||
|
Get the current version of the widget postMessage API
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- No additional fields.
|
||||||
|
Response:
|
||||||
|
{
|
||||||
|
api_version: "0.0.1"
|
||||||
|
}
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
api: "widget",
|
||||||
|
action: "api_version",
|
||||||
|
}
|
||||||
|
|
||||||
|
supported_api_versions
|
||||||
|
----------------------
|
||||||
|
Get versions of the widget postMessage API that are currently supported
|
||||||
|
|
||||||
|
Request:
|
||||||
|
- No additional fields.
|
||||||
|
Response:
|
||||||
|
{
|
||||||
|
api: "widget"
|
||||||
|
supported_versions: ["0.0.1"]
|
||||||
|
}
|
||||||
|
Example:
|
||||||
|
{
|
||||||
|
api: "widget",
|
||||||
|
action: "supported_api_versions",
|
||||||
|
}
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
import URL from 'url';
|
||||||
|
|
||||||
|
const WIDGET_API_VERSION = '0.0.1'; // Current API version
|
||||||
|
const SUPPORTED_WIDGET_API_VERSIONS = [
|
||||||
|
'0.0.1',
|
||||||
|
];
|
||||||
|
|
||||||
|
import dis from './dispatcher';
|
||||||
|
|
||||||
|
if (!global.mxWidgetMessagingListenerCount) {
|
||||||
|
global.mxWidgetMessagingListenerCount = 0;
|
||||||
|
}
|
||||||
|
if (!global.mxWidgetMessagingMessageEndpoints) {
|
||||||
|
global.mxWidgetMessagingMessageEndpoints = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register widget message event listeners
|
||||||
|
*/
|
||||||
|
function startListening() {
|
||||||
|
if (global.mxWidgetMessagingListenerCount === 0) {
|
||||||
|
window.addEventListener("message", onMessage, false);
|
||||||
|
}
|
||||||
|
global.mxWidgetMessagingListenerCount += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* De-register widget message event listeners
|
||||||
|
*/
|
||||||
|
function stopListening() {
|
||||||
|
global.mxWidgetMessagingListenerCount -= 1;
|
||||||
|
if (global.mxWidgetMessagingListenerCount === 0) {
|
||||||
|
window.removeEventListener("message", onMessage);
|
||||||
|
}
|
||||||
|
if (global.mxWidgetMessagingListenerCount < 0) {
|
||||||
|
// Make an error so we get a stack trace
|
||||||
|
const e = new Error(
|
||||||
|
"WidgetMessaging: mismatched startListening / stopListening detected." +
|
||||||
|
" Negative count",
|
||||||
|
);
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a widget endpoint for trusted postMessage communication
|
||||||
|
* @param {string} widgetId Unique widget identifier
|
||||||
|
* @param {string} endpointUrl Widget wurl origin (protocol + (optional port) + host)
|
||||||
|
*/
|
||||||
|
function addEndpoint(widgetId, endpointUrl) {
|
||||||
|
const u = URL.parse(endpointUrl);
|
||||||
|
if (!u || !u.protocol || !u.host) {
|
||||||
|
console.warn("Invalid origin:", endpointUrl);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const origin = u.protocol + '//' + u.host;
|
||||||
|
const endpoint = new WidgetMessageEndpoint(widgetId, origin);
|
||||||
|
if (global.mxWidgetMessagingMessageEndpoints) {
|
||||||
|
if (global.mxWidgetMessagingMessageEndpoints.some(function(ep) {
|
||||||
|
return (ep.widgetId === widgetId && ep.endpointUrl === endpointUrl);
|
||||||
|
})) {
|
||||||
|
// Message endpoint already registered
|
||||||
|
console.warn("Endpoint already registered");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
global.mxWidgetMessagingMessageEndpoints.push(endpoint);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* De-register a widget endpoint from trusted communication sources
|
||||||
|
* @param {string} widgetId Unique widget identifier
|
||||||
|
* @param {string} endpointUrl Widget wurl origin (protocol + (optional port) + host)
|
||||||
|
* @return {boolean} True if endpoint was successfully removed
|
||||||
|
*/
|
||||||
|
function removeEndpoint(widgetId, endpointUrl) {
|
||||||
|
const u = URL.parse(endpointUrl);
|
||||||
|
if (!u || !u.protocol || !u.host) {
|
||||||
|
console.warn("Invalid origin");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const origin = u.protocol + '//' + u.host;
|
||||||
|
if (global.mxWidgetMessagingMessageEndpoints && global.mxWidgetMessagingMessageEndpoints.length > 0) {
|
||||||
|
const length = global.mxWidgetMessagingMessageEndpoints.length;
|
||||||
|
global.mxWidgetMessagingMessageEndpoints = global.mxWidgetMessagingMessageEndpoints.filter(function(endpoint) {
|
||||||
|
return (endpoint.widgetId != widgetId || endpoint.endpointUrl != origin);
|
||||||
|
});
|
||||||
|
return (length > global.mxWidgetMessagingMessageEndpoints.length);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle widget postMessage events
|
||||||
|
* @param {Event} event Event to handle
|
||||||
|
* @return {undefined}
|
||||||
|
*/
|
||||||
|
function onMessage(event) {
|
||||||
|
if (!event.origin) { // Handle chrome
|
||||||
|
event.origin = event.originalEvent.origin;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event origin is empty string if undefined
|
||||||
|
if (
|
||||||
|
event.origin.length === 0 ||
|
||||||
|
!trustedEndpoint(event.origin) ||
|
||||||
|
event.data.api !== "widget" ||
|
||||||
|
!event.data.widgetId
|
||||||
|
) {
|
||||||
|
return; // don't log this - debugging APIs like to spam postMessage which floods the log otherwise
|
||||||
|
}
|
||||||
|
|
||||||
|
const action = event.data.action;
|
||||||
|
const widgetId = event.data.widgetId;
|
||||||
|
if (action === 'content_loaded') {
|
||||||
|
dis.dispatch({
|
||||||
|
action: 'widget_content_loaded',
|
||||||
|
widgetId: widgetId,
|
||||||
|
});
|
||||||
|
sendResponse(event, {success: true});
|
||||||
|
} else if (action === 'supported_api_versions') {
|
||||||
|
sendResponse(event, {
|
||||||
|
api: "widget",
|
||||||
|
supported_versions: SUPPORTED_WIDGET_API_VERSIONS,
|
||||||
|
});
|
||||||
|
} else if (action === 'api_version') {
|
||||||
|
sendResponse(event, {
|
||||||
|
api: "widget",
|
||||||
|
version: WIDGET_API_VERSION,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.warn("Widget postMessage event unhandled");
|
||||||
|
sendError(event, {message: "The postMessage was unhandled"});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if message origin is registered as trusted
|
||||||
|
* @param {string} origin PostMessage origin to check
|
||||||
|
* @return {boolean} True if trusted
|
||||||
|
*/
|
||||||
|
function trustedEndpoint(origin) {
|
||||||
|
if (!origin) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return global.mxWidgetMessagingMessageEndpoints.some((endpoint) => {
|
||||||
|
return endpoint.endpointUrl === origin;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a postmessage response to a postMessage request
|
||||||
|
* @param {Event} event The original postMessage request event
|
||||||
|
* @param {Object} res Response data
|
||||||
|
*/
|
||||||
|
function sendResponse(event, res) {
|
||||||
|
const data = JSON.parse(JSON.stringify(event.data));
|
||||||
|
data.response = res;
|
||||||
|
event.source.postMessage(data, event.origin);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send an error response to a postMessage request
|
||||||
|
* @param {Event} event The original postMessage request event
|
||||||
|
* @param {string} msg Error message
|
||||||
|
* @param {Error} nestedError Nested error event (optional)
|
||||||
|
*/
|
||||||
|
function sendError(event, msg, nestedError) {
|
||||||
|
console.error("Action:" + event.data.action + " failed with message: " + msg);
|
||||||
|
const data = JSON.parse(JSON.stringify(event.data));
|
||||||
|
data.response = {
|
||||||
|
error: {
|
||||||
|
message: msg,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
if (nestedError) {
|
||||||
|
data.response.error._error = nestedError;
|
||||||
|
}
|
||||||
|
event.source.postMessage(data, event.origin);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents mapping of widget instance to URLs for trusted postMessage communication.
|
||||||
|
*/
|
||||||
|
class WidgetMessageEndpoint {
|
||||||
|
/**
|
||||||
|
* Mapping of widget instance to URL for trusted postMessage communication.
|
||||||
|
* @param {string} widgetId Unique widget identifier
|
||||||
|
* @param {string} endpointUrl Widget wurl origin.
|
||||||
|
*/
|
||||||
|
constructor(widgetId, endpointUrl) {
|
||||||
|
if (!widgetId) {
|
||||||
|
throw new Error("No widgetId specified in widgetMessageEndpoint constructor");
|
||||||
|
}
|
||||||
|
if (!endpointUrl) {
|
||||||
|
throw new Error("No endpoint specified in widgetMessageEndpoint constructor");
|
||||||
|
}
|
||||||
|
this.widgetId = widgetId;
|
||||||
|
this.endpointUrl = endpointUrl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
startListening: startListening,
|
||||||
|
stopListening: stopListening,
|
||||||
|
addEndpoint: addEndpoint,
|
||||||
|
removeEndpoint: removeEndpoint,
|
||||||
|
};
|
58
src/WidgetUtils.js
Normal file
58
src/WidgetUtils.js
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Vector Creations 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 MatrixClientPeg from './MatrixClientPeg';
|
||||||
|
|
||||||
|
export default class WidgetUtils {
|
||||||
|
|
||||||
|
/* Returns true if user is able to send state events to modify widgets in this room
|
||||||
|
* @param roomId -- The ID of the room to check
|
||||||
|
* @return Boolean -- true if the user can modify widgets in this room
|
||||||
|
* @throws Error -- specifies the error reason
|
||||||
|
*/
|
||||||
|
static canUserModifyWidgets(roomId) {
|
||||||
|
if (!roomId) {
|
||||||
|
console.warn('No room ID specified');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
if (!client) {
|
||||||
|
console.warn('User must be be logged in');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const room = client.getRoom(roomId);
|
||||||
|
if (!room) {
|
||||||
|
console.warn(`Room ID ${roomId} is not recognised`);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const me = client.credentials.userId;
|
||||||
|
if (!me) {
|
||||||
|
console.warn('Failed to get user ID');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const member = room.getMember(me);
|
||||||
|
if (!member || member.membership !== "join") {
|
||||||
|
console.warn(`User ${me} is not in room ${roomId}`);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return room.currentState.maySendStateEvent('im.vector.modular.widgets', me);
|
||||||
|
}
|
||||||
|
}
|
34
src/actions/GroupActions.js
Normal file
34
src/actions/GroupActions.js
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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 { asyncAction } from './actionCreators';
|
||||||
|
|
||||||
|
const GroupActions = {};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an action thunk that will do an asynchronous request to fetch
|
||||||
|
* the groups to which a user is joined.
|
||||||
|
*
|
||||||
|
* @param {MatrixClient} matrixClient the matrix client to query.
|
||||||
|
* @returns {function} an action thunk that will dispatch actions
|
||||||
|
* indicating the status of the request.
|
||||||
|
* @see asyncAction
|
||||||
|
*/
|
||||||
|
GroupActions.fetchJoinedGroups = function(matrixClient) {
|
||||||
|
return asyncAction('GroupActions.fetchJoinedGroups', () => matrixClient.getJoinedGroups());
|
||||||
|
};
|
||||||
|
|
||||||
|
export default GroupActions;
|
108
src/actions/MatrixActionCreators.js
Normal file
108
src/actions/MatrixActionCreators.js
Normal file
|
@ -0,0 +1,108 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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 dis from '../dispatcher';
|
||||||
|
|
||||||
|
// TODO: migrate from sync_state to MatrixActions.sync so that more js-sdk events
|
||||||
|
// become dispatches in the same place.
|
||||||
|
/**
|
||||||
|
* Create a MatrixActions.sync action that represents a MatrixClient `sync` event,
|
||||||
|
* each parameter mapping to a key-value in the action.
|
||||||
|
*
|
||||||
|
* @param {MatrixClient} matrixClient the matrix client
|
||||||
|
* @param {string} state the current sync state.
|
||||||
|
* @param {string} prevState the previous sync state.
|
||||||
|
* @returns {Object} an action of type MatrixActions.sync.
|
||||||
|
*/
|
||||||
|
function createSyncAction(matrixClient, state, prevState) {
|
||||||
|
return {
|
||||||
|
action: 'MatrixActions.sync',
|
||||||
|
state,
|
||||||
|
prevState,
|
||||||
|
matrixClient,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef AccountDataAction
|
||||||
|
* @type {Object}
|
||||||
|
* @property {string} action 'MatrixActions.accountData'.
|
||||||
|
* @property {MatrixEvent} event the MatrixEvent that triggered the dispatch.
|
||||||
|
* @property {string} event_type the type of the MatrixEvent, e.g. "m.direct".
|
||||||
|
* @property {Object} event_content the content of the MatrixEvent.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a MatrixActions.accountData action that represents a MatrixClient `accountData`
|
||||||
|
* matrix event.
|
||||||
|
*
|
||||||
|
* @param {MatrixClient} matrixClient the matrix client.
|
||||||
|
* @param {MatrixEvent} accountDataEvent the account data event.
|
||||||
|
* @returns {AccountDataAction} an action of type MatrixActions.accountData.
|
||||||
|
*/
|
||||||
|
function createAccountDataAction(matrixClient, accountDataEvent) {
|
||||||
|
return {
|
||||||
|
action: 'MatrixActions.accountData',
|
||||||
|
event: accountDataEvent,
|
||||||
|
event_type: accountDataEvent.getType(),
|
||||||
|
event_content: accountDataEvent.getContent(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This object is responsible for dispatching actions when certain events are emitted by
|
||||||
|
* the given MatrixClient.
|
||||||
|
*/
|
||||||
|
export default {
|
||||||
|
// A list of callbacks to call to unregister all listeners added
|
||||||
|
_matrixClientListenersStop: [],
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start listening to certain events from the MatrixClient and dispatch actions when
|
||||||
|
* they are emitted.
|
||||||
|
* @param {MatrixClient} matrixClient the MatrixClient to listen to events from
|
||||||
|
*/
|
||||||
|
start(matrixClient) {
|
||||||
|
this._addMatrixClientListener(matrixClient, 'sync', createSyncAction);
|
||||||
|
this._addMatrixClientListener(matrixClient, 'accountData', createAccountDataAction);
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start listening to events of type eventName on matrixClient and when they are emitted,
|
||||||
|
* dispatch an action created by the actionCreator function.
|
||||||
|
* @param {MatrixClient} matrixClient a MatrixClient to register a listener with.
|
||||||
|
* @param {string} eventName the event to listen to on MatrixClient.
|
||||||
|
* @param {function} actionCreator a function that should return an action to dispatch
|
||||||
|
* when given the MatrixClient as an argument as well as
|
||||||
|
* arguments emitted in the MatrixClient event.
|
||||||
|
*/
|
||||||
|
_addMatrixClientListener(matrixClient, eventName, actionCreator) {
|
||||||
|
const listener = (...args) => {
|
||||||
|
dis.dispatch(actionCreator(matrixClient, ...args));
|
||||||
|
};
|
||||||
|
matrixClient.on(eventName, listener);
|
||||||
|
this._matrixClientListenersStop.push(() => {
|
||||||
|
matrixClient.removeListener(eventName, listener);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop listening to events.
|
||||||
|
*/
|
||||||
|
stop() {
|
||||||
|
this._matrixClientListenersStop.forEach((stopListener) => stopListener());
|
||||||
|
},
|
||||||
|
};
|
59
src/actions/TagOrderActions.js
Normal file
59
src/actions/TagOrderActions.js
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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 Analytics from '../Analytics';
|
||||||
|
import { asyncAction } from './actionCreators';
|
||||||
|
import TagOrderStore from '../stores/TagOrderStore';
|
||||||
|
|
||||||
|
const TagOrderActions = {};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an action thunk that will do an asynchronous request to
|
||||||
|
* move a tag in TagOrderStore to destinationIx.
|
||||||
|
*
|
||||||
|
* @param {MatrixClient} matrixClient the matrix client to set the
|
||||||
|
* account data on.
|
||||||
|
* @param {string} tag the tag to move.
|
||||||
|
* @param {number} destinationIx the new position of the tag.
|
||||||
|
* @returns {function} an action thunk that will dispatch actions
|
||||||
|
* indicating the status of the request.
|
||||||
|
* @see asyncAction
|
||||||
|
*/
|
||||||
|
TagOrderActions.moveTag = function(matrixClient, tag, destinationIx) {
|
||||||
|
// Only commit tags if the state is ready, i.e. not null
|
||||||
|
let tags = TagOrderStore.getOrderedTags();
|
||||||
|
if (!tags) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
tags = tags.filter((t) => t !== tag);
|
||||||
|
tags = [...tags.slice(0, destinationIx), tag, ...tags.slice(destinationIx)];
|
||||||
|
|
||||||
|
const storeId = TagOrderStore.getStoreId();
|
||||||
|
|
||||||
|
return asyncAction('TagOrderActions.moveTag', () => {
|
||||||
|
Analytics.trackEvent('TagOrderActions', 'commitTagOrdering');
|
||||||
|
return matrixClient.setAccountData(
|
||||||
|
'im.vector.web.tag_ordering',
|
||||||
|
{tags, _storeId: storeId},
|
||||||
|
);
|
||||||
|
}, () => {
|
||||||
|
// For an optimistic update
|
||||||
|
return {tags};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export default TagOrderActions;
|
48
src/actions/actionCreators.js
Normal file
48
src/actions/actionCreators.js
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an action thunk that will dispatch actions indicating the current
|
||||||
|
* status of the Promise returned by fn.
|
||||||
|
*
|
||||||
|
* @param {string} id the id to give the dispatched actions. This is given a
|
||||||
|
* suffix determining whether it is pending, successful or
|
||||||
|
* a failure.
|
||||||
|
* @param {function} fn a function that returns a Promise.
|
||||||
|
* @param {function?} pendingFn a function that returns an object to assign
|
||||||
|
* to the `request` key of the ${id}.pending
|
||||||
|
* payload.
|
||||||
|
* @returns {function} an action thunk - a function that uses its single
|
||||||
|
* argument as a dispatch function to dispatch the
|
||||||
|
* following actions:
|
||||||
|
* `${id}.pending` and either
|
||||||
|
* `${id}.success` or
|
||||||
|
* `${id}.failure`.
|
||||||
|
*/
|
||||||
|
export function asyncAction(id, fn, pendingFn) {
|
||||||
|
return (dispatch) => {
|
||||||
|
dispatch({
|
||||||
|
action: id + '.pending',
|
||||||
|
request:
|
||||||
|
typeof pendingFn === 'function' ? pendingFn() : undefined,
|
||||||
|
});
|
||||||
|
fn().then((result) => {
|
||||||
|
dispatch({action: id + '.success', result});
|
||||||
|
}).catch((err) => {
|
||||||
|
dispatch({action: id + '.failure', err});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
|
@ -14,36 +14,46 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var React = require("react");
|
const React = require("react");
|
||||||
var sdk = require('../../../index');
|
import PropTypes from 'prop-types';
|
||||||
var MatrixClientPeg = require("../../../MatrixClientPeg");
|
import { _t } from '../../../languageHandler';
|
||||||
|
const sdk = require('../../../index');
|
||||||
|
const MatrixClientPeg = require("../../../MatrixClientPeg");
|
||||||
|
|
||||||
module.exports = React.createClass({
|
module.exports = React.createClass({
|
||||||
displayName: 'EncryptedEventDialog',
|
displayName: 'EncryptedEventDialog',
|
||||||
|
|
||||||
propTypes: {
|
propTypes: {
|
||||||
event: React.PropTypes.object.isRequired,
|
event: PropTypes.object.isRequired,
|
||||||
onFinished: React.PropTypes.func.isRequired,
|
onFinished: PropTypes.func.isRequired,
|
||||||
},
|
},
|
||||||
|
|
||||||
getInitialState: function() {
|
getInitialState: function() {
|
||||||
return { device: this.refreshDevice() };
|
return { device: null };
|
||||||
},
|
},
|
||||||
|
|
||||||
componentWillMount: function() {
|
componentWillMount: function() {
|
||||||
this._unmounted = false;
|
this._unmounted = false;
|
||||||
var client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
client.on("deviceVerificationChanged", this.onDeviceVerificationChanged);
|
|
||||||
|
|
||||||
// no need to redownload keys if we already have the device
|
// first try to load the device from our store.
|
||||||
if (this.state.device) {
|
//
|
||||||
return;
|
this.refreshDevice().then((dev) => {
|
||||||
}
|
if (dev) {
|
||||||
client.downloadKeys([this.props.event.getSender()], true).done(()=>{
|
return dev;
|
||||||
|
}
|
||||||
|
|
||||||
|
// tell the client to try to refresh the device list for this user
|
||||||
|
return client.downloadKeys([this.props.event.getSender()], true).then(() => {
|
||||||
|
return this.refreshDevice();
|
||||||
|
});
|
||||||
|
}).then((dev) => {
|
||||||
if (this._unmounted) {
|
if (this._unmounted) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.setState({ device: this.refreshDevice() });
|
|
||||||
|
this.setState({ device: dev });
|
||||||
|
client.on("deviceVerificationChanged", this.onDeviceVerificationChanged);
|
||||||
}, (err)=>{
|
}, (err)=>{
|
||||||
console.log("Error downloading devices", err);
|
console.log("Error downloading devices", err);
|
||||||
});
|
});
|
||||||
|
@ -51,19 +61,23 @@ module.exports = React.createClass({
|
||||||
|
|
||||||
componentWillUnmount: function() {
|
componentWillUnmount: function() {
|
||||||
this._unmounted = true;
|
this._unmounted = true;
|
||||||
var client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
if (client) {
|
if (client) {
|
||||||
client.removeListener("deviceVerificationChanged", this.onDeviceVerificationChanged);
|
client.removeListener("deviceVerificationChanged", this.onDeviceVerificationChanged);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
refreshDevice: function() {
|
refreshDevice: function() {
|
||||||
return MatrixClientPeg.get().getEventSenderDeviceInfo(this.props.event);
|
// Promise.resolve to handle transition from static result to promise; can be removed
|
||||||
|
// in future
|
||||||
|
return Promise.resolve(MatrixClientPeg.get().getEventSenderDeviceInfo(this.props.event));
|
||||||
},
|
},
|
||||||
|
|
||||||
onDeviceVerificationChanged: function(userId, device) {
|
onDeviceVerificationChanged: function(userId, device) {
|
||||||
if (userId == this.props.event.getSender()) {
|
if (userId == this.props.event.getSender()) {
|
||||||
this.setState({ device: this.refreshDevice() });
|
this.refreshDevice().then((dev) => {
|
||||||
|
this.setState({ device: dev });
|
||||||
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -76,36 +90,36 @@ module.exports = React.createClass({
|
||||||
},
|
},
|
||||||
|
|
||||||
_renderDeviceInfo: function() {
|
_renderDeviceInfo: function() {
|
||||||
var device = this.state.device;
|
const device = this.state.device;
|
||||||
if (!device) {
|
if (!device) {
|
||||||
return (<i>unknown device</i>);
|
return (<i>{ _t('unknown device') }</i>);
|
||||||
}
|
}
|
||||||
|
|
||||||
var verificationStatus = (<b>NOT verified</b>);
|
let verificationStatus = (<b>{ _t('NOT verified') }</b>);
|
||||||
if (device.isBlocked()) {
|
if (device.isBlocked()) {
|
||||||
verificationStatus = (<b>Blacklisted</b>);
|
verificationStatus = (<b>{ _t('Blacklisted') }</b>);
|
||||||
} else if (device.isVerified()) {
|
} else if (device.isVerified()) {
|
||||||
verificationStatus = "verified";
|
verificationStatus = _t('verified');
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<table>
|
<table>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Name</td>
|
<td>{ _t('Name') }</td>
|
||||||
<td>{ device.getDisplayName() }</td>
|
<td>{ device.getDisplayName() }</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Device ID</td>
|
<td>{ _t('Device ID') }</td>
|
||||||
<td><code>{ device.deviceId }</code></td>
|
<td><code>{ device.deviceId }</code></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Verification</td>
|
<td>{ _t('Verification') }</td>
|
||||||
<td>{ verificationStatus }</td>
|
<td>{ verificationStatus }</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Ed25519 fingerprint</td>
|
<td>{ _t('Ed25519 fingerprint') }</td>
|
||||||
<td><code>{device.getFingerprint()}</code></td>
|
<td><code>{ device.getFingerprint() }</code></td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
@ -113,38 +127,38 @@ module.exports = React.createClass({
|
||||||
},
|
},
|
||||||
|
|
||||||
_renderEventInfo: function() {
|
_renderEventInfo: function() {
|
||||||
var event = this.props.event;
|
const event = this.props.event;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<table>
|
<table>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td>User ID</td>
|
<td>{ _t('User ID') }</td>
|
||||||
<td>{ event.getSender() }</td>
|
<td>{ event.getSender() }</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Curve25519 identity key</td>
|
<td>{ _t('Curve25519 identity key') }</td>
|
||||||
<td><code>{ event.getSenderKey() || <i>none</i> }</code></td>
|
<td><code>{ event.getSenderKey() || <i>{ _t('none') }</i> }</code></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Claimed Ed25519 fingerprint key</td>
|
<td>{ _t('Claimed Ed25519 fingerprint key') }</td>
|
||||||
<td><code>{ event.getKeysClaimed().ed25519 || <i>none</i> }</code></td>
|
<td><code>{ event.getKeysClaimed().ed25519 || <i>{ _t('none') }</i> }</code></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Algorithm</td>
|
<td>{ _t('Algorithm') }</td>
|
||||||
<td>{ event.getWireContent().algorithm || <i>unencrypted</i> }</td>
|
<td>{ event.getWireContent().algorithm || <i>{ _t('unencrypted') }</i> }</td>
|
||||||
</tr>
|
</tr>
|
||||||
{
|
{
|
||||||
event.getContent().msgtype === 'm.bad.encrypted' ? (
|
event.getContent().msgtype === 'm.bad.encrypted' ? (
|
||||||
<tr>
|
<tr>
|
||||||
<td>Decryption error</td>
|
<td>{ _t('Decryption error') }</td>
|
||||||
<td>{ event.getContent().body }</td>
|
<td>{ event.getContent().body }</td>
|
||||||
</tr>
|
</tr>
|
||||||
) : null
|
) : null
|
||||||
}
|
}
|
||||||
<tr>
|
<tr>
|
||||||
<td>Session ID</td>
|
<td>{ _t('Session ID') }</td>
|
||||||
<td><code>{ event.getWireContent().session_id || <i>none</i> }</code></td>
|
<td><code>{ event.getWireContent().session_id || <i>{ _t('none') }</i> }</code></td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
@ -152,36 +166,36 @@ module.exports = React.createClass({
|
||||||
},
|
},
|
||||||
|
|
||||||
render: function() {
|
render: function() {
|
||||||
var DeviceVerifyButtons = sdk.getComponent('elements.DeviceVerifyButtons');
|
const DeviceVerifyButtons = sdk.getComponent('elements.DeviceVerifyButtons');
|
||||||
|
|
||||||
var buttons = null;
|
let buttons = null;
|
||||||
if (this.state.device) {
|
if (this.state.device) {
|
||||||
buttons = (
|
buttons = (
|
||||||
<DeviceVerifyButtons device={ this.state.device }
|
<DeviceVerifyButtons device={this.state.device}
|
||||||
userId={ this.props.event.getSender() }
|
userId={this.props.event.getSender()}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mx_EncryptedEventDialog" onKeyDown={ this.onKeyDown }>
|
<div className="mx_EncryptedEventDialog" onKeyDown={this.onKeyDown}>
|
||||||
<div className="mx_Dialog_title">
|
<div className="mx_Dialog_title">
|
||||||
End-to-end encryption information
|
{ _t('End-to-end encryption information') }
|
||||||
</div>
|
</div>
|
||||||
<div className="mx_Dialog_content">
|
<div className="mx_Dialog_content">
|
||||||
<h4>Event information</h4>
|
<h4>{ _t('Event information') }</h4>
|
||||||
{this._renderEventInfo()}
|
{ this._renderEventInfo() }
|
||||||
|
|
||||||
<h4>Sender device information</h4>
|
<h4>{ _t('Sender device information') }</h4>
|
||||||
{this._renderDeviceInfo()}
|
{ this._renderDeviceInfo() }
|
||||||
</div>
|
</div>
|
||||||
<div className="mx_Dialog_buttons">
|
<div className="mx_Dialog_buttons">
|
||||||
<button className="mx_Dialog_primary" onClick={ this.props.onFinished } autoFocus={ true }>
|
<button className="mx_Dialog_primary" onClick={this.props.onFinished} autoFocus={true}>
|
||||||
OK
|
{ _t('OK') }
|
||||||
</button>
|
</button>
|
||||||
{buttons}
|
{ buttons }
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
},
|
||||||
});
|
});
|
183
src/async-components/views/dialogs/ExportE2eKeysDialog.js
Normal file
183
src/async-components/views/dialogs/ExportE2eKeysDialog.js
Normal file
|
@ -0,0 +1,183 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Vector Creations 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 FileSaver from 'file-saver';
|
||||||
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { _t } from '../../../languageHandler';
|
||||||
|
|
||||||
|
import * as Matrix from 'matrix-js-sdk';
|
||||||
|
import * as MegolmExportEncryption from '../../../utils/MegolmExportEncryption';
|
||||||
|
import sdk from '../../../index';
|
||||||
|
|
||||||
|
const PHASE_EDIT = 1;
|
||||||
|
const PHASE_EXPORTING = 2;
|
||||||
|
|
||||||
|
export default React.createClass({
|
||||||
|
displayName: 'ExportE2eKeysDialog',
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
matrixClient: PropTypes.instanceOf(Matrix.MatrixClient).isRequired,
|
||||||
|
onFinished: PropTypes.func.isRequired,
|
||||||
|
},
|
||||||
|
|
||||||
|
getInitialState: function() {
|
||||||
|
return {
|
||||||
|
phase: PHASE_EDIT,
|
||||||
|
errStr: null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillMount: function() {
|
||||||
|
this._unmounted = false;
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillUnmount: function() {
|
||||||
|
this._unmounted = true;
|
||||||
|
},
|
||||||
|
|
||||||
|
_onPassphraseFormSubmit: function(ev) {
|
||||||
|
ev.preventDefault();
|
||||||
|
|
||||||
|
const passphrase = this.refs.passphrase1.value;
|
||||||
|
if (passphrase !== this.refs.passphrase2.value) {
|
||||||
|
this.setState({errStr: _t('Passphrases must match')});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!passphrase) {
|
||||||
|
this.setState({errStr: _t('Passphrase must not be empty')});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
this._startExport(passphrase);
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
_startExport: function(passphrase) {
|
||||||
|
// extra Promise.resolve() to turn synchronous exceptions into
|
||||||
|
// asynchronous ones.
|
||||||
|
Promise.resolve().then(() => {
|
||||||
|
return this.props.matrixClient.exportRoomKeys();
|
||||||
|
}).then((k) => {
|
||||||
|
return MegolmExportEncryption.encryptMegolmKeyFile(
|
||||||
|
JSON.stringify(k), passphrase,
|
||||||
|
);
|
||||||
|
}).then((f) => {
|
||||||
|
const blob = new Blob([f], {
|
||||||
|
type: 'text/plain;charset=us-ascii',
|
||||||
|
});
|
||||||
|
FileSaver.saveAs(blob, 'riot-keys.txt');
|
||||||
|
this.props.onFinished(true);
|
||||||
|
}).catch((e) => {
|
||||||
|
console.error("Error exporting e2e keys:", e);
|
||||||
|
if (this._unmounted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const msg = e.friendlyText || _t('Unknown error');
|
||||||
|
this.setState({
|
||||||
|
errStr: msg,
|
||||||
|
phase: PHASE_EDIT,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
errStr: null,
|
||||||
|
phase: PHASE_EXPORTING,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
_onCancelClick: function(ev) {
|
||||||
|
ev.preventDefault();
|
||||||
|
this.props.onFinished(false);
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
render: function() {
|
||||||
|
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
|
||||||
|
|
||||||
|
const disableForm = (this.state.phase === PHASE_EXPORTING);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseDialog className='mx_exportE2eKeysDialog'
|
||||||
|
onFinished={this.props.onFinished}
|
||||||
|
title={_t("Export room keys")}
|
||||||
|
>
|
||||||
|
<form onSubmit={this._onPassphraseFormSubmit}>
|
||||||
|
<div className="mx_Dialog_content">
|
||||||
|
<p>
|
||||||
|
{ _t(
|
||||||
|
'This process allows you to export the keys for messages ' +
|
||||||
|
'you have received in encrypted rooms to a local file. You ' +
|
||||||
|
'will then be able to import the file into another Matrix ' +
|
||||||
|
'client in the future, so that client will also be able to ' +
|
||||||
|
'decrypt these messages.',
|
||||||
|
) }
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
{ _t(
|
||||||
|
'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.',
|
||||||
|
) }
|
||||||
|
</p>
|
||||||
|
<div className='error'>
|
||||||
|
{ this.state.errStr }
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputTable'>
|
||||||
|
<div className='mx_E2eKeysDialog_inputRow'>
|
||||||
|
<div className='mx_E2eKeysDialog_inputLabel'>
|
||||||
|
<label htmlFor='passphrase1'>
|
||||||
|
{ _t("Enter passphrase") }
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputCell'>
|
||||||
|
<input ref='passphrase1' id='passphrase1'
|
||||||
|
autoFocus={true} size='64' type='password'
|
||||||
|
disabled={disableForm}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputRow'>
|
||||||
|
<div className='mx_E2eKeysDialog_inputLabel'>
|
||||||
|
<label htmlFor='passphrase2'>
|
||||||
|
{ _t("Confirm passphrase") }
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputCell'>
|
||||||
|
<input ref='passphrase2' id='passphrase2'
|
||||||
|
size='64' type='password'
|
||||||
|
disabled={disableForm}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className='mx_Dialog_buttons'>
|
||||||
|
<input className='mx_Dialog_primary' type='submit' value={_t('Export')}
|
||||||
|
disabled={disableForm}
|
||||||
|
/>
|
||||||
|
<button onClick={this._onCancelClick} disabled={disableForm}>
|
||||||
|
{ _t("Cancel") }
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</BaseDialog>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
181
src/async-components/views/dialogs/ImportE2eKeysDialog.js
Normal file
181
src/async-components/views/dialogs/ImportE2eKeysDialog.js
Normal file
|
@ -0,0 +1,181 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Vector Creations 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 * as Matrix from 'matrix-js-sdk';
|
||||||
|
import * as MegolmExportEncryption from '../../../utils/MegolmExportEncryption';
|
||||||
|
import sdk from '../../../index';
|
||||||
|
import { _t } from '../../../languageHandler';
|
||||||
|
|
||||||
|
function readFileAsArrayBuffer(file) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = (e) => {
|
||||||
|
resolve(e.target.result);
|
||||||
|
};
|
||||||
|
reader.onerror = reject;
|
||||||
|
|
||||||
|
reader.readAsArrayBuffer(file);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const PHASE_EDIT = 1;
|
||||||
|
const PHASE_IMPORTING = 2;
|
||||||
|
|
||||||
|
export default React.createClass({
|
||||||
|
displayName: 'ImportE2eKeysDialog',
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
matrixClient: PropTypes.instanceOf(Matrix.MatrixClient).isRequired,
|
||||||
|
onFinished: PropTypes.func.isRequired,
|
||||||
|
},
|
||||||
|
|
||||||
|
getInitialState: function() {
|
||||||
|
return {
|
||||||
|
enableSubmit: false,
|
||||||
|
phase: PHASE_EDIT,
|
||||||
|
errStr: null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillMount: function() {
|
||||||
|
this._unmounted = false;
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillUnmount: function() {
|
||||||
|
this._unmounted = true;
|
||||||
|
},
|
||||||
|
|
||||||
|
_onFormChange: function(ev) {
|
||||||
|
const files = this.refs.file.files || [];
|
||||||
|
this.setState({
|
||||||
|
enableSubmit: (this.refs.passphrase.value !== "" && files.length > 0),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
_onFormSubmit: function(ev) {
|
||||||
|
ev.preventDefault();
|
||||||
|
this._startImport(this.refs.file.files[0], this.refs.passphrase.value);
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
_startImport: function(file, passphrase) {
|
||||||
|
this.setState({
|
||||||
|
errStr: null,
|
||||||
|
phase: PHASE_IMPORTING,
|
||||||
|
});
|
||||||
|
|
||||||
|
return readFileAsArrayBuffer(file).then((arrayBuffer) => {
|
||||||
|
return MegolmExportEncryption.decryptMegolmKeyFile(
|
||||||
|
arrayBuffer, passphrase,
|
||||||
|
);
|
||||||
|
}).then((keys) => {
|
||||||
|
return this.props.matrixClient.importRoomKeys(JSON.parse(keys));
|
||||||
|
}).then(() => {
|
||||||
|
// TODO: it would probably be nice to give some feedback about what we've imported here.
|
||||||
|
this.props.onFinished(true);
|
||||||
|
}).catch((e) => {
|
||||||
|
console.error("Error importing e2e keys:", e);
|
||||||
|
if (this._unmounted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const msg = e.friendlyText || _t('Unknown error');
|
||||||
|
this.setState({
|
||||||
|
errStr: msg,
|
||||||
|
phase: PHASE_EDIT,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
_onCancelClick: function(ev) {
|
||||||
|
ev.preventDefault();
|
||||||
|
this.props.onFinished(false);
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
render: function() {
|
||||||
|
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
|
||||||
|
|
||||||
|
const disableForm = (this.state.phase !== PHASE_EDIT);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseDialog className='mx_importE2eKeysDialog'
|
||||||
|
onFinished={this.props.onFinished}
|
||||||
|
title={_t("Import room keys")}
|
||||||
|
>
|
||||||
|
<form onSubmit={this._onFormSubmit}>
|
||||||
|
<div className="mx_Dialog_content">
|
||||||
|
<p>
|
||||||
|
{ _t(
|
||||||
|
'This process allows you to import encryption keys ' +
|
||||||
|
'that you had previously exported from another Matrix ' +
|
||||||
|
'client. You will then be able to decrypt any ' +
|
||||||
|
'messages that the other client could decrypt.',
|
||||||
|
) }
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
{ _t(
|
||||||
|
'The export file will be protected with a passphrase. ' +
|
||||||
|
'You should enter the passphrase here, to decrypt the file.',
|
||||||
|
) }
|
||||||
|
</p>
|
||||||
|
<div className='error'>
|
||||||
|
{ this.state.errStr }
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputTable'>
|
||||||
|
<div className='mx_E2eKeysDialog_inputRow'>
|
||||||
|
<div className='mx_E2eKeysDialog_inputLabel'>
|
||||||
|
<label htmlFor='importFile'>
|
||||||
|
{ _t("File to import") }
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputCell'>
|
||||||
|
<input ref='file' id='importFile' type='file'
|
||||||
|
autoFocus={true}
|
||||||
|
onChange={this._onFormChange}
|
||||||
|
disabled={disableForm} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputRow'>
|
||||||
|
<div className='mx_E2eKeysDialog_inputLabel'>
|
||||||
|
<label htmlFor='passphrase'>
|
||||||
|
{ _t("Enter passphrase") }
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div className='mx_E2eKeysDialog_inputCell'>
|
||||||
|
<input ref='passphrase' id='passphrase'
|
||||||
|
size='64' type='password'
|
||||||
|
onChange={this._onFormChange}
|
||||||
|
disabled={disableForm} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className='mx_Dialog_buttons'>
|
||||||
|
<input className='mx_Dialog_primary' type='submit' value={_t('Import')}
|
||||||
|
disabled={!this.state.enableSubmit || disableForm}
|
||||||
|
/>
|
||||||
|
<button onClick={this._onCancelClick} disabled={disableForm}>
|
||||||
|
{ _t("Cancel") }
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</BaseDialog>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
|
@ -1,8 +1,26 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 Aviral Dasgupta
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
Copyright 2017 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 React from 'react';
|
||||||
import type {Completion, SelectionRange} from './Autocompleter';
|
import type {Completion, SelectionRange} from './Autocompleter';
|
||||||
|
|
||||||
export default class AutocompleteProvider {
|
export default class AutocompleteProvider {
|
||||||
constructor(commandRegex?: RegExp, fuseOpts?: any) {
|
constructor(commandRegex?: RegExp) {
|
||||||
if (commandRegex) {
|
if (commandRegex) {
|
||||||
if (!commandRegex.global) {
|
if (!commandRegex.global) {
|
||||||
throw new Error('commandRegex must have global flag set');
|
throw new Error('commandRegex must have global flag set');
|
||||||
|
@ -11,6 +29,10 @@ export default class AutocompleteProvider {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
destroy() {
|
||||||
|
// stub
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Of the matched commands in the query, returns the first that contains or is contained by the selection, or null.
|
* Of the matched commands in the query, returns the first that contains or is contained by the selection, or null.
|
||||||
*/
|
*/
|
||||||
|
@ -26,7 +48,7 @@ export default class AutocompleteProvider {
|
||||||
}
|
}
|
||||||
|
|
||||||
commandRegex.lastIndex = 0;
|
commandRegex.lastIndex = 0;
|
||||||
|
|
||||||
let match;
|
let match;
|
||||||
while ((match = commandRegex.exec(query)) != null) {
|
while ((match = commandRegex.exec(query)) != null) {
|
||||||
let matchStart = match.index,
|
let matchStart = match.index,
|
||||||
|
|
|
@ -1,3 +1,20 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 Aviral Dasgupta
|
||||||
|
Copyright 2017 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.
|
||||||
|
*/
|
||||||
|
|
||||||
// @flow
|
// @flow
|
||||||
|
|
||||||
import type {Component} from 'react';
|
import type {Component} from 'react';
|
||||||
|
@ -6,7 +23,8 @@ import DuckDuckGoProvider from './DuckDuckGoProvider';
|
||||||
import RoomProvider from './RoomProvider';
|
import RoomProvider from './RoomProvider';
|
||||||
import UserProvider from './UserProvider';
|
import UserProvider from './UserProvider';
|
||||||
import EmojiProvider from './EmojiProvider';
|
import EmojiProvider from './EmojiProvider';
|
||||||
import Q from 'q';
|
import NotifProvider from './NotifProvider';
|
||||||
|
import Promise from 'bluebird';
|
||||||
|
|
||||||
export type SelectionRange = {
|
export type SelectionRange = {
|
||||||
start: number,
|
start: number,
|
||||||
|
@ -18,46 +36,68 @@ export type Completion = {
|
||||||
component: ?Component,
|
component: ?Component,
|
||||||
range: SelectionRange,
|
range: SelectionRange,
|
||||||
command: ?string,
|
command: ?string,
|
||||||
|
// If provided, apply a LINK entity to the completion with the
|
||||||
|
// data = { url: href }.
|
||||||
|
href: ?string,
|
||||||
};
|
};
|
||||||
|
|
||||||
const PROVIDERS = [
|
const PROVIDERS = [
|
||||||
UserProvider,
|
UserProvider,
|
||||||
RoomProvider,
|
RoomProvider,
|
||||||
EmojiProvider,
|
EmojiProvider,
|
||||||
|
NotifProvider,
|
||||||
CommandProvider,
|
CommandProvider,
|
||||||
DuckDuckGoProvider,
|
DuckDuckGoProvider,
|
||||||
].map(completer => completer.getInstance());
|
];
|
||||||
|
|
||||||
// Providers will get rejected if they take longer than this.
|
// Providers will get rejected if they take longer than this.
|
||||||
const PROVIDER_COMPLETION_TIMEOUT = 3000;
|
const PROVIDER_COMPLETION_TIMEOUT = 3000;
|
||||||
|
|
||||||
export async function getCompletions(query: string, selection: SelectionRange, force: boolean = false): Array<Completion> {
|
export default class Autocompleter {
|
||||||
/* Note: That this waits for all providers to return is *intentional*
|
constructor(room) {
|
||||||
otherwise, we run into a condition where new completions are displayed
|
this.room = room;
|
||||||
while the user is interacting with the list, which makes it difficult
|
this.providers = PROVIDERS.map((p) => {
|
||||||
to predict whether an action will actually do what is intended
|
return new p(room);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
It ends up containing a list of Q promise states, which are objects with
|
destroy() {
|
||||||
state (== "fulfilled" || "rejected") and value. */
|
this.providers.forEach((p) => {
|
||||||
const completionsList = await Q.allSettled(
|
p.destroy();
|
||||||
PROVIDERS.map(provider => {
|
});
|
||||||
return Q(provider.getCompletions(query, selection, force))
|
}
|
||||||
.timeout(PROVIDER_COMPLETION_TIMEOUT);
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
return completionsList
|
async getCompletions(query: string, selection: SelectionRange, force: boolean = false): Array<Completion> {
|
||||||
.filter(completion => completion.state === "fulfilled")
|
/* Note: This intentionally waits for all providers to return,
|
||||||
.map((completionsState, i) => {
|
otherwise, we run into a condition where new completions are displayed
|
||||||
|
while the user is interacting with the list, which makes it difficult
|
||||||
|
to predict whether an action will actually do what is intended
|
||||||
|
*/
|
||||||
|
const completionsList = await Promise.all(
|
||||||
|
// Array of inspections of promises that might timeout. Instead of allowing a
|
||||||
|
// single timeout to reject the Promise.all, reflect each one and once they've all
|
||||||
|
// settled, filter for the fulfilled ones
|
||||||
|
this.providers.map((provider) => {
|
||||||
|
return provider
|
||||||
|
.getCompletions(query, selection, force)
|
||||||
|
.timeout(PROVIDER_COMPLETION_TIMEOUT)
|
||||||
|
.reflect();
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
return completionsList.filter(
|
||||||
|
(inspection) => inspection.isFulfilled(),
|
||||||
|
).map((completionsState, i) => {
|
||||||
return {
|
return {
|
||||||
completions: completionsState.value,
|
completions: completionsState.value(),
|
||||||
provider: PROVIDERS[i],
|
provider: this.providers[i],
|
||||||
|
|
||||||
/* the currently matched "command" the completer tried to complete
|
/* the currently matched "command" the completer tried to complete
|
||||||
* we pass this through so that Autocomplete can figure out when to
|
* we pass this through so that Autocomplete can figure out when to
|
||||||
* re-show itself once hidden.
|
* re-show itself once hidden.
|
||||||
*/
|
*/
|
||||||
command: PROVIDERS[i].getCurrentCommand(query, selection, force),
|
command: this.providers[i].getCurrentCommand(query, selection, force),
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,74 +1,134 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 Aviral Dasgupta
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
Copyright 2017 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 React from 'react';
|
||||||
|
import { _t, _td } from '../languageHandler';
|
||||||
import AutocompleteProvider from './AutocompleteProvider';
|
import AutocompleteProvider from './AutocompleteProvider';
|
||||||
import Fuse from 'fuse.js';
|
import FuzzyMatcher from './FuzzyMatcher';
|
||||||
import {TextualCompletion} from './Components';
|
import {TextualCompletion} from './Components';
|
||||||
|
|
||||||
|
// TODO merge this with the factory mechanics of SlashCommands?
|
||||||
|
// Warning: Since the description string will be translated in _t(result.description), all these strings below must be in i18n/strings/en_EN.json file
|
||||||
const COMMANDS = [
|
const COMMANDS = [
|
||||||
{
|
{
|
||||||
command: '/me',
|
command: '/me',
|
||||||
args: '<message>',
|
args: '<message>',
|
||||||
description: 'Displays action',
|
description: _td('Displays action'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/ban',
|
command: '/ban',
|
||||||
args: '<user-id> [reason]',
|
args: '<user-id> [reason]',
|
||||||
description: 'Bans user with given id',
|
description: _td('Bans user with given id'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/unban',
|
||||||
|
args: '<user-id>',
|
||||||
|
description: _td('Unbans user with given id'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/op',
|
||||||
|
args: '<user-id> [<power-level>]',
|
||||||
|
description: _td('Define the power level of a user'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/deop',
|
command: '/deop',
|
||||||
args: '<user-id>',
|
args: '<user-id>',
|
||||||
description: 'Deops user with given id',
|
description: _td('Deops user with given id'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/invite',
|
command: '/invite',
|
||||||
args: '<user-id>',
|
args: '<user-id>',
|
||||||
description: 'Invites user with given id to current room',
|
description: _td('Invites user with given id to current room'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/join',
|
command: '/join',
|
||||||
args: '<room-alias>',
|
args: '<room-alias>',
|
||||||
description: 'Joins room with given alias',
|
description: _td('Joins room with given alias'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/part',
|
||||||
|
args: '[<room-alias>]',
|
||||||
|
description: _td('Leave room'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/topic',
|
||||||
|
args: '<topic>',
|
||||||
|
description: _td('Sets the room topic'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/kick',
|
command: '/kick',
|
||||||
args: '<user-id> [reason]',
|
args: '<user-id> [reason]',
|
||||||
description: 'Kicks user with given id',
|
description: _td('Kicks user with given id'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/nick',
|
command: '/nick',
|
||||||
args: '<display-name>',
|
args: '<display-name>',
|
||||||
description: 'Changes your display nickname',
|
description: _td('Changes your display nickname'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
command: '/ddg',
|
command: '/ddg',
|
||||||
args: '<query>',
|
args: '<query>',
|
||||||
description: 'Searches DuckDuckGo for results',
|
description: _td('Searches DuckDuckGo for results'),
|
||||||
}
|
},
|
||||||
|
{
|
||||||
|
command: '/tint',
|
||||||
|
args: '<color1> [<color2>]',
|
||||||
|
description: _td('Changes colour scheme of current room'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/verify',
|
||||||
|
args: '<user-id> <device-id> <device-signing-key>',
|
||||||
|
description: _td('Verifies a user, device, and pubkey tuple'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/ignore',
|
||||||
|
args: '<user-id>',
|
||||||
|
description: _td('Ignores a user, hiding their messages from you'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
command: '/unignore',
|
||||||
|
args: '<user-id>',
|
||||||
|
description: _td('Stops ignoring a user, showing their messages going forward'),
|
||||||
|
},
|
||||||
|
// Omitting `/markdown` as it only seems to apply to OldComposer
|
||||||
];
|
];
|
||||||
|
|
||||||
let COMMAND_RE = /(^\/\w*)/g;
|
const COMMAND_RE = /(^\/\w*)/g;
|
||||||
|
|
||||||
let instance = null;
|
|
||||||
|
|
||||||
export default class CommandProvider extends AutocompleteProvider {
|
export default class CommandProvider extends AutocompleteProvider {
|
||||||
constructor() {
|
constructor() {
|
||||||
super(COMMAND_RE);
|
super(COMMAND_RE);
|
||||||
this.fuse = new Fuse(COMMANDS, {
|
this.matcher = new FuzzyMatcher(COMMANDS, {
|
||||||
keys: ['command', 'args', 'description'],
|
keys: ['command', 'args', 'description'],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getCompletions(query: string, selection: {start: number, end: number}) {
|
async getCompletions(query: string, selection: {start: number, end: number}) {
|
||||||
let completions = [];
|
let completions = [];
|
||||||
let {command, range} = this.getCurrentCommand(query, selection);
|
const {command, range} = this.getCurrentCommand(query, selection);
|
||||||
if (command) {
|
if (command) {
|
||||||
completions = this.fuse.search(command[0]).map(result => {
|
completions = this.matcher.match(command[0]).map((result) => {
|
||||||
return {
|
return {
|
||||||
completion: result.command + ' ',
|
completion: result.command + ' ',
|
||||||
component: (<TextualCompletion
|
component: (<TextualCompletion
|
||||||
title={result.command}
|
title={result.command}
|
||||||
subtitle={result.args}
|
subtitle={result.args}
|
||||||
description={result.description}
|
description={_t(result.description)}
|
||||||
/>),
|
/>),
|
||||||
range,
|
range,
|
||||||
};
|
};
|
||||||
|
@ -78,19 +138,12 @@ export default class CommandProvider extends AutocompleteProvider {
|
||||||
}
|
}
|
||||||
|
|
||||||
getName() {
|
getName() {
|
||||||
return '*️⃣ Commands';
|
return '*️⃣ ' + _t('Commands');
|
||||||
}
|
|
||||||
|
|
||||||
static getInstance(): CommandProvider {
|
|
||||||
if (instance == null)
|
|
||||||
instance = new CommandProvider();
|
|
||||||
|
|
||||||
return instance;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
renderCompletions(completions: [React.Component]): ?React.Component {
|
renderCompletions(completions: [React.Component]): ?React.Component {
|
||||||
return <div className="mx_Autocomplete_Completion_container_block">
|
return <div className="mx_Autocomplete_Completion_container_block">
|
||||||
{completions}
|
{ completions }
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,21 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 Aviral Dasgupta
|
||||||
|
|
||||||
|
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 React from 'react';
|
||||||
import ReactDOM from 'react-dom';
|
import PropTypes from 'prop-types';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
|
||||||
/* These were earlier stateless functional components but had to be converted
|
/* These were earlier stateless functional components but had to be converted
|
||||||
|
@ -15,22 +31,22 @@ export class TextualCompletion extends React.Component {
|
||||||
subtitle,
|
subtitle,
|
||||||
description,
|
description,
|
||||||
className,
|
className,
|
||||||
...restProps,
|
...restProps
|
||||||
} = this.props;
|
} = this.props;
|
||||||
return (
|
return (
|
||||||
<div className={classNames('mx_Autocomplete_Completion_block', className)} {...restProps}>
|
<div className={classNames('mx_Autocomplete_Completion_block', className)} {...restProps}>
|
||||||
<span className="mx_Autocomplete_Completion_title">{title}</span>
|
<span className="mx_Autocomplete_Completion_title">{ title }</span>
|
||||||
<span className="mx_Autocomplete_Completion_subtitle">{subtitle}</span>
|
<span className="mx_Autocomplete_Completion_subtitle">{ subtitle }</span>
|
||||||
<span className="mx_Autocomplete_Completion_description">{description}</span>
|
<span className="mx_Autocomplete_Completion_description">{ description }</span>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
TextualCompletion.propTypes = {
|
TextualCompletion.propTypes = {
|
||||||
title: React.PropTypes.string,
|
title: PropTypes.string,
|
||||||
subtitle: React.PropTypes.string,
|
subtitle: PropTypes.string,
|
||||||
description: React.PropTypes.string,
|
description: PropTypes.string,
|
||||||
className: React.PropTypes.string,
|
className: PropTypes.string,
|
||||||
};
|
};
|
||||||
|
|
||||||
export class PillCompletion extends React.Component {
|
export class PillCompletion extends React.Component {
|
||||||
|
@ -41,22 +57,22 @@ export class PillCompletion extends React.Component {
|
||||||
description,
|
description,
|
||||||
initialComponent,
|
initialComponent,
|
||||||
className,
|
className,
|
||||||
...restProps,
|
...restProps
|
||||||
} = this.props;
|
} = this.props;
|
||||||
return (
|
return (
|
||||||
<div className={classNames('mx_Autocomplete_Completion_pill', className)} {...restProps}>
|
<div className={classNames('mx_Autocomplete_Completion_pill', className)} {...restProps}>
|
||||||
{initialComponent}
|
{ initialComponent }
|
||||||
<span className="mx_Autocomplete_Completion_title">{title}</span>
|
<span className="mx_Autocomplete_Completion_title">{ title }</span>
|
||||||
<span className="mx_Autocomplete_Completion_subtitle">{subtitle}</span>
|
<span className="mx_Autocomplete_Completion_subtitle">{ subtitle }</span>
|
||||||
<span className="mx_Autocomplete_Completion_description">{description}</span>
|
<span className="mx_Autocomplete_Completion_description">{ description }</span>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
PillCompletion.propTypes = {
|
PillCompletion.propTypes = {
|
||||||
title: React.PropTypes.string,
|
title: PropTypes.string,
|
||||||
subtitle: React.PropTypes.string,
|
subtitle: PropTypes.string,
|
||||||
description: React.PropTypes.string,
|
description: PropTypes.string,
|
||||||
initialComponent: React.PropTypes.element,
|
initialComponent: PropTypes.element,
|
||||||
className: React.PropTypes.string,
|
className: PropTypes.string,
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,4 +1,23 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 Aviral Dasgupta
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
Copyright 2017 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 React from 'react';
|
||||||
|
import { _t } from '../languageHandler';
|
||||||
import AutocompleteProvider from './AutocompleteProvider';
|
import AutocompleteProvider from './AutocompleteProvider';
|
||||||
import 'whatwg-fetch';
|
import 'whatwg-fetch';
|
||||||
|
|
||||||
|
@ -7,20 +26,18 @@ import {TextualCompletion} from './Components';
|
||||||
const DDG_REGEX = /\/ddg\s+(.+)$/g;
|
const DDG_REGEX = /\/ddg\s+(.+)$/g;
|
||||||
const REFERRER = 'vector';
|
const REFERRER = 'vector';
|
||||||
|
|
||||||
let instance = null;
|
|
||||||
|
|
||||||
export default class DuckDuckGoProvider extends AutocompleteProvider {
|
export default class DuckDuckGoProvider extends AutocompleteProvider {
|
||||||
constructor() {
|
constructor() {
|
||||||
super(DDG_REGEX);
|
super(DDG_REGEX);
|
||||||
}
|
}
|
||||||
|
|
||||||
static getQueryUri(query: String) {
|
static getQueryUri(query: String) {
|
||||||
return `https://api.duckduckgo.com/?q=${encodeURIComponent(query)}`
|
return `https://api.duckduckgo.com/?q=${encodeURIComponent(query)}`
|
||||||
+ `&format=json&no_redirect=1&no_html=1&t=${encodeURIComponent(REFERRER)}`;
|
+ `&format=json&no_redirect=1&no_html=1&t=${encodeURIComponent(REFERRER)}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getCompletions(query: string, selection: {start: number, end: number}) {
|
async getCompletions(query: string, selection: {start: number, end: number}) {
|
||||||
let {command, range} = this.getCurrentCommand(query, selection);
|
const {command, range} = this.getCurrentCommand(query, selection);
|
||||||
if (!query || !command) {
|
if (!query || !command) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
@ -29,7 +46,7 @@ export default class DuckDuckGoProvider extends AutocompleteProvider {
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
});
|
});
|
||||||
const json = await response.json();
|
const json = await response.json();
|
||||||
let results = json.Results.map(result => {
|
const results = json.Results.map((result) => {
|
||||||
return {
|
return {
|
||||||
completion: result.Text,
|
completion: result.Text,
|
||||||
component: (
|
component: (
|
||||||
|
@ -75,19 +92,12 @@ export default class DuckDuckGoProvider extends AutocompleteProvider {
|
||||||
}
|
}
|
||||||
|
|
||||||
getName() {
|
getName() {
|
||||||
return '🔍 Results from DuckDuckGo';
|
return '🔍 ' + _t('Results from DuckDuckGo');
|
||||||
}
|
|
||||||
|
|
||||||
static getInstance(): DuckDuckGoProvider {
|
|
||||||
if (instance == null) {
|
|
||||||
instance = new DuckDuckGoProvider();
|
|
||||||
}
|
|
||||||
return instance;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
renderCompletions(completions: [React.Component]): ?React.Component {
|
renderCompletions(completions: [React.Component]): ?React.Component {
|
||||||
return <div className="mx_Autocomplete_Completion_container_block">
|
return <div className="mx_Autocomplete_Completion_container_block">
|
||||||
{completions}
|
{ completions }
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,56 +1,158 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 Aviral Dasgupta
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
Copyright 2017 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 React from 'react';
|
||||||
|
import { _t } from '../languageHandler';
|
||||||
import AutocompleteProvider from './AutocompleteProvider';
|
import AutocompleteProvider from './AutocompleteProvider';
|
||||||
import {emojioneList, shortnameToImage, shortnameToUnicode} from 'emojione';
|
import {emojioneList, shortnameToImage, shortnameToUnicode, asciiRegexp, unicodeRegexp} from 'emojione';
|
||||||
import Fuse from 'fuse.js';
|
import FuzzyMatcher from './FuzzyMatcher';
|
||||||
import sdk from '../index';
|
import sdk from '../index';
|
||||||
import {PillCompletion} from './Components';
|
import {PillCompletion} from './Components';
|
||||||
import type {SelectionRange, Completion} from './Autocompleter';
|
import type {SelectionRange, Completion} from './Autocompleter';
|
||||||
|
import _uniq from 'lodash/uniq';
|
||||||
|
import _sortBy from 'lodash/sortBy';
|
||||||
|
import SettingsStore from "../settings/SettingsStore";
|
||||||
|
|
||||||
const EMOJI_REGEX = /:\w*:?/g;
|
import EmojiData from '../stripped-emoji.json';
|
||||||
const EMOJI_SHORTNAMES = Object.keys(emojioneList);
|
|
||||||
|
|
||||||
let instance = null;
|
const LIMIT = 20;
|
||||||
|
const CATEGORY_ORDER = [
|
||||||
|
'people',
|
||||||
|
'food',
|
||||||
|
'objects',
|
||||||
|
'activity',
|
||||||
|
'nature',
|
||||||
|
'travel',
|
||||||
|
'flags',
|
||||||
|
'regional',
|
||||||
|
'symbols',
|
||||||
|
'modifier',
|
||||||
|
];
|
||||||
|
|
||||||
|
// Match for ":wink:" or ascii-style ";-)" provided by emojione
|
||||||
|
// (^|\s|(emojiUnicode)) to make sure we're either at the start of the string or there's a
|
||||||
|
// whitespace character or an emoji before the emoji. The reason for unicodeRegexp is
|
||||||
|
// that we need to support inputting multiple emoji with no space between them.
|
||||||
|
const EMOJI_REGEX = new RegExp('(?:^|\\s|' + unicodeRegexp + ')(' + asciiRegexp + '|:\\w*:?)$', 'g');
|
||||||
|
|
||||||
|
// We also need to match the non-zero-length prefixes to remove them from the final match,
|
||||||
|
// and update the range so that we don't replace the whitespace or the previous emoji.
|
||||||
|
const MATCH_PREFIX_REGEX = new RegExp('(\\s|' + unicodeRegexp + ')');
|
||||||
|
|
||||||
|
const EMOJI_SHORTNAMES = Object.keys(EmojiData).map((key) => EmojiData[key]).sort(
|
||||||
|
(a, b) => {
|
||||||
|
if (a.category === b.category) {
|
||||||
|
return a.emoji_order - b.emoji_order;
|
||||||
|
}
|
||||||
|
return CATEGORY_ORDER.indexOf(a.category) - CATEGORY_ORDER.indexOf(b.category);
|
||||||
|
},
|
||||||
|
).map((a, index) => {
|
||||||
|
return {
|
||||||
|
name: a.name,
|
||||||
|
shortname: a.shortname,
|
||||||
|
aliases_ascii: a.aliases_ascii ? a.aliases_ascii.join(' ') : '',
|
||||||
|
// Include the index so that we can preserve the original order
|
||||||
|
_orderBy: index,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
function score(query, space) {
|
||||||
|
const index = space.indexOf(query);
|
||||||
|
if (index === -1) {
|
||||||
|
return Infinity;
|
||||||
|
} else {
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default class EmojiProvider extends AutocompleteProvider {
|
export default class EmojiProvider extends AutocompleteProvider {
|
||||||
constructor() {
|
constructor() {
|
||||||
super(EMOJI_REGEX);
|
super(EMOJI_REGEX);
|
||||||
this.fuse = new Fuse(EMOJI_SHORTNAMES);
|
this.matcher = new FuzzyMatcher(EMOJI_SHORTNAMES, {
|
||||||
|
keys: ['aliases_ascii', 'shortname'],
|
||||||
|
// For matching against ascii equivalents
|
||||||
|
shouldMatchWordsOnly: false,
|
||||||
|
});
|
||||||
|
this.nameMatcher = new FuzzyMatcher(EMOJI_SHORTNAMES, {
|
||||||
|
keys: ['name'],
|
||||||
|
// For removing punctuation
|
||||||
|
shouldMatchWordsOnly: true,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getCompletions(query: string, selection: SelectionRange) {
|
async getCompletions(query: string, selection: SelectionRange) {
|
||||||
|
if (SettingsStore.getValue("MessageComposerInput.dontSuggestEmoji")) {
|
||||||
|
return []; // don't give any suggestions if the user doesn't want them
|
||||||
|
}
|
||||||
|
|
||||||
const EmojiText = sdk.getComponent('views.elements.EmojiText');
|
const EmojiText = sdk.getComponent('views.elements.EmojiText');
|
||||||
|
|
||||||
let completions = [];
|
let completions = [];
|
||||||
let {command, range} = this.getCurrentCommand(query, selection);
|
const {command, range} = this.getCurrentCommand(query, selection);
|
||||||
if (command) {
|
if (command) {
|
||||||
completions = this.fuse.search(command[0]).map(result => {
|
let matchedString = command[0];
|
||||||
const shortname = EMOJI_SHORTNAMES[result];
|
|
||||||
|
// Remove prefix of any length (single whitespace or unicode emoji)
|
||||||
|
const prefixMatch = MATCH_PREFIX_REGEX.exec(matchedString);
|
||||||
|
if (prefixMatch) {
|
||||||
|
matchedString = matchedString.slice(prefixMatch[0].length);
|
||||||
|
range.start += prefixMatch[0].length;
|
||||||
|
}
|
||||||
|
completions = this.matcher.match(matchedString);
|
||||||
|
|
||||||
|
// Do second match with shouldMatchWordsOnly in order to match against 'name'
|
||||||
|
completions = completions.concat(this.nameMatcher.match(matchedString));
|
||||||
|
|
||||||
|
const sorters = [];
|
||||||
|
// First, sort by score (Infinity if matchedString not in shortname)
|
||||||
|
sorters.push((c) => score(matchedString, c.shortname));
|
||||||
|
// If the matchedString is not empty, sort by length of shortname. Example:
|
||||||
|
// matchedString = ":bookmark"
|
||||||
|
// completions = [":bookmark:", ":bookmark_tabs:", ...]
|
||||||
|
if (matchedString.length > 1) {
|
||||||
|
sorters.push((c) => c.shortname.length);
|
||||||
|
}
|
||||||
|
// Finally, sort by original ordering
|
||||||
|
sorters.push((c) => c._orderBy);
|
||||||
|
completions = _sortBy(_uniq(completions), sorters);
|
||||||
|
|
||||||
|
completions = completions.map((result) => {
|
||||||
|
const {shortname} = result;
|
||||||
const unicode = shortnameToUnicode(shortname);
|
const unicode = shortnameToUnicode(shortname);
|
||||||
return {
|
return {
|
||||||
completion: unicode,
|
completion: unicode,
|
||||||
component: (
|
component: (
|
||||||
<PillCompletion title={shortname} initialComponent={<EmojiText style={{maxWidth: '1em'}}>{unicode}</EmojiText>} />
|
<PillCompletion title={shortname} initialComponent={<EmojiText style={{maxWidth: '1em'}}>{ unicode }</EmojiText>} />
|
||||||
),
|
),
|
||||||
range,
|
range,
|
||||||
};
|
};
|
||||||
}).slice(0, 8);
|
}).slice(0, LIMIT);
|
||||||
}
|
}
|
||||||
return completions;
|
return completions;
|
||||||
}
|
}
|
||||||
|
|
||||||
getName() {
|
getName() {
|
||||||
return '😃 Emoji';
|
return '😃 ' + _t('Emoji');
|
||||||
}
|
|
||||||
|
|
||||||
static getInstance() {
|
|
||||||
if (instance == null)
|
|
||||||
instance = new EmojiProvider();
|
|
||||||
return instance;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
renderCompletions(completions: [React.Component]): ?React.Component {
|
renderCompletions(completions: [React.Component]): ?React.Component {
|
||||||
return <div className="mx_Autocomplete_Completion_container_pill">
|
return <div className="mx_Autocomplete_Completion_container_pill">
|
||||||
{completions}
|
{ completions }
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
107
src/autocomplete/FuzzyMatcher.js
Normal file
107
src/autocomplete/FuzzyMatcher.js
Normal file
|
@ -0,0 +1,107 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Aviral Dasgupta
|
||||||
|
|
||||||
|
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 Levenshtein from 'liblevenshtein';
|
||||||
|
//import _at from 'lodash/at';
|
||||||
|
//import _flatMap from 'lodash/flatMap';
|
||||||
|
//import _sortBy from 'lodash/sortBy';
|
||||||
|
//import _sortedUniq from 'lodash/sortedUniq';
|
||||||
|
//import _keys from 'lodash/keys';
|
||||||
|
//
|
||||||
|
//class KeyMap {
|
||||||
|
// keys: Array<String>;
|
||||||
|
// objectMap: {[String]: Array<Object>};
|
||||||
|
// priorityMap: {[String]: number}
|
||||||
|
//}
|
||||||
|
//
|
||||||
|
//const DEFAULT_RESULT_COUNT = 10;
|
||||||
|
//const DEFAULT_DISTANCE = 5;
|
||||||
|
|
||||||
|
// FIXME Until Fuzzy matching works better, we use prefix matching.
|
||||||
|
|
||||||
|
import PrefixMatcher from './QueryMatcher';
|
||||||
|
export default PrefixMatcher;
|
||||||
|
|
||||||
|
//class FuzzyMatcher { // eslint-disable-line no-unused-vars
|
||||||
|
// /**
|
||||||
|
// * @param {object[]} objects the objects to perform a match on
|
||||||
|
// * @param {string[]} keys an array of keys within each object to match on
|
||||||
|
// * Keys can refer to object properties by name and as in JavaScript (for nested properties)
|
||||||
|
// *
|
||||||
|
// * To use, simply presort objects by required criteria, run through this function and create a FuzzyMatcher with the
|
||||||
|
// * resulting KeyMap.
|
||||||
|
// *
|
||||||
|
// * TODO: Handle arrays and objects (Fuse did this, RoomProvider uses it)
|
||||||
|
// * @return {KeyMap}
|
||||||
|
// */
|
||||||
|
// static valuesToKeyMap(objects: Array<Object>, keys: Array<String>): KeyMap {
|
||||||
|
// const keyMap = new KeyMap();
|
||||||
|
// const map = {};
|
||||||
|
// const priorities = {};
|
||||||
|
//
|
||||||
|
// objects.forEach((object, i) => {
|
||||||
|
// const keyValues = _at(object, keys);
|
||||||
|
// console.log(object, keyValues, keys);
|
||||||
|
// for (const keyValue of keyValues) {
|
||||||
|
// if (!map.hasOwnProperty(keyValue)) {
|
||||||
|
// map[keyValue] = [];
|
||||||
|
// }
|
||||||
|
// map[keyValue].push(object);
|
||||||
|
// }
|
||||||
|
// priorities[object] = i;
|
||||||
|
// });
|
||||||
|
//
|
||||||
|
// keyMap.objectMap = map;
|
||||||
|
// keyMap.priorityMap = priorities;
|
||||||
|
// keyMap.keys = _sortBy(_keys(map), [(value) => priorities[value]]);
|
||||||
|
// return keyMap;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// constructor(objects: Array<Object>, options: {[Object]: Object} = {}) {
|
||||||
|
// this.options = options;
|
||||||
|
// this.keys = options.keys;
|
||||||
|
// this.setObjects(objects);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// setObjects(objects: Array<Object>) {
|
||||||
|
// this.keyMap = FuzzyMatcher.valuesToKeyMap(objects, this.keys);
|
||||||
|
// console.log(this.keyMap.keys);
|
||||||
|
// this.matcher = new Levenshtein.Builder()
|
||||||
|
// .dictionary(this.keyMap.keys, true)
|
||||||
|
// .algorithm('transposition')
|
||||||
|
// .sort_candidates(false)
|
||||||
|
// .case_insensitive_sort(true)
|
||||||
|
// .include_distance(true)
|
||||||
|
// .maximum_candidates(this.options.resultCount || DEFAULT_RESULT_COUNT) // result count 0 doesn't make much sense
|
||||||
|
// .build();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// match(query: String): Array<Object> {
|
||||||
|
// const candidates = this.matcher.transduce(query, this.options.distance || DEFAULT_DISTANCE);
|
||||||
|
// // TODO FIXME This is hideous. Clean up when possible.
|
||||||
|
// const val = _sortedUniq(_sortBy(_flatMap(candidates, (candidate) => {
|
||||||
|
// return this.keyMap.objectMap[candidate[0]].map((value) => {
|
||||||
|
// return {
|
||||||
|
// distance: candidate[1],
|
||||||
|
// ...value,
|
||||||
|
// };
|
||||||
|
// });
|
||||||
|
// }),
|
||||||
|
// [(candidate) => candidate.distance, (candidate) => this.keyMap.priorityMap[candidate]]));
|
||||||
|
// console.log(val);
|
||||||
|
// return val;
|
||||||
|
// }
|
||||||
|
//}
|
62
src/autocomplete/NotifProvider.js
Normal file
62
src/autocomplete/NotifProvider.js
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 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 AutocompleteProvider from './AutocompleteProvider';
|
||||||
|
import { _t } from '../languageHandler';
|
||||||
|
import MatrixClientPeg from '../MatrixClientPeg';
|
||||||
|
import {PillCompletion} from './Components';
|
||||||
|
import sdk from '../index';
|
||||||
|
|
||||||
|
const AT_ROOM_REGEX = /@\S*/g;
|
||||||
|
|
||||||
|
export default class NotifProvider extends AutocompleteProvider {
|
||||||
|
constructor(room) {
|
||||||
|
super(AT_ROOM_REGEX);
|
||||||
|
this.room = room;
|
||||||
|
}
|
||||||
|
|
||||||
|
async getCompletions(query: string, selection: {start: number, end: number}, force = false) {
|
||||||
|
const RoomAvatar = sdk.getComponent('views.avatars.RoomAvatar');
|
||||||
|
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
|
||||||
|
if (!this.room.currentState.mayTriggerNotifOfType('room', client.credentials.userId)) return [];
|
||||||
|
|
||||||
|
const {command, range} = this.getCurrentCommand(query, selection, force);
|
||||||
|
if (command && command[0] && '@room'.startsWith(command[0]) && command[0].length > 1) {
|
||||||
|
return [{
|
||||||
|
completion: '@room',
|
||||||
|
suffix: ' ',
|
||||||
|
component: (
|
||||||
|
<PillCompletion initialComponent={<RoomAvatar width={24} height={24} room={this.room} />} title="@room" description={_t("Notify the whole room")} />
|
||||||
|
),
|
||||||
|
range,
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
getName() {
|
||||||
|
return '❗️ ' + _t('Room Notification');
|
||||||
|
}
|
||||||
|
|
||||||
|
renderCompletions(completions: [React.Component]): ?React.Component {
|
||||||
|
return <div className="mx_Autocomplete_Completion_container_pill mx_Autocomplete_Completion_container_truncate">
|
||||||
|
{ completions }
|
||||||
|
</div>;
|
||||||
|
}
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue