Compare commits
405 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
56b5f2845d | ||
|
|
afee9eaa26 | ||
|
|
d53ad9a8f3 | ||
|
|
d8634eed3d | ||
|
|
be4b70c1e1 | ||
|
|
e79cded57f | ||
|
|
486430d1f0 | ||
|
|
e6ddf40b1b | ||
|
|
9f521a79f7 | ||
|
|
83784a717a | ||
|
|
0729deee79 | ||
|
|
77c3114cf8 | ||
|
|
82a56c8204 | ||
|
|
b39896d8c6 | ||
|
|
79b3fdb645 | ||
|
|
0f877cd021 | ||
|
|
db2acc75b2 | ||
|
|
a5dbfbf2c1 | ||
|
|
34c7d02de2 | ||
|
|
ca45067158 | ||
|
|
5a6eb7c573 | ||
|
|
5ad2a27a92 | ||
|
|
68daaa45f9 | ||
|
|
c40ea35937 | ||
|
|
d27f433175 | ||
|
|
8a6101cd14 | ||
|
|
4db7c2bc68 | ||
|
|
18740fc686 | ||
|
|
0c39398493 | ||
|
|
949145f04b | ||
|
|
8578dcadf2 | ||
|
|
959db44eca | ||
|
|
a031c0e128 | ||
|
|
591833505f | ||
|
|
f7ad5074d8 | ||
|
|
e0aef74bf5 | ||
|
|
b2378bf899 | ||
|
|
255f6b1814 | ||
|
|
4c491b5363 | ||
|
|
61c808d4cf | ||
|
|
13ef3183e2 | ||
|
|
afd4fdcea2 | ||
|
|
982181ccd4 | ||
|
|
30629ebba2 | ||
|
|
7f6a32d21a | ||
|
|
320ade0a50 | ||
|
|
8c6fee3150 | ||
|
|
5c6acaf915 | ||
|
|
c46549b2b6 | ||
|
|
97a58f6db7 | ||
|
|
b6288579c9 | ||
|
|
44bf987cdc | ||
|
|
a7d55824bb | ||
|
|
8fa038c61f | ||
|
|
869d9b43cb | ||
|
|
974a2fe49b | ||
|
|
022497c8e5 | ||
|
|
207a5f047c | ||
|
|
d3c63f9314 | ||
|
|
f9ef037cea | ||
|
|
8666ffec81 | ||
|
|
4b85879891 | ||
|
|
f376291f50 | ||
|
|
f0f56bf101 | ||
|
|
ba647780e8 | ||
|
|
812ae2ce89 | ||
|
|
09ca3b4dc0 | ||
|
|
86afde8612 | ||
|
|
469f0b5983 | ||
|
|
69d1beaf28 | ||
|
|
7f22f442b1 | ||
|
|
ab0f8fa2e3 | ||
|
|
f5abbb1e5e | ||
|
|
a80c96d187 | ||
|
|
fb0c8fb92b | ||
|
|
438a6c2a42 | ||
|
|
0c8cd0842a | ||
|
|
b39d35d5d1 | ||
|
|
6f00a961c9 | ||
|
|
0ed7853958 | ||
|
|
4acf279f32 | ||
|
|
a7c065f300 | ||
|
|
a0d248065d | ||
|
|
c18008e039 | ||
|
|
0bab898c25 | ||
|
|
53d1f717c9 | ||
|
|
0a69664186 | ||
|
|
d1307e61b9 | ||
|
|
a267eca78d | ||
|
|
51a8d2b718 | ||
|
|
2d76a3780d | ||
|
|
e70818b2da | ||
|
|
03144783ac | ||
|
|
b9ab2fdf1b | ||
|
|
183a1845bb | ||
|
|
e423fc0ace | ||
|
|
40cbe5408e | ||
|
|
2c3e95f401 | ||
|
|
1fdff9bbd4 | ||
|
|
bd7209cd40 | ||
|
|
70fdc68b13 | ||
|
|
f9bed2c2a9 | ||
|
|
33437d9743 | ||
|
|
1f4139ae0a | ||
|
|
b8ad8baf9d | ||
|
|
694048dd7f | ||
|
|
a5088d4ae9 | ||
|
|
d2bdaf7049 | ||
|
|
0aeb68b445 | ||
|
|
f40740edd3 | ||
|
|
cd47b63d29 | ||
|
|
07ce272e9f | ||
|
|
a3c4e3e2a5 | ||
|
|
6b6ad16306 | ||
|
|
0f38445fdd | ||
|
|
b9b53ec251 | ||
|
|
b1a5c8c120 | ||
|
|
b44680149d | ||
|
|
cd0cec32b5 | ||
|
|
2faa9c9d50 | ||
|
|
caab45ee7f | ||
|
|
d3687298e0 | ||
|
|
42612476b8 | ||
|
|
7021ea6a5c | ||
|
|
ff09631546 | ||
|
|
28a3dfef23 | ||
|
|
2b5561a88c | ||
|
|
9dc740c2de | ||
|
|
0b93374e86 | ||
|
|
d36af0cae6 | ||
|
|
7fae106da2 | ||
|
|
a84b692f20 | ||
|
|
86a5c24750 | ||
|
|
ed6f02ac56 | ||
|
|
f701886aa9 | ||
|
|
d298e3438c | ||
|
|
c405b61c66 | ||
|
|
6be67aa145 | ||
|
|
603c658949 | ||
|
|
ebc33e003d | ||
|
|
023ab9fc47 | ||
|
|
417faf795d | ||
|
|
c8fe393fcf | ||
|
|
e6a9555a91 | ||
|
|
fd65baed58 | ||
|
|
d079bee5e0 | ||
|
|
1918478069 | ||
|
|
c6d8d5e137 | ||
|
|
7a4583dcb0 | ||
|
|
929175d826 | ||
|
|
49ce642c2d | ||
|
|
ce14c0f6fe | ||
|
|
1a10b67248 | ||
|
|
eda5f14a19 | ||
|
|
e6683569f8 | ||
|
|
90273c1924 | ||
|
|
10f49d0d84 | ||
|
|
418ee89e0f | ||
|
|
a4a57e5307 | ||
|
|
477eb0034a | ||
|
|
5ec57f04c6 | ||
|
|
69cb17adc8 | ||
|
|
8718a7139b | ||
|
|
94f267b93e | ||
|
|
2151696374 | ||
|
|
2679948dbe | ||
|
|
86e84028e1 | ||
|
|
dea6f05b51 | ||
|
|
ac450443a0 | ||
|
|
2a0375d93f | ||
|
|
56312b2753 | ||
|
|
137867b096 | ||
|
|
0ae62b121e | ||
|
|
70682a7490 | ||
|
|
7fc0f96ca6 | ||
|
|
f50b00e00f | ||
|
|
e9c98a02f0 | ||
|
|
dcb4d10afb | ||
|
|
705f9daf5f | ||
|
|
c5e60744a2 | ||
|
|
3670c36fac | ||
|
|
b5d25f1f2d | ||
|
|
51926cad3d | ||
|
|
2ab909fab1 | ||
|
|
cfe0b4d8e4 | ||
|
|
83795ae4bf | ||
|
|
8a4d4d3144 | ||
|
|
0e7fd791b5 | ||
|
|
96e6ca0c0e | ||
|
|
2e0ad5ca69 | ||
|
|
43f7f9b76c | ||
|
|
224cd53481 | ||
|
|
a5f7921a32 | ||
|
|
b8c8e36449 | ||
|
|
f07a491bd9 | ||
|
|
7fb0eb150d | ||
|
|
6e0a20a213 | ||
|
|
af99d2a60b | ||
|
|
b32d066a76 | ||
|
|
a0f9c55194 | ||
|
|
bb9cbe26b3 | ||
|
|
5284479ece | ||
|
|
be25d77e8b | ||
|
|
46732cf86b | ||
|
|
42d697068d | ||
|
|
96a1c4bf7a | ||
|
|
85c140bc32 | ||
|
|
affac2da40 | ||
|
|
bfb26ca500 | ||
|
|
257211e8f9 | ||
|
|
d352fefcaa | ||
|
|
8234211f03 | ||
|
|
6d0de07f07 | ||
|
|
1ed1b7c60c | ||
|
|
e8810882ef | ||
|
|
244387dc04 | ||
|
|
bfd3a61aef | ||
|
|
c5793b9e7c | ||
|
|
4283716dcc | ||
|
|
4288037f51 | ||
|
|
c52d5dc573 | ||
|
|
379050ff80 | ||
|
|
df4da4e4ec | ||
|
|
99d6ced566 | ||
|
|
46cfa65bc2 | ||
|
|
067c13ebf7 | ||
|
|
9f4a107865 | ||
|
|
5e2b652690 | ||
|
|
fc4ced7bb3 | ||
|
|
1381640cdb | ||
|
|
d56dcaf0eb | ||
|
|
1d43bd26ed | ||
|
|
51cc406af7 | ||
|
|
03d2818c1f | ||
|
|
e1c71327c5 | ||
|
|
1baed1184a | ||
|
|
c6fc0bb798 | ||
|
|
d5ccfa2ac5 | ||
|
|
7793805f1b | ||
|
|
c67040afe7 | ||
|
|
444297f3d9 | ||
|
|
afb43d049d | ||
|
|
93c2ae768e | ||
|
|
2bbd882425 | ||
|
|
e781dfa164 | ||
|
|
fc65c799d8 | ||
|
|
d9d218cb1f | ||
|
|
9946888d63 | ||
|
|
202388bd79 | ||
|
|
1df5ee05c9 | ||
|
|
ee2214c14d | ||
|
|
a2066f300b | ||
|
|
a0938bad13 | ||
|
|
9675df8434 | ||
|
|
b9a6ee70a8 | ||
|
|
5c399fbfb2 | ||
|
|
a5231983e8 | ||
|
|
76d3117a1c | ||
|
|
6bcfe61a79 | ||
|
|
9fe7f62a63 | ||
|
|
7455104aaf | ||
|
|
96bf809e8a | ||
|
|
b5a7c55d70 | ||
|
|
c578bcaf91 | ||
|
|
908ca2325a | ||
|
|
a1659c1fe2 | ||
|
|
3d7ef061cd | ||
|
|
a04500f102 | ||
|
|
b8aaa6b079 | ||
|
|
89911c1747 | ||
|
|
f7e63bd2f5 | ||
|
|
76bb72d3ea | ||
|
|
fd8ee1542b | ||
|
|
ef395efaf8 | ||
|
|
8da2f5c1c7 | ||
|
|
0bfe12bcbc | ||
|
|
63023f542b | ||
|
|
74ddcee2af | ||
|
|
9cce4bc0ad | ||
|
|
be44aa0157 | ||
|
|
e74095be54 | ||
|
|
3458ff9716 | ||
|
|
dcc04bb10f | ||
|
|
36d754f27f | ||
|
|
1daca7a0d9 | ||
|
|
884491d60e | ||
|
|
9c00d74924 | ||
|
|
f32934d7e4 | ||
|
|
4b1505abb7 | ||
|
|
21b4d46f83 | ||
|
|
cb113cbfe4 | ||
|
|
fd6032b44f | ||
|
|
229e5f3ce7 | ||
|
|
4271685c66 | ||
|
|
c59610a5c1 | ||
|
|
94ef74b8bf | ||
|
|
ea014e1b0c | ||
|
|
922bb00722 | ||
|
|
c412e2550d | ||
|
|
5bad53c63e | ||
|
|
4cc077a72a | ||
|
|
708cb18332 | ||
|
|
3c3150a6b6 | ||
|
|
fc525a6e45 | ||
|
|
545034d26f | ||
|
|
5f7410c809 | ||
|
|
db24e544bd | ||
|
|
71e3fefc00 | ||
|
|
1aaa161522 | ||
|
|
0b2c78d563 | ||
|
|
876c904eb0 | ||
|
|
ffd0832b31 | ||
|
|
907e7bdf52 | ||
|
|
0d997254d0 | ||
|
|
e40efd57c8 | ||
|
|
096ddce173 | ||
|
|
cc279d849f | ||
|
|
dd81ff68dd | ||
|
|
750e35015e | ||
|
|
97693639dd | ||
|
|
a88458aebf | ||
|
|
beb408aa04 | ||
|
|
a15ded502f | ||
|
|
5963bdae6e | ||
|
|
6c5e73513c | ||
|
|
d35a070844 | ||
|
|
c2ff3dc8b7 | ||
|
|
63e3a98c02 | ||
|
|
a7053ac595 | ||
|
|
75d529e66c | ||
|
|
fe7bb76f01 | ||
|
|
d78b894a6b | ||
|
|
5121b956ca | ||
|
|
255b3218a5 | ||
|
|
1614b2e8bd | ||
|
|
3f5c7257b3 | ||
|
|
ccea9fc961 | ||
|
|
5ebaad014c | ||
|
|
aa14051755 | ||
|
|
e9e79f8ec3 | ||
|
|
88471f1462 | ||
|
|
9b8088fb43 | ||
|
|
8e950e7c0a | ||
|
|
8de238dfc6 | ||
|
|
2f37787ea3 | ||
|
|
7cf50101b3 | ||
|
|
518f8c8783 | ||
|
|
f6094378fe | ||
|
|
44a8dffd08 | ||
|
|
9d2711a736 | ||
|
|
1a7048f2b8 | ||
|
|
16bc438ee3 | ||
|
|
5d670d4bfb | ||
|
|
ca761b9176 | ||
|
|
d5de8cac2d | ||
|
|
1c47a83c04 | ||
|
|
318702ad13 | ||
|
|
bf8a2bba45 | ||
|
|
592386daae | ||
|
|
e2bd381f65 | ||
|
|
28fcdaf6cb | ||
|
|
88a67b8fc8 | ||
|
|
09eb666daf | ||
|
|
c8ebd129ed | ||
|
|
1aede1e24e | ||
|
|
c932dd8c9b | ||
|
|
4a9a58475f | ||
|
|
b503260caa | ||
|
|
7883db3434 | ||
|
|
3310484859 | ||
|
|
e6aa60edd0 | ||
|
|
5a3d625333 | ||
|
|
163586a80e | ||
|
|
6d5887b33e | ||
|
|
8f2371757e | ||
|
|
478745a38d | ||
|
|
d79359ae66 | ||
|
|
f2b5ba24de | ||
|
|
33ec5b1451 | ||
|
|
0c87fd3b75 | ||
|
|
b1586e07a3 | ||
|
|
b254b38427 | ||
|
|
c3ed8bf139 | ||
|
|
661e11a2a4 | ||
|
|
134a356ba2 | ||
|
|
be3b1c40d0 | ||
|
|
c09d1ecbff | ||
|
|
186be88c24 | ||
|
|
f8fa8e3722 | ||
|
|
58b97d8c04 | ||
|
|
419bf22f2a | ||
|
|
322bbe38f8 | ||
|
|
5bd1832770 | ||
|
|
13fa572f26 | ||
|
|
bddf6fb856 | ||
|
|
ecc7f01933 | ||
|
|
48aef98440 | ||
|
|
46a12c7476 | ||
|
|
f7681a60ab | ||
|
|
19f56f27a7 | ||
|
|
1d128bcfe0 | ||
|
|
57548b5d43 | ||
|
|
9b83cebcad | ||
|
|
d543c1a9c0 | ||
|
|
6cfc2066c9 |
49
.github/workflows/build.yaml
vendored
49
.github/workflows/build.yaml
vendored
@@ -1,34 +1,25 @@
|
|||||||
name: Build
|
name: Build
|
||||||
on:
|
on:
|
||||||
pull_request: {}
|
pull_request:
|
||||||
|
types:
|
||||||
|
- synchronize
|
||||||
|
- opened
|
||||||
|
- labeled
|
||||||
|
paths-ignore:
|
||||||
|
- ".github/**"
|
||||||
|
- "docs/**"
|
||||||
push:
|
push:
|
||||||
branches: [livekit, full-mesh]
|
branches: [livekit, full-mesh]
|
||||||
|
paths-ignore:
|
||||||
|
- ".github/**"
|
||||||
|
- "docs/**"
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build_element_call:
|
||||||
name: Build
|
uses: ./.github/workflows/element-call.yaml
|
||||||
runs-on: ubuntu-latest
|
with:
|
||||||
steps:
|
vite_app_version: ${{ github.event.release.tag_name || github.sha }}
|
||||||
- name: Checkout code
|
secrets:
|
||||||
uses: actions/checkout@v4
|
SENTRY_ORG: ${{ secrets.SENTRY_ORG }}
|
||||||
- name: Yarn cache
|
SENTRY_PROJECT: ${{ secrets.SENTRY_PROJECT }}
|
||||||
uses: actions/setup-node@v4
|
SENTRY_URL: ${{ secrets.SENTRY_URL }}
|
||||||
with:
|
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
||||||
cache: "yarn"
|
|
||||||
- name: Install dependencies
|
|
||||||
run: "yarn install"
|
|
||||||
- name: Build
|
|
||||||
run: "yarn run build"
|
|
||||||
env:
|
|
||||||
SENTRY_ORG: ${{ secrets.SENTRY_ORG }}
|
|
||||||
SENTRY_PROJECT: ${{ secrets.SENTRY_PROJECT }}
|
|
||||||
SENTRY_URL: ${{ secrets.SENTRY_URL }}
|
|
||||||
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
|
||||||
VITE_APP_VERSION: ${{ github.sha }}
|
|
||||||
NODE_OPTIONS: "--max-old-space-size=4096"
|
|
||||||
- name: Upload Artifact
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: build
|
|
||||||
path: dist
|
|
||||||
# We'll only use this in a triggered job, then we're done with it
|
|
||||||
retention-days: 1
|
|
||||||
|
|||||||
60
.github/workflows/docker.yaml
vendored
Normal file
60
.github/workflows/docker.yaml
vendored
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
name: Docker - Deploy
|
||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
inputs:
|
||||||
|
docker_tags:
|
||||||
|
required: true
|
||||||
|
type: string
|
||||||
|
artifact_run_id:
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
default: ${{ github.run_id }}
|
||||||
|
|
||||||
|
env:
|
||||||
|
REGISTRY: ghcr.io
|
||||||
|
IMAGE_NAME: ${{ github.repository }}
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build_and_deploy:
|
||||||
|
name: Build & publish docker
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: write # required to upload release asset
|
||||||
|
packages: write
|
||||||
|
steps:
|
||||||
|
- name: Check it out
|
||||||
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
|
|
||||||
|
- name: 📥 Download artifact
|
||||||
|
uses: actions/download-artifact@65a9edc5881444af0b9093a5e628f2fe47ea3b2e # v4
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
run-id: ${{ inputs.artifact_run_id }}
|
||||||
|
name: build-output
|
||||||
|
path: dist
|
||||||
|
|
||||||
|
- name: Log in to container registry
|
||||||
|
uses: docker/login-action@0d4c9c5ea7693da7b068278f7b52bda2a190a446 # v3.2.0
|
||||||
|
with:
|
||||||
|
registry: ${{ env.REGISTRY }}
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Extract metadata (tags, labels) for Docker
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@8e5442c4ef9f78752691e2d8f8d19755c6f78e81 # v5.5.1
|
||||||
|
with:
|
||||||
|
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
||||||
|
tags: ${{ inputs.docker_tags}}
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@d70bba72b1f3fd22344832f00baa16ece964efeb # v3.3.0
|
||||||
|
|
||||||
|
- name: Build and push Docker image
|
||||||
|
uses: docker/build-push-action@1a162644f9a7e87d8f4b053101d1d9a712edc18c # v6.3.0
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
push: true
|
||||||
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
2
.github/workflows/e2e.yml
vendored
2
.github/workflows/e2e.yml
vendored
@@ -12,7 +12,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out test private repo
|
- name: Check out test private repo
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
with:
|
with:
|
||||||
repository: element-hq/static-call-participant
|
repository: element-hq/static-call-participant
|
||||||
ref: refs/heads/main
|
ref: refs/heads/main
|
||||||
|
|||||||
46
.github/workflows/element-call.yaml
vendored
Normal file
46
.github/workflows/element-call.yaml
vendored
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
name: Element Call - Build
|
||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
inputs:
|
||||||
|
vite_app_version:
|
||||||
|
required: true
|
||||||
|
type: string
|
||||||
|
secrets:
|
||||||
|
SENTRY_ORG:
|
||||||
|
required: true
|
||||||
|
SENTRY_PROJECT:
|
||||||
|
required: true
|
||||||
|
SENTRY_URL:
|
||||||
|
required: true
|
||||||
|
SENTRY_AUTH_TOKEN:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
name: Build Element Call
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
|
- name: Yarn cache
|
||||||
|
uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4
|
||||||
|
with:
|
||||||
|
cache: "yarn"
|
||||||
|
- name: Install dependencies
|
||||||
|
run: "yarn install"
|
||||||
|
- name: Build
|
||||||
|
run: "yarn run build"
|
||||||
|
env:
|
||||||
|
SENTRY_ORG: ${{ secrets.SENTRY_ORG }}
|
||||||
|
SENTRY_PROJECT: ${{ secrets.SENTRY_PROJECT }}
|
||||||
|
SENTRY_URL: ${{ secrets.SENTRY_URL }}
|
||||||
|
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
||||||
|
VITE_APP_VERSION: ${{ inputs.vite_app_version }}
|
||||||
|
NODE_OPTIONS: "--max-old-space-size=4096"
|
||||||
|
- name: Upload Artifact
|
||||||
|
uses: actions/upload-artifact@65462800fd760344b1a7b4382951275a0abb4808 # v4
|
||||||
|
with:
|
||||||
|
name: build-output
|
||||||
|
path: dist
|
||||||
|
# We'll only use this in a triggered job, then we're done with it
|
||||||
|
retention-days: 1
|
||||||
4
.github/workflows/lint.yaml
vendored
4
.github/workflows/lint.yaml
vendored
@@ -7,9 +7,9 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
- name: Yarn cache
|
- name: Yarn cache
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4
|
||||||
with:
|
with:
|
||||||
cache: "yarn"
|
cache: "yarn"
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
|
|||||||
@@ -1,45 +1,56 @@
|
|||||||
name: Netlify PR Preview
|
name: Netlify - Deploy
|
||||||
on:
|
on:
|
||||||
workflow_run:
|
workflow_call:
|
||||||
workflows: ["Build"]
|
inputs:
|
||||||
types:
|
pr_number:
|
||||||
- completed
|
required: true
|
||||||
branches-ignore:
|
type: string
|
||||||
- "main"
|
pr_head_full_name:
|
||||||
- "livekit"
|
required: true
|
||||||
|
type: string
|
||||||
|
pr_head_ref:
|
||||||
|
required: true
|
||||||
|
type: string
|
||||||
|
deployment_ref:
|
||||||
|
required: true
|
||||||
|
type: string
|
||||||
|
artifact_run_id:
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
default: ${{ github.run_id }}
|
||||||
|
secrets:
|
||||||
|
ELEMENT_BOT_TOKEN:
|
||||||
|
required: true
|
||||||
|
NETLIFY_AUTH_TOKEN:
|
||||||
|
required: true
|
||||||
|
NETLIFY_SITE_ID:
|
||||||
|
required: true
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
deploy:
|
deploy:
|
||||||
if: github.event.workflow_run.conclusion == 'success' && github.event.workflow_run.event == 'pull_request'
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
permissions:
|
||||||
deployments: write
|
deployments: write
|
||||||
environment: Netlify
|
environment: Netlify
|
||||||
steps:
|
steps:
|
||||||
- name: 📝 Create Deployment
|
- name: 📝 Create Deployment
|
||||||
uses: bobheadxi/deployments@v1
|
uses: bobheadxi/deployments@648679e8e4915b27893bd7dbc35cb504dc915bc8 # v1
|
||||||
id: deployment
|
id: deployment
|
||||||
with:
|
with:
|
||||||
step: start
|
step: start
|
||||||
token: ${{ secrets.GITHUB_TOKEN }}
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
env: Netlify
|
env: Netlify
|
||||||
ref: ${{ github.event.workflow_run.head_sha }}
|
ref: ${{ inputs.deployment_ref }}
|
||||||
desc: |
|
desc: |
|
||||||
Do you trust the author of this PR? Maybe this build will steal your keys or give you malware.
|
Do you trust the author of this PR? Maybe this build will steal your keys or give you malware.
|
||||||
Exercise caution. Use test accounts.
|
Exercise caution. Use test accounts.
|
||||||
|
|
||||||
- id: prdetails
|
|
||||||
uses: matrix-org/pr-details-action@v1.3
|
|
||||||
with:
|
|
||||||
owner: ${{ github.event.workflow_run.head_repository.owner.login }}
|
|
||||||
branch: ${{ github.event.workflow_run.head_branch }}
|
|
||||||
|
|
||||||
# There's a 'download artifact' action, but it hasn't been updated for the workflow_run action
|
|
||||||
# (https://github.com/actions/download-artifact/issues/60) so instead we get this mess:
|
|
||||||
- name: 📥 Download artifact
|
- name: 📥 Download artifact
|
||||||
uses: dawidd6/action-download-artifact@v3
|
uses: actions/download-artifact@65a9edc5881444af0b9093a5e628f2fe47ea3b2e # v4
|
||||||
with:
|
with:
|
||||||
run_id: ${{ github.event.workflow_run.id }}
|
github-token: ${{ secrets.ELEMENT_BOT_TOKEN }}
|
||||||
name: build
|
run-id: ${{ inputs.artifact_run_id }}
|
||||||
|
name: build-output
|
||||||
path: webapp
|
path: webapp
|
||||||
|
|
||||||
- name: Add redirects file
|
- name: Add redirects file
|
||||||
@@ -47,25 +58,22 @@ jobs:
|
|||||||
run: curl -s https://raw.githubusercontent.com/element-hq/element-call/main/config/netlify_redirects > webapp/_redirects
|
run: curl -s https://raw.githubusercontent.com/element-hq/element-call/main/config/netlify_redirects > webapp/_redirects
|
||||||
|
|
||||||
- name: Add config file
|
- name: Add config file
|
||||||
run: curl -s "https://raw.githubusercontent.com/${{ github.event.workflow_run.head_repository.full_name }}/${{ github.event.workflow_run.head_branch }}/config/element_io_preview.json" > webapp/config.json
|
run: curl -s "https://raw.githubusercontent.com/${{ inputs.pr_head_full_name }}/${{ inputs.pr_head_ref }}/config/element_io_preview.json" > webapp/config.json
|
||||||
|
|
||||||
- name: ☁️ Deploy to Netlify
|
- name: ☁️ Deploy to Netlify
|
||||||
id: netlify
|
id: netlify
|
||||||
uses: nwtgck/actions-netlify@v2.1
|
uses: nwtgck/actions-netlify@4cbaf4c08f1a7bfa537d6113472ef4424e4eb654 # v3.0
|
||||||
with:
|
with:
|
||||||
publish-dir: webapp
|
publish-dir: webapp
|
||||||
deploy-message: "Deploy from GitHub Actions"
|
deploy-message: "Deploy from GitHub Actions"
|
||||||
# These don't work because we're in workflow_run
|
alias: pr${{ inputs.pr_number }}
|
||||||
enable-pull-request-comment: false
|
|
||||||
enable-commit-comment: false
|
|
||||||
alias: pr${{ steps.prdetails.outputs.pr_id }}
|
|
||||||
env:
|
env:
|
||||||
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
||||||
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}
|
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}
|
||||||
timeout-minutes: 1
|
timeout-minutes: 1
|
||||||
|
|
||||||
- name: 🚦 Update deployment status
|
- name: 🚦 Update deployment status
|
||||||
uses: bobheadxi/deployments@v1
|
uses: bobheadxi/deployments@648679e8e4915b27893bd7dbc35cb504dc915bc8 # v1
|
||||||
if: always()
|
if: always()
|
||||||
with:
|
with:
|
||||||
step: finish
|
step: finish
|
||||||
50
.github/workflows/pr-deploy.yaml
vendored
Normal file
50
.github/workflows/pr-deploy.yaml
vendored
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
name: PR Preview Deployments
|
||||||
|
on:
|
||||||
|
workflow_run:
|
||||||
|
workflows: ["Build"]
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
prdetails:
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'success' && github.event.workflow_run.event == 'pull_request' }}
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
pr_number: ${{ steps.prdetails.outputs.pr_id }}
|
||||||
|
pr_data_json: ${{ steps.prdetails.outputs.data }}
|
||||||
|
steps:
|
||||||
|
- id: prdetails
|
||||||
|
uses: matrix-org/pr-details-action@15bde5285d7850ba276cc3bd8a03733e3f24622a # v1.3
|
||||||
|
continue-on-error: true
|
||||||
|
with:
|
||||||
|
owner: ${{ github.event.workflow_run.head_repository.owner.login }}
|
||||||
|
branch: ${{ github.event.workflow_run.head_branch }}
|
||||||
|
|
||||||
|
netlify:
|
||||||
|
needs: prdetails
|
||||||
|
permissions:
|
||||||
|
deployments: write
|
||||||
|
uses: ./.github/workflows/netlify.yaml
|
||||||
|
with:
|
||||||
|
artifact_run_id: ${{ github.event.workflow_run.id || github.run_id }}
|
||||||
|
pr_number: ${{ needs.prdetails.outputs.pr_number }}
|
||||||
|
pr_head_full_name: ${{ github.event.workflow_run.head_repository.full_name }}
|
||||||
|
pr_head_ref: ${{ needs.prdetails.outputs.pr_data_json && fromJSON(needs.prdetails.outputs.pr_data_json).head.ref }}
|
||||||
|
deployment_ref: ${{ needs.prdetails.outputs.pr_data_json && fromJSON(needs.prdetails.outputs.pr_data_json).head.sha || github.ref || github.head_ref }}
|
||||||
|
secrets:
|
||||||
|
ELEMENT_BOT_TOKEN: ${{ secrets.ELEMENT_BOT_TOKEN }}
|
||||||
|
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
||||||
|
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}
|
||||||
|
|
||||||
|
docker:
|
||||||
|
if: ${{ needs.prdetails.outputs.pr_data_json && contains(fromJSON(needs.prdetails.outputs.pr_data_json).labels.*.name, 'docker build') }}
|
||||||
|
needs: prdetails
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
packages: write
|
||||||
|
uses: ./.github/workflows/docker.yaml
|
||||||
|
with:
|
||||||
|
artifact_run_id: ${{ github.event.workflow_run.id || github.run_id }}
|
||||||
|
docker_tags: |
|
||||||
|
type=sha,format=short,event=branch
|
||||||
|
type=raw,value=pr_${{ needs.prdetails.outputs.pr_number }}
|
||||||
94
.github/workflows/publish.yaml
vendored
94
.github/workflows/publish.yaml
vendored
@@ -3,17 +3,34 @@ name: Build & publish images to the package registry for tags
|
|||||||
on:
|
on:
|
||||||
release:
|
release:
|
||||||
types: [published]
|
types: [published]
|
||||||
push:
|
workflow_run:
|
||||||
|
workflows: ["Build"]
|
||||||
branches: [livekit]
|
branches: [livekit]
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
|
||||||
env:
|
env:
|
||||||
REGISTRY: ghcr.io
|
REGISTRY: ghcr.io
|
||||||
IMAGE_NAME: ${{ github.repository }}
|
IMAGE_NAME: ${{ github.repository }}
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build_element_call:
|
||||||
name: Build & publish
|
if: ${{ github.event_name == 'release' }}
|
||||||
|
uses: ./.github/workflows/element-call.yaml
|
||||||
|
with:
|
||||||
|
vite_app_version: ${{ github.event.release.tag_name || github.sha }}
|
||||||
|
secrets:
|
||||||
|
SENTRY_ORG: ${{ secrets.SENTRY_ORG }}
|
||||||
|
SENTRY_PROJECT: ${{ secrets.SENTRY_PROJECT }}
|
||||||
|
SENTRY_URL: ${{ secrets.SENTRY_URL }}
|
||||||
|
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
||||||
|
publish_tarball:
|
||||||
|
needs: build_element_call
|
||||||
|
if: always()
|
||||||
|
name: Publish tarball
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
unix_time: ${{steps.current-time.outputs.unix_time}}
|
||||||
permissions:
|
permissions:
|
||||||
contents: write # required to upload release asset
|
contents: write # required to upload release asset
|
||||||
packages: write
|
packages: write
|
||||||
@@ -21,64 +38,35 @@ jobs:
|
|||||||
- name: Get current time
|
- name: Get current time
|
||||||
id: current-time
|
id: current-time
|
||||||
run: echo "unix_time=$(date +'%s')" >> $GITHUB_OUTPUT
|
run: echo "unix_time=$(date +'%s')" >> $GITHUB_OUTPUT
|
||||||
|
- name: 📥 Download artifact
|
||||||
- name: Check it out
|
uses: actions/download-artifact@65a9edc5881444af0b9093a5e628f2fe47ea3b2e # v4
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Log in to container registry
|
|
||||||
uses: docker/login-action@5139682d94efc37792e6b54386b5b470a68a4737
|
|
||||||
with:
|
with:
|
||||||
registry: ${{ env.REGISTRY }}
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
username: ${{ github.actor }}
|
run-id: ${{ github.event.workflow_run.id || github.run_id }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
name: build-output
|
||||||
|
path: dist
|
||||||
- name: Yarn cache
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
cache: "yarn"
|
|
||||||
- name: Install dependencies
|
|
||||||
run: "yarn install"
|
|
||||||
- name: Build
|
|
||||||
run: "yarn run build"
|
|
||||||
env:
|
|
||||||
SENTRY_ORG: ${{ secrets.SENTRY_ORG }}
|
|
||||||
SENTRY_PROJECT: ${{ secrets.SENTRY_PROJECT }}
|
|
||||||
SENTRY_URL: ${{ secrets.SENTRY_URL }}
|
|
||||||
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
|
||||||
VITE_APP_VERSION: ${{ github.event.release.tag_name || github.sha }}
|
|
||||||
|
|
||||||
- name: Create Tarball
|
- name: Create Tarball
|
||||||
env:
|
env:
|
||||||
TARBALL_VERSION: ${{ github.event.release.tag_name || github.sha }}
|
TARBALL_VERSION: ${{ github.event.release.tag_name || github.sha }}
|
||||||
run: |
|
run: |
|
||||||
tar --numeric-owner --transform "s/dist/element-call-${TARBALL_VERSION}/" -cvzf element-call-${TARBALL_VERSION}.tar.gz dist
|
tar --numeric-owner --transform "s/dist/element-call-${TARBALL_VERSION}/" -cvzf element-call-${TARBALL_VERSION}.tar.gz dist
|
||||||
|
|
||||||
- name: Upload
|
- name: Upload
|
||||||
uses: actions/upload-artifact@ef09cdac3e2d3e60d8ccadda691f4f1cec5035cb
|
uses: actions/upload-artifact@65462800fd760344b1a7b4382951275a0abb4808 # v4.3.3
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ github.token }}
|
GITHUB_TOKEN: ${{ github.token }}
|
||||||
with:
|
with:
|
||||||
path: "./element-call-*.tar.gz"
|
path: "./element-call-*.tar.gz"
|
||||||
|
publish_docker:
|
||||||
- name: Extract metadata (tags, labels) for Docker
|
needs: publish_tarball
|
||||||
id: meta
|
if: always()
|
||||||
uses: docker/metadata-action@232fc64e3a4e54539e087c5976439ea54be0959d
|
permissions:
|
||||||
with:
|
contents: write
|
||||||
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
packages: write
|
||||||
tags: |
|
uses: ./.github/workflows/docker.yaml
|
||||||
type=sha,format=short,event=branch
|
with:
|
||||||
type=semver,pattern=v{{version}}
|
artifact_run_id: ${{ github.event.workflow_run.id || github.run_id }}
|
||||||
type=raw,value=latest-ci,enable={{is_default_branch}}
|
docker_tags: |
|
||||||
type=raw,value=latest-ci_${{steps.current-time.outputs.unix_time}},enable={{is_default_branch}}
|
type=sha,format=short,event=branch
|
||||||
|
type=semver,pattern=v{{version}}
|
||||||
- name: Set up Docker Buildx
|
type=raw,value=latest-ci,enable={{is_default_branch}}
|
||||||
uses: docker/setup-buildx-action@0d103c3126aa41d772a8362f6aa67afac040f80c
|
type=raw,value=latest-ci_${{needs.publish_tarball.outputs.unix_time}},enable={{is_default_branch}}
|
||||||
|
|
||||||
- name: Build and push Docker image
|
|
||||||
uses: docker/build-push-action@b3eddbb94c4146a0988a620b01720afe50639271
|
|
||||||
with:
|
|
||||||
context: .
|
|
||||||
platforms: linux/amd64,linux/arm64
|
|
||||||
push: true
|
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
|
||||||
|
|||||||
6
.github/workflows/test.yaml
vendored
6
.github/workflows/test.yaml
vendored
@@ -9,9 +9,9 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
- name: Yarn cache
|
- name: Yarn cache
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4
|
||||||
with:
|
with:
|
||||||
cache: "yarn"
|
cache: "yarn"
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
@@ -19,6 +19,6 @@ jobs:
|
|||||||
- name: Vitest
|
- name: Vitest
|
||||||
run: "yarn run test"
|
run: "yarn run test"
|
||||||
- name: Upload to codecov
|
- name: Upload to codecov
|
||||||
uses: codecov/codecov-action@v4
|
uses: codecov/codecov-action@e28ff129e5465c2c0dcc6f003fc735cb6ae0c673 # v4
|
||||||
with:
|
with:
|
||||||
flags: unittests
|
flags: unittests
|
||||||
|
|||||||
8
.github/workflows/translations-download.yaml
vendored
8
.github/workflows/translations-download.yaml
vendored
@@ -13,9 +13,9 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the code
|
- name: Checkout the code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4
|
||||||
with:
|
with:
|
||||||
cache: "yarn"
|
cache: "yarn"
|
||||||
|
|
||||||
@@ -26,7 +26,7 @@ jobs:
|
|||||||
run: "rm -R public/locales"
|
run: "rm -R public/locales"
|
||||||
|
|
||||||
- name: Download translation files
|
- name: Download translation files
|
||||||
uses: localazy/download@v1.1.0
|
uses: localazy/download@0a79880fb66150601e3b43606fab69c88123c087 # v1.1.0
|
||||||
with:
|
with:
|
||||||
groups: "-p includeSourceLang:true"
|
groups: "-p includeSourceLang:true"
|
||||||
|
|
||||||
@@ -38,7 +38,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Create Pull Request
|
- name: Create Pull Request
|
||||||
id: cpr
|
id: cpr
|
||||||
uses: peter-evans/create-pull-request@v6.0.0
|
uses: peter-evans/create-pull-request@c5a7806660adbe173f04e3e038b0ccdcd758773c # v6.1.0
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.ELEMENT_BOT_TOKEN }}
|
token: ${{ secrets.ELEMENT_BOT_TOKEN }}
|
||||||
branch: actions/localazy-download
|
branch: actions/localazy-download
|
||||||
|
|||||||
6
.github/workflows/translations-upload.yaml
vendored
6
.github/workflows/translations-upload.yaml
vendored
@@ -3,6 +3,8 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- livekit
|
- livekit
|
||||||
|
paths-ignore:
|
||||||
|
- ".github/**"
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
upload:
|
upload:
|
||||||
@@ -12,9 +14,9 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the code
|
- name: Checkout the code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4
|
||||||
|
|
||||||
- name: Upload
|
- name: Upload
|
||||||
uses: localazy/upload@v1
|
uses: localazy/upload@27e6b5c0fddf4551596b42226b1c24124335d24a # v1
|
||||||
with:
|
with:
|
||||||
write_key: ${{ secrets.LOCALAZY_WRITE_KEY }}
|
write_key: ${{ secrets.LOCALAZY_WRITE_KEY }}
|
||||||
|
|||||||
38
README.md
38
README.md
@@ -54,6 +54,38 @@ Therefore, to use a self-hosted homeserver, this is recommended to be a new serv
|
|||||||
|
|
||||||
There are currently two different config files. `.env` holds variables that are used at build time, while `public/config.json` holds variables that are used at runtime. Documentation and default values for `public/config.json` can be found in [ConfigOptions.ts](src/config/ConfigOptions.ts).
|
There are currently two different config files. `.env` holds variables that are used at build time, while `public/config.json` holds variables that are used at runtime. Documentation and default values for `public/config.json` can be found in [ConfigOptions.ts](src/config/ConfigOptions.ts).
|
||||||
|
|
||||||
|
If you're using [Synapse](https://github.com/element-hq/synapse/), you'll need to additionally add the following to `homeserver.yaml` or Element Call won't work:
|
||||||
|
|
||||||
|
```
|
||||||
|
experimental_features:
|
||||||
|
msc3266_enabled: true
|
||||||
|
```
|
||||||
|
|
||||||
|
MSC3266 allows to request a room summary of rooms you are not joined.
|
||||||
|
The summary contains the room join rules. We need that to decide if the user gets prompted with the option to knock ("ask to join"), a cannot join error or the join view.
|
||||||
|
|
||||||
|
Element Call requires a Livekit SFU behind a Livekit jwt service to work. The url to the Livekit jwt service can either be configured in the config of Element Call (fallback/legacy configuration) or be configured by your homeserver via the `.well-known`.
|
||||||
|
This is the recommended method.
|
||||||
|
|
||||||
|
The configuration is a list of Foci configs:
|
||||||
|
|
||||||
|
```json
|
||||||
|
"org.matrix.msc4143.rtc_foci": [
|
||||||
|
{
|
||||||
|
"type": "livekit",
|
||||||
|
"livekit_service_url": "https://someurl.com"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "livekit",
|
||||||
|
"livekit_service_url": "https://livekit2.com"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "another_foci",
|
||||||
|
"props_for_another_foci": "val"
|
||||||
|
},
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
## Translation
|
## Translation
|
||||||
|
|
||||||
If you'd like to help translate Element Call, head over to [Localazy](https://localazy.com/p/element-call). You're also encouraged to join the [Element Translators](https://matrix.to/#/#translators:element.io) space to discuss and coordinate translation efforts.
|
If you'd like to help translate Element Call, head over to [Localazy](https://localazy.com/p/element-call). You're also encouraged to join the [Element Translators](https://matrix.to/#/#translators:element.io) space to discuss and coordinate translation efforts.
|
||||||
@@ -93,11 +125,13 @@ service for development. These use a test 'secret' published in this
|
|||||||
repository, so this must be used only for local development and
|
repository, so this must be used only for local development and
|
||||||
**_never be exposed to the public Internet._**
|
**_never be exposed to the public Internet._**
|
||||||
|
|
||||||
To use it, add SFU parameter in your local config `./public/config.json`:
|
To use it, add a SFU parameter in your local config `./public/config.json`:
|
||||||
|
(Be aware, that this is only the fallback Livekit SFU. If the homeserver
|
||||||
|
advertises one in the client well-known, this will not be used.)
|
||||||
|
|
||||||
```json
|
```json
|
||||||
"livekit": {
|
"livekit": {
|
||||||
"livekit_service_url": "http://localhost:8881"
|
"livekit_service_url": "http://localhost:7881"
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
port: 7880
|
port: 7880
|
||||||
environment: dev
|
|
||||||
bind_addresses:
|
bind_addresses:
|
||||||
- "0.0.0.0"
|
- "0.0.0.0"
|
||||||
rtc:
|
rtc:
|
||||||
@@ -22,5 +21,3 @@ turn:
|
|||||||
external_tls: true
|
external_tls: true
|
||||||
keys:
|
keys:
|
||||||
devkey: secret
|
devkey: secret
|
||||||
signal_relay:
|
|
||||||
enabled: true
|
|
||||||
|
|||||||
@@ -5,5 +5,8 @@
|
|||||||
"server_name": "call.ems.host"
|
"server_name": "call.ems.host"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"livekit": {
|
||||||
|
"livekit_service_url": "http://localhost:7881"
|
||||||
|
},
|
||||||
"eula": "https://static.element.io/legal/online-EULA.pdf"
|
"eula": "https://static.element.io/legal/online-EULA.pdf"
|
||||||
}
|
}
|
||||||
|
|||||||
30
config/httpd.conf
Normal file
30
config/httpd.conf
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<VirtualHost *:8080>
|
||||||
|
ServerName localhost
|
||||||
|
|
||||||
|
DocumentRoot "/app"
|
||||||
|
|
||||||
|
<Location "/">
|
||||||
|
# disable cache entriely by default (apart from Etag which is accurate enough)
|
||||||
|
Header add Cache-Control "private no-store, no-cache, must-revalidate, proxy-revalidate, max-age=0"
|
||||||
|
CacheDisable on
|
||||||
|
ExpiresActive off
|
||||||
|
|
||||||
|
# also turn off last-modified since they are just the timestamps of the file in the docker image
|
||||||
|
# and may or may not bear any resemblance to when the resource changed
|
||||||
|
Header add Last-Modified ""
|
||||||
|
|
||||||
|
DirectoryIndex index.html
|
||||||
|
</Location>
|
||||||
|
|
||||||
|
# assets can be cached because they have hashed filenames
|
||||||
|
<Location "/assets">
|
||||||
|
ExpiresActive on
|
||||||
|
ExpiresDefault "access plus 1 week"
|
||||||
|
Header add Cache-Control "public, no-transform"
|
||||||
|
</Location>
|
||||||
|
|
||||||
|
<Location "/apple-app-site-association">
|
||||||
|
ForceType application/json
|
||||||
|
</Location>
|
||||||
|
</VirtualHost>
|
||||||
|
|
||||||
33
package.json
33
package.json
@@ -1,4 +1,5 @@
|
|||||||
{
|
{
|
||||||
|
"name": "element-call",
|
||||||
"version": "0.0.0",
|
"version": "0.0.0",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -19,15 +20,15 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@juggle/resize-observer": "^3.3.1",
|
"@juggle/resize-observer": "^3.3.1",
|
||||||
"@livekit/components-core": "^0.9.0",
|
"@livekit/components-core": "^0.10.0",
|
||||||
"@livekit/components-react": "^2.0.0",
|
"@livekit/components-react": "^2.0.0",
|
||||||
"@matrix-org/olm": "https://gitlab.matrix.org/api/v4/projects/27/packages/npm/@matrix-org/olm/-/@matrix-org/olm-3.2.14.tgz",
|
"@matrix-org/olm": "https://gitlab.matrix.org/api/v4/projects/27/packages/npm/@matrix-org/olm/-/@matrix-org/olm-3.2.14.tgz",
|
||||||
"@opentelemetry/api": "^1.4.0",
|
"@opentelemetry/api": "^1.4.0",
|
||||||
"@opentelemetry/context-zone": "^1.9.1",
|
"@opentelemetry/context-zone": "^1.9.1",
|
||||||
"@opentelemetry/exporter-jaeger": "^1.9.1",
|
"@opentelemetry/exporter-jaeger": "^1.9.1",
|
||||||
"@opentelemetry/exporter-trace-otlp-http": "^0.48.0",
|
"@opentelemetry/exporter-trace-otlp-http": "^0.52.0",
|
||||||
"@opentelemetry/instrumentation-document-load": "^0.35.0",
|
"@opentelemetry/instrumentation-document-load": "^0.39.0",
|
||||||
"@opentelemetry/instrumentation-user-interaction": "^0.35.0",
|
"@opentelemetry/instrumentation-user-interaction": "^0.39.0",
|
||||||
"@opentelemetry/sdk-trace-web": "^1.9.1",
|
"@opentelemetry/sdk-trace-web": "^1.9.1",
|
||||||
"@radix-ui/react-dialog": "^1.0.4",
|
"@radix-ui/react-dialog": "^1.0.4",
|
||||||
"@radix-ui/react-slider": "^1.1.2",
|
"@radix-ui/react-slider": "^1.1.2",
|
||||||
@@ -46,7 +47,7 @@
|
|||||||
"@react-stately/select": "^3.1.3",
|
"@react-stately/select": "^3.1.3",
|
||||||
"@react-stately/tooltip": "^3.0.5",
|
"@react-stately/tooltip": "^3.0.5",
|
||||||
"@react-stately/tree": "^3.2.0",
|
"@react-stately/tree": "^3.2.0",
|
||||||
"@sentry/react": "^7.0.0",
|
"@sentry/react": "^8.0.0",
|
||||||
"@sentry/tracing": "^7.0.0",
|
"@sentry/tracing": "^7.0.0",
|
||||||
"@types/lodash": "^4.14.199",
|
"@types/lodash": "^4.14.199",
|
||||||
"@use-gesture/react": "^10.2.11",
|
"@use-gesture/react": "^10.2.11",
|
||||||
@@ -58,11 +59,11 @@
|
|||||||
"classnames": "^2.3.1",
|
"classnames": "^2.3.1",
|
||||||
"events": "^3.3.0",
|
"events": "^3.3.0",
|
||||||
"i18next": "^23.0.0",
|
"i18next": "^23.0.0",
|
||||||
"i18next-browser-languagedetector": "^7.0.0",
|
"i18next-browser-languagedetector": "^8.0.0",
|
||||||
"i18next-http-backend": "^2.0.0",
|
"i18next-http-backend": "^2.0.0",
|
||||||
"livekit-client": "^2.0.2",
|
"livekit-client": "^2.0.2",
|
||||||
"lodash": "^4.17.21",
|
"lodash": "^4.17.21",
|
||||||
"matrix-js-sdk": "github:matrix-org/matrix-js-sdk#8123e9a3f1142a7619758c0a238172b007e3a06a",
|
"matrix-js-sdk": "github:matrix-org/matrix-js-sdk#238eea0ef5c82d0a11b8d5cc5c04104d6c94c4c1",
|
||||||
"matrix-widget-api": "^1.3.1",
|
"matrix-widget-api": "^1.3.1",
|
||||||
"normalize.css": "^8.0.1",
|
"normalize.css": "^8.0.1",
|
||||||
"pako": "^2.0.4",
|
"pako": "^2.0.4",
|
||||||
@@ -70,15 +71,15 @@
|
|||||||
"posthog-js": "^1.29.0",
|
"posthog-js": "^1.29.0",
|
||||||
"react": "18",
|
"react": "18",
|
||||||
"react-dom": "18",
|
"react-dom": "18",
|
||||||
"react-i18next": "^14.0.0",
|
"react-i18next": "^15.0.0",
|
||||||
"react-router-dom": "^5.2.0",
|
"react-router-dom": "^5.2.0",
|
||||||
"react-use-clipboard": "^1.0.7",
|
"react-use-clipboard": "^1.0.7",
|
||||||
"react-use-measure": "^2.1.1",
|
"react-use-measure": "^2.1.1",
|
||||||
"rxjs": "^7.8.1",
|
"rxjs": "^7.8.1",
|
||||||
"sdp-transform": "^2.14.1",
|
"sdp-transform": "^2.14.1",
|
||||||
"tinyqueue": "^2.0.3",
|
"tinyqueue": "^3.0.0",
|
||||||
"unique-names-generator": "^4.6.0",
|
"unique-names-generator": "^4.6.0",
|
||||||
"uuid": "9",
|
"uuid": "10",
|
||||||
"vaul": "^0.9.0"
|
"vaul": "^0.9.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@@ -89,17 +90,19 @@
|
|||||||
"@react-spring/rafz": "^9.7.3",
|
"@react-spring/rafz": "^9.7.3",
|
||||||
"@react-types/dialog": "^3.5.5",
|
"@react-types/dialog": "^3.5.5",
|
||||||
"@sentry/vite-plugin": "^2.0.0",
|
"@sentry/vite-plugin": "^2.0.0",
|
||||||
"@testing-library/react": "^14.0.0",
|
"@testing-library/dom": "^10.1.0",
|
||||||
|
"@testing-library/react": "^16.0.0",
|
||||||
"@testing-library/user-event": "^14.5.1",
|
"@testing-library/user-event": "^14.5.1",
|
||||||
"@types/content-type": "^1.1.5",
|
"@types/content-type": "^1.1.5",
|
||||||
"@types/dom-screen-wake-lock": "^1.0.1",
|
"@types/dom-screen-wake-lock": "^1.0.1",
|
||||||
"@types/dompurify": "^3.0.2",
|
"@types/dompurify": "^3.0.2",
|
||||||
"@types/grecaptcha": "^3.0.4",
|
"@types/grecaptcha": "^3.0.4",
|
||||||
"@types/node": "^20.0.0",
|
"@types/node": "^20.0.0",
|
||||||
|
"@types/react-dom": "^18.3.0",
|
||||||
"@types/react-router-dom": "^5.3.3",
|
"@types/react-router-dom": "^5.3.3",
|
||||||
"@types/request": "^2.48.8",
|
"@types/request": "^2.48.8",
|
||||||
"@types/sdp-transform": "^2.4.5",
|
"@types/sdp-transform": "^2.4.5",
|
||||||
"@types/uuid": "9",
|
"@types/uuid": "10",
|
||||||
"@typescript-eslint/eslint-plugin": "^7.0.0",
|
"@typescript-eslint/eslint-plugin": "^7.0.0",
|
||||||
"@typescript-eslint/parser": "^7.0.0",
|
"@typescript-eslint/parser": "^7.0.0",
|
||||||
"babel-loader": "^9.0.0",
|
"babel-loader": "^9.0.0",
|
||||||
@@ -113,8 +116,8 @@
|
|||||||
"eslint-plugin-matrix-org": "^1.2.1",
|
"eslint-plugin-matrix-org": "^1.2.1",
|
||||||
"eslint-plugin-react": "^7.29.4",
|
"eslint-plugin-react": "^7.29.4",
|
||||||
"eslint-plugin-react-hooks": "^4.5.0",
|
"eslint-plugin-react-hooks": "^4.5.0",
|
||||||
"eslint-plugin-unicorn": "^51.0.0",
|
"eslint-plugin-unicorn": "^54.0.0",
|
||||||
"i18next-parser": "^8.0.0",
|
"i18next-parser": "^9.0.0",
|
||||||
"jsdom": "^24.0.0",
|
"jsdom": "^24.0.0",
|
||||||
"prettier": "^3.0.0",
|
"prettier": "^3.0.0",
|
||||||
"sass": "^1.42.1",
|
"sass": "^1.42.1",
|
||||||
@@ -123,6 +126,6 @@
|
|||||||
"vite": "^5.0.0",
|
"vite": "^5.0.0",
|
||||||
"vite-plugin-html-template": "^1.1.0",
|
"vite-plugin-html-template": "^1.1.0",
|
||||||
"vite-plugin-svgr": "^4.0.0",
|
"vite-plugin-svgr": "^4.0.0",
|
||||||
"vitest": "^1.2.2"
|
"vitest": "^2.0.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,7 +13,8 @@
|
|||||||
</script>
|
</script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body class="cpd-theme-dark">
|
<!-- The default class is: .no-theme {display: none}. It will be overwritten once the app is loaded. -->
|
||||||
|
<body class="no-theme">
|
||||||
<div id="root"></div>
|
<div id="root"></div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@@ -31,8 +31,6 @@
|
|||||||
"username": "Потребителско име",
|
"username": "Потребителско име",
|
||||||
"video": "Видео"
|
"video": "Видео"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Излез от цял екран",
|
|
||||||
"fullscreen_button_label": "Цял екран",
|
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Да, присъедини се",
|
"join_button": "Да, присъедини се",
|
||||||
"text": "Този разговор вече съществува, искате ли да се присъедините?",
|
"text": "Този разговор вече съществува, искате ли да се присъедините?",
|
||||||
@@ -42,7 +40,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Влез в разговора"
|
"join_button": "Влез в разговора"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Локална сила на звука",
|
|
||||||
"logging_in": "Влизане…",
|
"logging_in": "Влизане…",
|
||||||
"login_auth_links": "<0>Създайте акаунт</0> или <2>Влезте като гост</2>",
|
"login_auth_links": "<0>Създайте акаунт</0> или <2>Влезте като гост</2>",
|
||||||
"login_title": "Влез",
|
"login_title": "Влез",
|
||||||
|
|||||||
@@ -29,10 +29,8 @@
|
|||||||
"settings": "Nastavení",
|
"settings": "Nastavení",
|
||||||
"username": "Uživatelské jméno"
|
"username": "Uživatelské jméno"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Ukončit režim celé obrazovky",
|
|
||||||
"full_screen_view_description": "<0>Odeslání ladících záznamů nám pomůže diagnostikovat problém.</0>",
|
"full_screen_view_description": "<0>Odeslání ladících záznamů nám pomůže diagnostikovat problém.</0>",
|
||||||
"full_screen_view_h1": "<0>Oops, něco se pokazilo.</0>",
|
"full_screen_view_h1": "<0>Oops, něco se pokazilo.</0>",
|
||||||
"fullscreen_button_label": "Zvětšit na celou obrazovku",
|
|
||||||
"header_label": "Domov Element Call",
|
"header_label": "Domov Element Call",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Ano, připojit se",
|
"join_button": "Ano, připojit se",
|
||||||
@@ -43,7 +41,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Připojit se k hovoru"
|
"join_button": "Připojit se k hovoru"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Lokální hlasitost",
|
|
||||||
"logging_in": "Přihlašování se…",
|
"logging_in": "Přihlašování se…",
|
||||||
"login_auth_links": "<0>Vytvořit účet</0> Or <2>Jako host</2>",
|
"login_auth_links": "<0>Vytvořit účet</0> Or <2>Jako host</2>",
|
||||||
"login_title": "Přihlášení",
|
"login_title": "Přihlášení",
|
||||||
|
|||||||
@@ -56,10 +56,8 @@
|
|||||||
"video": "Video"
|
"video": "Video"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Die Verbindung zum Server wurde getrennt.",
|
"disconnected_banner": "Die Verbindung zum Server wurde getrennt.",
|
||||||
"exit_fullscreen_button_label": "Vollbildmodus verlassen",
|
|
||||||
"full_screen_view_description": "<0>Übermittelte Problemberichte helfen uns, Fehler zu beheben.</0>",
|
"full_screen_view_description": "<0>Übermittelte Problemberichte helfen uns, Fehler zu beheben.</0>",
|
||||||
"full_screen_view_h1": "<0>Hoppla, etwas ist schiefgelaufen.</0>",
|
"full_screen_view_h1": "<0>Hoppla, etwas ist schiefgelaufen.</0>",
|
||||||
"fullscreen_button_label": "Vollbild",
|
|
||||||
"group_call_loader_failed_heading": "Anruf nicht gefunden",
|
"group_call_loader_failed_heading": "Anruf nicht gefunden",
|
||||||
"group_call_loader_failed_text": "Anrufe sind nun Ende-zu-Ende-verschlüsselt und müssen auf der Startseite erstellt werden. Damit stellen wir sicher, dass alle denselben Schlüssel verwenden.",
|
"group_call_loader_failed_text": "Anrufe sind nun Ende-zu-Ende-verschlüsselt und müssen auf der Startseite erstellt werden. Damit stellen wir sicher, dass alle denselben Schlüssel verwenden.",
|
||||||
"hangup_button_label": "Anruf beenden",
|
"hangup_button_label": "Anruf beenden",
|
||||||
@@ -80,7 +78,6 @@
|
|||||||
"join_button": "Anruf beitreten",
|
"join_button": "Anruf beitreten",
|
||||||
"leave_button": "Zurück zu kürzlichen Anrufen"
|
"leave_button": "Zurück zu kürzlichen Anrufen"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Lokale Lautstärke",
|
|
||||||
"log_in": "Anmelden",
|
"log_in": "Anmelden",
|
||||||
"logging_in": "Anmelden …",
|
"logging_in": "Anmelden …",
|
||||||
"login_auth_links": "<0>Konto erstellen</0> Oder <2>Als Gast betreten</2>",
|
"login_auth_links": "<0>Konto erstellen</0> Oder <2>Als Gast betreten</2>",
|
||||||
@@ -141,7 +138,6 @@
|
|||||||
"unmute_microphone_button_label": "Mikrofon aktivieren",
|
"unmute_microphone_button_label": "Mikrofon aktivieren",
|
||||||
"version": "Version: {{version}}",
|
"version": "Version: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} präsentiert",
|
|
||||||
"sfu_participant_local": "Du"
|
"sfu_participant_local": "Du"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "Warte auf weitere Teilnehmer …"
|
"waiting_for_participants": "Warte auf weitere Teilnehmer …"
|
||||||
|
|||||||
@@ -37,10 +37,8 @@
|
|||||||
"username": "Όνομα χρήστη",
|
"username": "Όνομα χρήστη",
|
||||||
"video": "Βίντεο"
|
"video": "Βίντεο"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Έξοδος από πλήρη οθόνη",
|
|
||||||
"full_screen_view_description": "<0>Η υποβολή αρχείων καταγραφής σφαλμάτων θα μας βοηθήσει να εντοπίσουμε το πρόβλημα.</0>",
|
"full_screen_view_description": "<0>Η υποβολή αρχείων καταγραφής σφαλμάτων θα μας βοηθήσει να εντοπίσουμε το πρόβλημα.</0>",
|
||||||
"full_screen_view_h1": "<0>Ωχ, κάτι πήγε στραβά.</0>",
|
"full_screen_view_h1": "<0>Ωχ, κάτι πήγε στραβά.</0>",
|
||||||
"fullscreen_button_label": "Πλήρη οθόνη",
|
|
||||||
"header_label": "Element Κεντρική Οθόνη Κλήσεων",
|
"header_label": "Element Κεντρική Οθόνη Κλήσεων",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Ναι, συμμετοχή στην κλήση",
|
"join_button": "Ναι, συμμετοχή στην κλήση",
|
||||||
@@ -50,7 +48,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Συμμετοχή στην κλήση"
|
"join_button": "Συμμετοχή στην κλήση"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Τοπική ένταση",
|
|
||||||
"logging_in": "Σύνδεση…",
|
"logging_in": "Σύνδεση…",
|
||||||
"login_auth_links": "<0>Δημιουργήστε λογαριασμό</0> Ή <2>Συμμετέχετε ως επισκέπτης</2>",
|
"login_auth_links": "<0>Δημιουργήστε λογαριασμό</0> Ή <2>Συμμετέχετε ως επισκέπτης</2>",
|
||||||
"login_title": "Σύνδεση",
|
"login_title": "Σύνδεση",
|
||||||
@@ -94,8 +91,5 @@
|
|||||||
"unauthenticated_view_body": "Δεν έχετε εγγραφεί ακόμα; <2>Δημιουργήστε λογαριασμό</2>",
|
"unauthenticated_view_body": "Δεν έχετε εγγραφεί ακόμα; <2>Δημιουργήστε λογαριασμό</2>",
|
||||||
"unauthenticated_view_login_button": "Συνδεθείτε στον λογαριασμό σας",
|
"unauthenticated_view_login_button": "Συνδεθείτε στον λογαριασμό σας",
|
||||||
"version": "Έκδοση: {{version}}",
|
"version": "Έκδοση: {{version}}",
|
||||||
"video_tile": {
|
|
||||||
"presenter_label": "{{displayName}} παρουσιάζει"
|
|
||||||
},
|
|
||||||
"waiting_for_participants": "Αναμονή για άλλους συμμετέχοντες…"
|
"waiting_for_participants": "Αναμονή για άλλους συμμετέχοντες…"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -60,8 +60,17 @@
|
|||||||
"disconnected_banner": "Connectivity to the server has been lost.",
|
"disconnected_banner": "Connectivity to the server has been lost.",
|
||||||
"full_screen_view_description": "<0>Submitting debug logs will help us track down the problem.</0>",
|
"full_screen_view_description": "<0>Submitting debug logs will help us track down the problem.</0>",
|
||||||
"full_screen_view_h1": "<0>Oops, something's gone wrong.</0>",
|
"full_screen_view_h1": "<0>Oops, something's gone wrong.</0>",
|
||||||
"group_call_loader_failed_heading": "Call not found",
|
"group_call_loader": {
|
||||||
"group_call_loader_failed_text": "Calls are now end-to-end encrypted and need to be created from the home page. This helps make sure everyone's using the same encryption key.",
|
"banned_body": "You have been banned from the room.",
|
||||||
|
"banned_heading": "Banned",
|
||||||
|
"call_ended_body": "You have been removed from the call.",
|
||||||
|
"call_ended_heading": "Call ended",
|
||||||
|
"failed_heading": "Failed to join",
|
||||||
|
"failed_text": "Call not found or is not accessible.",
|
||||||
|
"knock_reject_body": "The room members declined your request to join.",
|
||||||
|
"knock_reject_heading": "Not allowed to join",
|
||||||
|
"reason": "Reason"
|
||||||
|
},
|
||||||
"hangup_button_label": "End call",
|
"hangup_button_label": "End call",
|
||||||
"header_label": "Element Call Home",
|
"header_label": "Element Call Home",
|
||||||
"header_participants_label": "Participants",
|
"header_participants_label": "Participants",
|
||||||
@@ -77,8 +86,10 @@
|
|||||||
"layout_grid_label": "Grid",
|
"layout_grid_label": "Grid",
|
||||||
"layout_spotlight_label": "Spotlight",
|
"layout_spotlight_label": "Spotlight",
|
||||||
"lobby": {
|
"lobby": {
|
||||||
|
"ask_to_join": "Ask to join call",
|
||||||
"join_button": "Join call",
|
"join_button": "Join call",
|
||||||
"leave_button": "Back to recents"
|
"leave_button": "Back to recents",
|
||||||
|
"waiting_for_invite": "Request sent"
|
||||||
},
|
},
|
||||||
"log_in": "Log In",
|
"log_in": "Log In",
|
||||||
"logging_in": "Logging in…",
|
"logging_in": "Logging in…",
|
||||||
|
|||||||
@@ -34,10 +34,8 @@
|
|||||||
"settings": "Ajustes",
|
"settings": "Ajustes",
|
||||||
"username": "Nombre de usuario"
|
"username": "Nombre de usuario"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Salir de pantalla completa",
|
|
||||||
"full_screen_view_description": "<0>Subir los registros de depuración nos ayudará a encontrar el problema.</0>",
|
"full_screen_view_description": "<0>Subir los registros de depuración nos ayudará a encontrar el problema.</0>",
|
||||||
"full_screen_view_h1": "<0>Ups, algo ha salido mal.</0>",
|
"full_screen_view_h1": "<0>Ups, algo ha salido mal.</0>",
|
||||||
"fullscreen_button_label": "Pantalla completa",
|
|
||||||
"header_label": "Inicio de Element Call",
|
"header_label": "Inicio de Element Call",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Si, unirse a la llamada",
|
"join_button": "Si, unirse a la llamada",
|
||||||
@@ -48,7 +46,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Unirse a la llamada"
|
"join_button": "Unirse a la llamada"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Volumen local",
|
|
||||||
"logging_in": "Iniciando sesión…",
|
"logging_in": "Iniciando sesión…",
|
||||||
"login_auth_links": "<0>Crear una cuenta</0> o <2>Acceder como invitado</2>",
|
"login_auth_links": "<0>Crear una cuenta</0> o <2>Acceder como invitado</2>",
|
||||||
"login_title": "Iniciar sesión",
|
"login_title": "Iniciar sesión",
|
||||||
@@ -95,8 +92,5 @@
|
|||||||
"unauthenticated_view_eula_caption": "Al hacer clic en \"Comenzar\", aceptas nuestro <2>Contrato de Licencia de Usuario Final (CLUF)</2>",
|
"unauthenticated_view_eula_caption": "Al hacer clic en \"Comenzar\", aceptas nuestro <2>Contrato de Licencia de Usuario Final (CLUF)</2>",
|
||||||
"unauthenticated_view_login_button": "Iniciar sesión en tu cuenta",
|
"unauthenticated_view_login_button": "Iniciar sesión en tu cuenta",
|
||||||
"version": "Versión: {{version}}",
|
"version": "Versión: {{version}}",
|
||||||
"video_tile": {
|
|
||||||
"presenter_label": "{{displayName}} está presentando"
|
|
||||||
},
|
|
||||||
"waiting_for_participants": "Esperando a los otros participantes…"
|
"waiting_for_participants": "Esperando a los otros participantes…"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,10 +52,8 @@
|
|||||||
"username": "Kasutajanimi"
|
"username": "Kasutajanimi"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Võrguühendus serveriga on katkenud.",
|
"disconnected_banner": "Võrguühendus serveriga on katkenud.",
|
||||||
"exit_fullscreen_button_label": "Välju täisekraanivaatest",
|
|
||||||
"full_screen_view_description": "<0>Kui saadad meile vealogid, siis on lihtsam vea põhjust otsida.</0>",
|
"full_screen_view_description": "<0>Kui saadad meile vealogid, siis on lihtsam vea põhjust otsida.</0>",
|
||||||
"full_screen_view_h1": "<0>Ohoo, midagi on nüüd katki.</0>",
|
"full_screen_view_h1": "<0>Ohoo, midagi on nüüd katki.</0>",
|
||||||
"fullscreen_button_label": "Täisekraan",
|
|
||||||
"group_call_loader_failed_heading": "Kõnet ei leidu",
|
"group_call_loader_failed_heading": "Kõnet ei leidu",
|
||||||
"group_call_loader_failed_text": "Kõned on nüüd läbivalt krüptitud ning need pead looma kodulehelt. Sellega tagad, et kõik kasutavad samu krüptovõtmeid.",
|
"group_call_loader_failed_text": "Kõned on nüüd läbivalt krüptitud ning need pead looma kodulehelt. Sellega tagad, et kõik kasutavad samu krüptovõtmeid.",
|
||||||
"hangup_button_label": "Lõpeta kõne",
|
"hangup_button_label": "Lõpeta kõne",
|
||||||
@@ -75,7 +73,6 @@
|
|||||||
"join_button": "Kõnega liitumine",
|
"join_button": "Kõnega liitumine",
|
||||||
"leave_button": "Tagasi hiljutiste kõnede juurde"
|
"leave_button": "Tagasi hiljutiste kõnede juurde"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Kohalik helitugevus",
|
|
||||||
"logging_in": "Sisselogimine …",
|
"logging_in": "Sisselogimine …",
|
||||||
"login_auth_links": "<0>Loo konto</0> Või <2>Sisene külalisena</2>",
|
"login_auth_links": "<0>Loo konto</0> Või <2>Sisene külalisena</2>",
|
||||||
"login_title": "Sisselogimine",
|
"login_title": "Sisselogimine",
|
||||||
@@ -133,7 +130,6 @@
|
|||||||
"unmute_microphone_button_label": "Lülita mikrofon sisse",
|
"unmute_microphone_button_label": "Lülita mikrofon sisse",
|
||||||
"version": "Versioon: {{version}}",
|
"version": "Versioon: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} on esitlemas",
|
|
||||||
"sfu_participant_local": "Sina"
|
"sfu_participant_local": "Sina"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "Ootame teiste osalejate lisandumist…"
|
"waiting_for_participants": "Ootame teiste osalejate lisandumist…"
|
||||||
|
|||||||
@@ -32,8 +32,6 @@
|
|||||||
"username": "نام کاربری",
|
"username": "نام کاربری",
|
||||||
"video": "ویدیو"
|
"video": "ویدیو"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "خروج از حالت تمام صفحه",
|
|
||||||
"fullscreen_button_label": "تمام صحفه",
|
|
||||||
"header_label": "خانهٔ تماس المنت",
|
"header_label": "خانهٔ تماس المنت",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "بله، به تماس بپیوندید",
|
"join_button": "بله، به تماس بپیوندید",
|
||||||
@@ -44,7 +42,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "پیوستن به تماس"
|
"join_button": "پیوستن به تماس"
|
||||||
},
|
},
|
||||||
"local_volume_label": "حجم داخلی",
|
|
||||||
"logging_in": "ورود…",
|
"logging_in": "ورود…",
|
||||||
"login_auth_links": "<0>ساخت حساب کاربری</0> Or <2>دسترسی به عنوان میهمان</2>",
|
"login_auth_links": "<0>ساخت حساب کاربری</0> Or <2>دسترسی به عنوان میهمان</2>",
|
||||||
"login_title": "ورود",
|
"login_title": "ورود",
|
||||||
|
|||||||
@@ -50,10 +50,8 @@
|
|||||||
"video": "Vidéo"
|
"video": "Vidéo"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "La connexion avec le serveur a été perdue.",
|
"disconnected_banner": "La connexion avec le serveur a été perdue.",
|
||||||
"exit_fullscreen_button_label": "Quitter le plein écran",
|
|
||||||
"full_screen_view_description": "<0>Soumettre les journaux de débogage nous aidera à déterminer le problème.</0>",
|
"full_screen_view_description": "<0>Soumettre les journaux de débogage nous aidera à déterminer le problème.</0>",
|
||||||
"full_screen_view_h1": "<0>Oups, quelque chose s’est mal passé.</0>",
|
"full_screen_view_h1": "<0>Oups, quelque chose s’est mal passé.</0>",
|
||||||
"fullscreen_button_label": "Plein écran",
|
|
||||||
"group_call_loader_failed_heading": "Appel non trouvé",
|
"group_call_loader_failed_heading": "Appel non trouvé",
|
||||||
"group_call_loader_failed_text": "Les appels sont maintenant chiffrés de bout-en-bout et doivent être créés depuis la page d’accueil. Cela permet d’être sûr que tout le monde utilise la même clé de chiffrement.",
|
"group_call_loader_failed_text": "Les appels sont maintenant chiffrés de bout-en-bout et doivent être créés depuis la page d’accueil. Cela permet d’être sûr que tout le monde utilise la même clé de chiffrement.",
|
||||||
"hangup_button_label": "Terminer l’appel",
|
"hangup_button_label": "Terminer l’appel",
|
||||||
@@ -73,7 +71,6 @@
|
|||||||
"join_button": "Rejoindre l’appel",
|
"join_button": "Rejoindre l’appel",
|
||||||
"leave_button": "Revenir à l’historique des appels"
|
"leave_button": "Revenir à l’historique des appels"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Volume local",
|
|
||||||
"logging_in": "Connexion…",
|
"logging_in": "Connexion…",
|
||||||
"login_auth_links": "<0>Créer un compte</0> Or <2>Accès invité</2>",
|
"login_auth_links": "<0>Créer un compte</0> Or <2>Accès invité</2>",
|
||||||
"login_title": "Connexion",
|
"login_title": "Connexion",
|
||||||
@@ -131,7 +128,6 @@
|
|||||||
"unmute_microphone_button_label": "Allumer le microphone",
|
"unmute_microphone_button_label": "Allumer le microphone",
|
||||||
"version": "Version : {{version}}",
|
"version": "Version : {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} est à l’écran",
|
|
||||||
"sfu_participant_local": "Vous"
|
"sfu_participant_local": "Vous"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "En attente d’autres participants…"
|
"waiting_for_participants": "En attente d’autres participants…"
|
||||||
|
|||||||
@@ -50,10 +50,8 @@
|
|||||||
"username": "Nama pengguna"
|
"username": "Nama pengguna"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Koneksi ke server telah hilang.",
|
"disconnected_banner": "Koneksi ke server telah hilang.",
|
||||||
"exit_fullscreen_button_label": "Keluar dari layar penuh",
|
|
||||||
"full_screen_view_description": "<0>Mengirim catatan pengawakutuan akan membantu kami melacak masalahnya.</0>",
|
"full_screen_view_description": "<0>Mengirim catatan pengawakutuan akan membantu kami melacak masalahnya.</0>",
|
||||||
"full_screen_view_h1": "<0>Aduh, ada yang salah.</0>",
|
"full_screen_view_h1": "<0>Aduh, ada yang salah.</0>",
|
||||||
"fullscreen_button_label": "Layar penuh",
|
|
||||||
"group_call_loader_failed_heading": "Panggilan tidak ditemukan",
|
"group_call_loader_failed_heading": "Panggilan tidak ditemukan",
|
||||||
"group_call_loader_failed_text": "Panggilan sekarang terenkripsi secara ujung ke ujung dan harus dibuat dari laman beranda. Ini memastikan bahwa semuanya menggunakan kunci enkripsi yang sama.",
|
"group_call_loader_failed_text": "Panggilan sekarang terenkripsi secara ujung ke ujung dan harus dibuat dari laman beranda. Ini memastikan bahwa semuanya menggunakan kunci enkripsi yang sama.",
|
||||||
"hangup_button_label": "Akhiri panggilan",
|
"hangup_button_label": "Akhiri panggilan",
|
||||||
@@ -74,7 +72,6 @@
|
|||||||
"join_button": "Bergabung ke panggilan",
|
"join_button": "Bergabung ke panggilan",
|
||||||
"leave_button": "Kembali ke terkini"
|
"leave_button": "Kembali ke terkini"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Volume lokal",
|
|
||||||
"logging_in": "Memasuki…",
|
"logging_in": "Memasuki…",
|
||||||
"login_auth_links": "<0>Buat akun</0> Atau <2>Akses sebagai tamu</2>",
|
"login_auth_links": "<0>Buat akun</0> Atau <2>Akses sebagai tamu</2>",
|
||||||
"login_title": "Masuk",
|
"login_title": "Masuk",
|
||||||
@@ -132,7 +129,6 @@
|
|||||||
"unmute_microphone_button_label": "Nyalakan mikrofon",
|
"unmute_microphone_button_label": "Nyalakan mikrofon",
|
||||||
"version": "Versi: {{version}}",
|
"version": "Versi: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} sedang menampilkan",
|
|
||||||
"sfu_participant_local": "Anda"
|
"sfu_participant_local": "Anda"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "Menunggu peserta lain…"
|
"waiting_for_participants": "Menunggu peserta lain…"
|
||||||
|
|||||||
@@ -48,10 +48,8 @@
|
|||||||
"username": "Nome utente"
|
"username": "Nome utente"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "La connessione al server è stata persa.",
|
"disconnected_banner": "La connessione al server è stata persa.",
|
||||||
"exit_fullscreen_button_label": "Esci da schermo intero",
|
|
||||||
"full_screen_view_description": "<0>L'invio di registri di debug ci aiuterà ad individuare il problema.</0>",
|
"full_screen_view_description": "<0>L'invio di registri di debug ci aiuterà ad individuare il problema.</0>",
|
||||||
"full_screen_view_h1": "<0>Ops, qualcosa è andato storto.</0>",
|
"full_screen_view_h1": "<0>Ops, qualcosa è andato storto.</0>",
|
||||||
"fullscreen_button_label": "Schermo intero",
|
|
||||||
"group_call_loader_failed_heading": "Chiamata non trovata",
|
"group_call_loader_failed_heading": "Chiamata non trovata",
|
||||||
"group_call_loader_failed_text": "Le chiamate ora sono cifrate end-to-end e devono essere create dalla pagina principale. Ciò assicura che chiunque usi la stessa chiave di crittografia.",
|
"group_call_loader_failed_text": "Le chiamate ora sono cifrate end-to-end e devono essere create dalla pagina principale. Ciò assicura che chiunque usi la stessa chiave di crittografia.",
|
||||||
"hangup_button_label": "Termina chiamata",
|
"hangup_button_label": "Termina chiamata",
|
||||||
@@ -72,7 +70,6 @@
|
|||||||
"join_button": "Entra in chiamata",
|
"join_button": "Entra in chiamata",
|
||||||
"leave_button": "Torna ai recenti"
|
"leave_button": "Torna ai recenti"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Volume locale",
|
|
||||||
"logging_in": "Accesso…",
|
"logging_in": "Accesso…",
|
||||||
"login_auth_links": "<0>Crea un profilo</0> o <2>Accedi come ospite</2>",
|
"login_auth_links": "<0>Crea un profilo</0> o <2>Accedi come ospite</2>",
|
||||||
"login_title": "Accedi",
|
"login_title": "Accedi",
|
||||||
@@ -129,7 +126,6 @@
|
|||||||
"unmute_microphone_button_label": "Riaccendi il microfono",
|
"unmute_microphone_button_label": "Riaccendi il microfono",
|
||||||
"version": "Versione: {{version}}",
|
"version": "Versione: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} sta presentando",
|
|
||||||
"sfu_participant_local": "Tu"
|
"sfu_participant_local": "Tu"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "In attesa di altri partecipanti…"
|
"waiting_for_participants": "In attesa di altri partecipanti…"
|
||||||
|
|||||||
@@ -30,9 +30,7 @@
|
|||||||
"username": "ユーザー名",
|
"username": "ユーザー名",
|
||||||
"video": "ビデオ"
|
"video": "ビデオ"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "全画面表示を終了",
|
|
||||||
"full_screen_view_h1": "<0>何かがうまく行きませんでした。</0>",
|
"full_screen_view_h1": "<0>何かがうまく行きませんでした。</0>",
|
||||||
"fullscreen_button_label": "全画面表示",
|
|
||||||
"header_label": "Element Call ホーム",
|
"header_label": "Element Call ホーム",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "はい、通話に参加",
|
"join_button": "はい、通話に参加",
|
||||||
|
|||||||
@@ -40,10 +40,8 @@
|
|||||||
"username": "Lietotājvārds"
|
"username": "Lietotājvārds"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Ir zaudēts savienojums ar serveri.",
|
"disconnected_banner": "Ir zaudēts savienojums ar serveri.",
|
||||||
"exit_fullscreen_button_label": "Iziet no pilnekrāna",
|
|
||||||
"full_screen_view_description": "<0>Atkļūdošanas žurnāla ierakstu iesūtīšana palīdzēs mums atklāt nepilnību.</0>",
|
"full_screen_view_description": "<0>Atkļūdošanas žurnāla ierakstu iesūtīšana palīdzēs mums atklāt nepilnību.</0>",
|
||||||
"full_screen_view_h1": "<0>Ak vai, kaut kas nogāja greizi!</0>",
|
"full_screen_view_h1": "<0>Ak vai, kaut kas nogāja greizi!</0>",
|
||||||
"fullscreen_button_label": "Pilnekrāns",
|
|
||||||
"header_label": "Element Call sākums",
|
"header_label": "Element Call sākums",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Jā, pievienoties zvanam",
|
"join_button": "Jā, pievienoties zvanam",
|
||||||
@@ -54,7 +52,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Pievienoties zvanam"
|
"join_button": "Pievienoties zvanam"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Vietējais skaļums",
|
|
||||||
"logging_in": "Piesakās…",
|
"logging_in": "Piesakās…",
|
||||||
"login_auth_links": "<0>Izveidot kontu</0> vai <2>Piekļūt kā viesim</2>",
|
"login_auth_links": "<0>Izveidot kontu</0> vai <2>Piekļūt kā viesim</2>",
|
||||||
"login_title": "Pieteikties",
|
"login_title": "Pieteikties",
|
||||||
@@ -103,8 +100,5 @@
|
|||||||
"unauthenticated_view_eula_caption": "Klikšķināšana uz \"Aiziet\" apliecina piekrišanu mūsu <2>galalietotāja licencēšanas nolīgumam (GLLN)</2>",
|
"unauthenticated_view_eula_caption": "Klikšķināšana uz \"Aiziet\" apliecina piekrišanu mūsu <2>galalietotāja licencēšanas nolīgumam (GLLN)</2>",
|
||||||
"unauthenticated_view_login_button": "Pieteikties kontā",
|
"unauthenticated_view_login_button": "Pieteikties kontā",
|
||||||
"version": "Versija: {{version}}",
|
"version": "Versija: {{version}}",
|
||||||
"video_tile": {
|
|
||||||
"presenter_label": "{{displayName}} uzstājas"
|
|
||||||
},
|
|
||||||
"waiting_for_participants": "Gaida citus dalībniekus…"
|
"waiting_for_participants": "Gaida citus dalībniekus…"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -53,10 +53,8 @@
|
|||||||
"video": "Wideo"
|
"video": "Wideo"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Utracono połączenie z serwerem.",
|
"disconnected_banner": "Utracono połączenie z serwerem.",
|
||||||
"exit_fullscreen_button_label": "Opuść pełny ekran",
|
|
||||||
"full_screen_view_description": "<0>Wysłanie dzienników debuggowania pomoże nam ustalić przyczynę problemu.</0>",
|
"full_screen_view_description": "<0>Wysłanie dzienników debuggowania pomoże nam ustalić przyczynę problemu.</0>",
|
||||||
"full_screen_view_h1": "<0>Ojej, coś poszło nie tak.</0>",
|
"full_screen_view_h1": "<0>Ojej, coś poszło nie tak.</0>",
|
||||||
"fullscreen_button_label": "Pełny ekran",
|
|
||||||
"group_call_loader_failed_heading": "Nie znaleziono połączenia",
|
"group_call_loader_failed_heading": "Nie znaleziono połączenia",
|
||||||
"group_call_loader_failed_text": "Połączenia są teraz szyfrowane end-to-end i muszą zostać utworzone ze strony głównej. Pomaga to upewnić się, że każdy korzysta z tego samego klucza szyfrującego.",
|
"group_call_loader_failed_text": "Połączenia są teraz szyfrowane end-to-end i muszą zostać utworzone ze strony głównej. Pomaga to upewnić się, że każdy korzysta z tego samego klucza szyfrującego.",
|
||||||
"hangup_button_label": "Zakończ połączenie",
|
"hangup_button_label": "Zakończ połączenie",
|
||||||
@@ -77,7 +75,6 @@
|
|||||||
"join_button": "Dołącz do połączenia",
|
"join_button": "Dołącz do połączenia",
|
||||||
"leave_button": "Wróć do ostatnie"
|
"leave_button": "Wróć do ostatnie"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Głośność lokalna",
|
|
||||||
"logging_in": "Logowanie…",
|
"logging_in": "Logowanie…",
|
||||||
"login_auth_links": "<0>Utwórz konto</0> lub <2>Dołącz jako gość</2>",
|
"login_auth_links": "<0>Utwórz konto</0> lub <2>Dołącz jako gość</2>",
|
||||||
"login_title": "Zaloguj się",
|
"login_title": "Zaloguj się",
|
||||||
@@ -135,7 +132,6 @@
|
|||||||
"unmute_microphone_button_label": "Odcisz mikrofon",
|
"unmute_microphone_button_label": "Odcisz mikrofon",
|
||||||
"version": "Wersja: {{version}}",
|
"version": "Wersja: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} prezentuje",
|
|
||||||
"sfu_participant_local": "Ty"
|
"sfu_participant_local": "Ty"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "Oczekiwanie na pozostałych uczestników…"
|
"waiting_for_participants": "Oczekiwanie na pozostałych uczestników…"
|
||||||
|
|||||||
@@ -38,10 +38,8 @@
|
|||||||
"username": "Имя пользователя",
|
"username": "Имя пользователя",
|
||||||
"video": "Видео"
|
"video": "Видео"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Выйти из полноэкранного режима",
|
|
||||||
"full_screen_view_description": "<0>Отправка журналов поможет нам найти и устранить проблему.</0>",
|
"full_screen_view_description": "<0>Отправка журналов поможет нам найти и устранить проблему.</0>",
|
||||||
"full_screen_view_h1": "<0>Упс, что-то пошло не так.</0>",
|
"full_screen_view_h1": "<0>Упс, что-то пошло не так.</0>",
|
||||||
"fullscreen_button_label": "Полноэкранный режим",
|
|
||||||
"header_label": "Главная Element Call",
|
"header_label": "Главная Element Call",
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Да, присоединиться",
|
"join_button": "Да, присоединиться",
|
||||||
@@ -52,7 +50,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Присоединиться"
|
"join_button": "Присоединиться"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Местная громкость",
|
|
||||||
"logging_in": "Вход…",
|
"logging_in": "Вход…",
|
||||||
"login_auth_links": "<0>Создать аккаунт</0> или <2>Зайти как гость</2>",
|
"login_auth_links": "<0>Создать аккаунт</0> или <2>Зайти как гость</2>",
|
||||||
"login_title": "Вход",
|
"login_title": "Вход",
|
||||||
@@ -96,8 +93,5 @@
|
|||||||
"unauthenticated_view_body": "Ещё не зарегистрированы? <2>Создайте аккаунт</2>",
|
"unauthenticated_view_body": "Ещё не зарегистрированы? <2>Создайте аккаунт</2>",
|
||||||
"unauthenticated_view_login_button": "Войдите в свой аккаунт",
|
"unauthenticated_view_login_button": "Войдите в свой аккаунт",
|
||||||
"version": "Версия: {{version}}",
|
"version": "Версия: {{version}}",
|
||||||
"video_tile": {
|
|
||||||
"presenter_label": "{{displayName}} представляет"
|
|
||||||
},
|
|
||||||
"waiting_for_participants": "Ожидание других участников…"
|
"waiting_for_participants": "Ожидание других участников…"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,10 +51,8 @@
|
|||||||
"username": "Meno používateľa"
|
"username": "Meno používateľa"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Spojenie so serverom sa stratilo.",
|
"disconnected_banner": "Spojenie so serverom sa stratilo.",
|
||||||
"exit_fullscreen_button_label": "Ukončiť zobrazenie na celú obrazovku",
|
|
||||||
"full_screen_view_description": "<0>Odoslanie záznamov ladenia nám pomôže nájsť problém.</0>",
|
"full_screen_view_description": "<0>Odoslanie záznamov ladenia nám pomôže nájsť problém.</0>",
|
||||||
"full_screen_view_h1": "<0>Hups, niečo sa pokazilo.</0>",
|
"full_screen_view_h1": "<0>Hups, niečo sa pokazilo.</0>",
|
||||||
"fullscreen_button_label": "Zobrazenie na celú obrazovku",
|
|
||||||
"group_call_loader_failed_heading": "Hovor nebol nájdený",
|
"group_call_loader_failed_heading": "Hovor nebol nájdený",
|
||||||
"group_call_loader_failed_text": "Hovory sú teraz end-to-end šifrované a je potrebné ich vytvoriť z domovskej stránky. To pomáha zabezpečiť, aby všetci používali rovnaký šifrovací kľúč.",
|
"group_call_loader_failed_text": "Hovory sú teraz end-to-end šifrované a je potrebné ich vytvoriť z domovskej stránky. To pomáha zabezpečiť, aby všetci používali rovnaký šifrovací kľúč.",
|
||||||
"hangup_button_label": "Ukončiť hovor",
|
"hangup_button_label": "Ukončiť hovor",
|
||||||
@@ -75,7 +73,6 @@
|
|||||||
"join_button": "Pripojiť sa k hovoru",
|
"join_button": "Pripojiť sa k hovoru",
|
||||||
"leave_button": "Späť k nedávnym"
|
"leave_button": "Späť k nedávnym"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Lokálna hlasitosť",
|
|
||||||
"logging_in": "Prihlasovanie…",
|
"logging_in": "Prihlasovanie…",
|
||||||
"login_auth_links": "<0>Vytvoriť konto</0> Alebo <2>Prihlásiť sa ako hosť</2>",
|
"login_auth_links": "<0>Vytvoriť konto</0> Alebo <2>Prihlásiť sa ako hosť</2>",
|
||||||
"login_title": "Prihlásiť sa",
|
"login_title": "Prihlásiť sa",
|
||||||
@@ -133,7 +130,6 @@
|
|||||||
"unmute_microphone_button_label": "Zrušiť stlmenie mikrofónu",
|
"unmute_microphone_button_label": "Zrušiť stlmenie mikrofónu",
|
||||||
"version": "Verzia: {{version}}",
|
"version": "Verzia: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} prezentuje",
|
|
||||||
"sfu_participant_local": "Vy"
|
"sfu_participant_local": "Vy"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "Čaká sa na ďalších účastníkov…"
|
"waiting_for_participants": "Čaká sa na ďalších účastníkov…"
|
||||||
|
|||||||
@@ -3,8 +3,5 @@
|
|||||||
"headline": "{{displayName}}, ditt samtal har avslutats."
|
"headline": "{{displayName}}, ditt samtal har avslutats."
|
||||||
},
|
},
|
||||||
"star_rating_input_label_one": "{{count}} stjärna",
|
"star_rating_input_label_one": "{{count}} stjärna",
|
||||||
"star_rating_input_label_other": "{{count}} stjärnor",
|
"star_rating_input_label_other": "{{count}} stjärnor"
|
||||||
"video_tile": {
|
|
||||||
"presenter_label": "{{displayName}} presenterar"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,8 +24,6 @@
|
|||||||
"password": "Parola",
|
"password": "Parola",
|
||||||
"settings": "Ayarlar"
|
"settings": "Ayarlar"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Tam ekranı terk et",
|
|
||||||
"fullscreen_button_label": "Tam ekran",
|
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"text": "Bu arama zaten var, katılmak ister misiniz?",
|
"text": "Bu arama zaten var, katılmak ister misiniz?",
|
||||||
"title": "Mevcut aramaya katıl?"
|
"title": "Mevcut aramaya katıl?"
|
||||||
@@ -33,7 +31,6 @@
|
|||||||
"lobby": {
|
"lobby": {
|
||||||
"join_button": "Aramaya katıl"
|
"join_button": "Aramaya katıl"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Yerel ses seviyesi",
|
|
||||||
"logging_in": "Giriliyor…",
|
"logging_in": "Giriliyor…",
|
||||||
"login_auth_links": "<0>Hesap oluştur</0> yahut <2>Konuk olarak gir</2>",
|
"login_auth_links": "<0>Hesap oluştur</0> yahut <2>Konuk olarak gir</2>",
|
||||||
"login_title": "Gir",
|
"login_title": "Gir",
|
||||||
|
|||||||
@@ -53,10 +53,8 @@
|
|||||||
"video": "Відео"
|
"video": "Відео"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "Втрачено зв'язок з сервером.",
|
"disconnected_banner": "Втрачено зв'язок з сервером.",
|
||||||
"exit_fullscreen_button_label": "Вийти з повноекранного режиму",
|
|
||||||
"full_screen_view_description": "<0>Надсилання журналів налагодження допоможе нам виявити проблему.</0>",
|
"full_screen_view_description": "<0>Надсилання журналів налагодження допоможе нам виявити проблему.</0>",
|
||||||
"full_screen_view_h1": "<0>Йой, щось пішло не за планом.</0>",
|
"full_screen_view_h1": "<0>Йой, щось пішло не за планом.</0>",
|
||||||
"fullscreen_button_label": "Повноекранний режим",
|
|
||||||
"group_call_loader_failed_heading": "Виклик не знайдено",
|
"group_call_loader_failed_heading": "Виклик не знайдено",
|
||||||
"group_call_loader_failed_text": "Відтепер виклики захищено наскрізним шифруванням, і їх потрібно створювати з домашньої сторінки. Це допомагає переконатися, що всі користувачі використовують один і той самий ключ шифрування.",
|
"group_call_loader_failed_text": "Відтепер виклики захищено наскрізним шифруванням, і їх потрібно створювати з домашньої сторінки. Це допомагає переконатися, що всі користувачі використовують один і той самий ключ шифрування.",
|
||||||
"hangup_button_label": "Завершити виклик",
|
"hangup_button_label": "Завершити виклик",
|
||||||
@@ -77,7 +75,6 @@
|
|||||||
"join_button": "Приєднатися до виклику",
|
"join_button": "Приєднатися до виклику",
|
||||||
"leave_button": "Повернутися до недавніх"
|
"leave_button": "Повернутися до недавніх"
|
||||||
},
|
},
|
||||||
"local_volume_label": "Локальна гучність",
|
|
||||||
"logging_in": "Вхід…",
|
"logging_in": "Вхід…",
|
||||||
"login_auth_links": "<0>Створити обліковий запис</0> або <2>Отримати доступ як гість</2>",
|
"login_auth_links": "<0>Створити обліковий запис</0> або <2>Отримати доступ як гість</2>",
|
||||||
"login_title": "Увійти",
|
"login_title": "Увійти",
|
||||||
@@ -135,7 +132,6 @@
|
|||||||
"unmute_microphone_button_label": "Увімкнути мікрофон",
|
"unmute_microphone_button_label": "Увімкнути мікрофон",
|
||||||
"version": "Версія: {{version}}",
|
"version": "Версія: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} представляє",
|
|
||||||
"sfu_participant_local": "Ви"
|
"sfu_participant_local": "Ви"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "Очікування на інших учасників…"
|
"waiting_for_participants": "Очікування на інших учасників…"
|
||||||
|
|||||||
@@ -29,10 +29,8 @@
|
|||||||
"username": "Tên người dùng",
|
"username": "Tên người dùng",
|
||||||
"video": "Truyền hình"
|
"video": "Truyền hình"
|
||||||
},
|
},
|
||||||
"exit_fullscreen_button_label": "Rời chế độ toàn màn hình",
|
|
||||||
"full_screen_view_description": "<0>Gửi nhật ký gỡ lỗi sẽ giúp chúng tôi theo dõi vấn đề.</0>",
|
"full_screen_view_description": "<0>Gửi nhật ký gỡ lỗi sẽ giúp chúng tôi theo dõi vấn đề.</0>",
|
||||||
"full_screen_view_h1": "<0>Ối, có cái gì đó sai.</0>",
|
"full_screen_view_h1": "<0>Ối, có cái gì đó sai.</0>",
|
||||||
"fullscreen_button_label": "Toàn màn hình",
|
|
||||||
"join_existing_call_modal": {
|
"join_existing_call_modal": {
|
||||||
"join_button": "Vâng, tham gia cuộc gọi",
|
"join_button": "Vâng, tham gia cuộc gọi",
|
||||||
"text": "Cuộc gọi đã tồn tại, bạn có muốn tham gia không?",
|
"text": "Cuộc gọi đã tồn tại, bạn có muốn tham gia không?",
|
||||||
@@ -73,8 +71,5 @@
|
|||||||
"unauthenticated_view_body": "Chưa đăng ký? <2>Tạo tài khoản</2>",
|
"unauthenticated_view_body": "Chưa đăng ký? <2>Tạo tài khoản</2>",
|
||||||
"unauthenticated_view_login_button": "Đăng nhập vào tài khoản của bạn",
|
"unauthenticated_view_login_button": "Đăng nhập vào tài khoản của bạn",
|
||||||
"version": "Phiên bản: {{version}}",
|
"version": "Phiên bản: {{version}}",
|
||||||
"video_tile": {
|
|
||||||
"presenter_label": "{{displayName}} đang trình bày"
|
|
||||||
},
|
|
||||||
"waiting_for_participants": "Đang đợi những người khác…"
|
"waiting_for_participants": "Đang đợi những người khác…"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,10 +51,8 @@
|
|||||||
"video": "视频"
|
"video": "视频"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "与服务器的连接中断。",
|
"disconnected_banner": "与服务器的连接中断。",
|
||||||
"exit_fullscreen_button_label": "退出全屏",
|
|
||||||
"full_screen_view_description": "<0>提交日志以帮助我们修复问题。</0>",
|
"full_screen_view_description": "<0>提交日志以帮助我们修复问题。</0>",
|
||||||
"full_screen_view_h1": "<0>哎哟,出问题了。</0>",
|
"full_screen_view_h1": "<0>哎哟,出问题了。</0>",
|
||||||
"fullscreen_button_label": "全屏",
|
|
||||||
"group_call_loader_failed_heading": "未找到通话",
|
"group_call_loader_failed_heading": "未找到通话",
|
||||||
"group_call_loader_failed_text": "现在,通话是端对端加密的,需要从主页创建。这有助于确保每个人都使用相同的加密密钥。",
|
"group_call_loader_failed_text": "现在,通话是端对端加密的,需要从主页创建。这有助于确保每个人都使用相同的加密密钥。",
|
||||||
"hangup_button_label": "通话结束",
|
"hangup_button_label": "通话结束",
|
||||||
@@ -70,7 +68,6 @@
|
|||||||
"join_button": "加入通话",
|
"join_button": "加入通话",
|
||||||
"leave_button": "返回最近通话"
|
"leave_button": "返回最近通话"
|
||||||
},
|
},
|
||||||
"local_volume_label": "本地音量",
|
|
||||||
"logging_in": "登录中……",
|
"logging_in": "登录中……",
|
||||||
"login_auth_links": "<0>创建账户</0> Or <2>以访客身份继续</2>",
|
"login_auth_links": "<0>创建账户</0> Or <2>以访客身份继续</2>",
|
||||||
"login_title": "登录",
|
"login_title": "登录",
|
||||||
@@ -128,7 +125,6 @@
|
|||||||
"unmute_microphone_button_label": "取消麦克风静音",
|
"unmute_microphone_button_label": "取消麦克风静音",
|
||||||
"version": "版本:{{version}}",
|
"version": "版本:{{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}}正在展示",
|
|
||||||
"sfu_participant_local": "你"
|
"sfu_participant_local": "你"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "等待其他参与者……"
|
"waiting_for_participants": "等待其他参与者……"
|
||||||
|
|||||||
@@ -53,10 +53,8 @@
|
|||||||
"video": "視訊"
|
"video": "視訊"
|
||||||
},
|
},
|
||||||
"disconnected_banner": "到伺服器的連線已遺失。",
|
"disconnected_banner": "到伺服器的連線已遺失。",
|
||||||
"exit_fullscreen_button_label": "退出全螢幕",
|
|
||||||
"full_screen_view_description": "<0>送出除錯紀錄,可幫助我們修正問題。</0>",
|
"full_screen_view_description": "<0>送出除錯紀錄,可幫助我們修正問題。</0>",
|
||||||
"full_screen_view_h1": "<0>喔喔,有些地方怪怪的。</0>",
|
"full_screen_view_h1": "<0>喔喔,有些地方怪怪的。</0>",
|
||||||
"fullscreen_button_label": "全螢幕",
|
|
||||||
"group_call_loader_failed_heading": "找不到通話",
|
"group_call_loader_failed_heading": "找不到通話",
|
||||||
"group_call_loader_failed_text": "通話現在是端對端加密的,必須從首頁建立。這有助於確保每個人都使用相同的加密金鑰。",
|
"group_call_loader_failed_text": "通話現在是端對端加密的,必須從首頁建立。這有助於確保每個人都使用相同的加密金鑰。",
|
||||||
"hangup_button_label": "結束通話",
|
"hangup_button_label": "結束通話",
|
||||||
@@ -77,7 +75,6 @@
|
|||||||
"join_button": "加入通話",
|
"join_button": "加入通話",
|
||||||
"leave_button": "回到最近的通話"
|
"leave_button": "回到最近的通話"
|
||||||
},
|
},
|
||||||
"local_volume_label": "您的音量",
|
|
||||||
"logging_in": "登入中…",
|
"logging_in": "登入中…",
|
||||||
"login_auth_links": "<0>建立帳號</0> 或<2>以訪客身份登入</2>",
|
"login_auth_links": "<0>建立帳號</0> 或<2>以訪客身份登入</2>",
|
||||||
"login_title": "登入",
|
"login_title": "登入",
|
||||||
@@ -135,7 +132,6 @@
|
|||||||
"unmute_microphone_button_label": "將麥克風取消靜音",
|
"unmute_microphone_button_label": "將麥克風取消靜音",
|
||||||
"version": "版本: {{version}}",
|
"version": "版本: {{version}}",
|
||||||
"video_tile": {
|
"video_tile": {
|
||||||
"presenter_label": "{{displayName}} 正在展示",
|
|
||||||
"sfu_participant_local": "您"
|
"sfu_participant_local": "您"
|
||||||
},
|
},
|
||||||
"waiting_for_participants": "等待其他參加者…"
|
"waiting_for_participants": "等待其他參加者…"
|
||||||
|
|||||||
@@ -3,9 +3,42 @@
|
|||||||
"extends": ["config:base"],
|
"extends": ["config:base"],
|
||||||
"packageRules": [
|
"packageRules": [
|
||||||
{
|
{
|
||||||
"description": "Disable renoavte for packages we want to monitor ourselves",
|
"extends": ["group:allNonMajor", "schedule:weekly"]
|
||||||
"matchPackagePatterns": ["matrix-js-sdk"],
|
},
|
||||||
|
{
|
||||||
|
"groupName": "GitHub Actions",
|
||||||
|
"matchDepTypes": ["action"],
|
||||||
|
"pinDigests": true,
|
||||||
|
"extends": ["schedule:monthly"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"description": "Disable Renovate for packages we want to monitor ourselves",
|
||||||
|
"groupName": "manually updated packages",
|
||||||
|
"matchDepNames": ["matrix-js-sdk"],
|
||||||
"enabled": false
|
"enabled": false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"groupName": "matrix-widget-api",
|
||||||
|
"matchDepNames": ["matrix-widget-api"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"groupName": "Compound",
|
||||||
|
"matchPackagePrefixes": ["@vector-im/compound-"],
|
||||||
|
"schedule": "before 5am on Tuesday and Friday"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"groupName": "LiveKit client",
|
||||||
|
"matchDepNames": ["livekit-client"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"groupName": "LiveKit components",
|
||||||
|
"matchPackagePrefixes": ["@livekit/components-"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"groupName": "Vaul",
|
||||||
|
"matchDepNames": ["vaul"],
|
||||||
|
"extends": ["schedule:monthly"],
|
||||||
|
"prHeader": "Please review modals on mobile for visual regressions."
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"semanticCommits": "disabled",
|
"semanticCommits": "disabled",
|
||||||
|
|||||||
6
src/@types/global.d.ts
vendored
6
src/@types/global.d.ts
vendored
@@ -28,12 +28,6 @@ declare global {
|
|||||||
OLM_OPTIONS: Record<string, string>;
|
OLM_OPTIONS: Record<string, string>;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TypeScript doesn't know about the experimental setSinkId method, so we
|
|
||||||
// declare it ourselves
|
|
||||||
interface MediaElement extends HTMLVideoElement {
|
|
||||||
setSinkId: (id: string) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface HTMLElement {
|
interface HTMLElement {
|
||||||
// Safari only supports this prefixed, so tell the type system about it
|
// Safari only supports this prefixed, so tell the type system about it
|
||||||
webkitRequestFullscreen: () => void;
|
webkitRequestFullscreen: () => void;
|
||||||
|
|||||||
@@ -72,7 +72,9 @@ export const App: FC<AppProps> = ({ history }) => {
|
|||||||
const [loaded, setLoaded] = useState(false);
|
const [loaded, setLoaded] = useState(false);
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
Initializer.init()?.then(() => {
|
Initializer.init()?.then(() => {
|
||||||
|
if (loaded) return;
|
||||||
setLoaded(true);
|
setLoaded(true);
|
||||||
|
widget?.api.sendContentLoaded();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -25,7 +25,11 @@ import {
|
|||||||
useMemo,
|
useMemo,
|
||||||
} from "react";
|
} from "react";
|
||||||
import { useHistory } from "react-router-dom";
|
import { useHistory } from "react-router-dom";
|
||||||
import { ClientEvent, MatrixClient } from "matrix-js-sdk/src/client";
|
import {
|
||||||
|
ClientEvent,
|
||||||
|
ICreateClientOpts,
|
||||||
|
MatrixClient,
|
||||||
|
} from "matrix-js-sdk/src/client";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { ISyncStateData, SyncState } from "matrix-js-sdk/src/sync";
|
import { ISyncStateData, SyncState } from "matrix-js-sdk/src/sync";
|
||||||
@@ -317,7 +321,7 @@ export const ClientProvider: FC<Props> = ({ children }) => {
|
|||||||
initClientState.client.on(ClientEvent.Sync, onSync);
|
initClientState.client.on(ClientEvent.Sync, onSync);
|
||||||
}
|
}
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
if (initClientState.client) {
|
if (initClientState.client) {
|
||||||
initClientState.client.removeListener(ClientEvent.Sync, onSync);
|
initClientState.client.removeListener(ClientEvent.Sync, onSync);
|
||||||
}
|
}
|
||||||
@@ -360,13 +364,13 @@ async function loadClient(): Promise<InitResult | null> {
|
|||||||
|
|
||||||
/* eslint-disable camelcase */
|
/* eslint-disable camelcase */
|
||||||
const { user_id, device_id, access_token, passwordlessUser } = session;
|
const { user_id, device_id, access_token, passwordlessUser } = session;
|
||||||
const initClientParams = {
|
const initClientParams: ICreateClientOpts = {
|
||||||
baseUrl: Config.defaultHomeserverUrl()!,
|
baseUrl: Config.defaultHomeserverUrl()!,
|
||||||
accessToken: access_token,
|
accessToken: access_token,
|
||||||
userId: user_id,
|
userId: user_id,
|
||||||
deviceId: device_id,
|
deviceId: device_id,
|
||||||
fallbackICEServerAllowed: fallbackICEServerAllowed,
|
fallbackICEServerAllowed: fallbackICEServerAllowed,
|
||||||
livekitServiceURL: Config.get().livekit!.livekit_service_url,
|
livekitServiceURL: Config.get().livekit?.livekit_service_url,
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -27,6 +27,7 @@ import styles from "./FullScreenView.module.css";
|
|||||||
import { TranslatedError } from "./TranslatedError";
|
import { TranslatedError } from "./TranslatedError";
|
||||||
import { Config } from "./config/Config";
|
import { Config } from "./config/Config";
|
||||||
import { RageshakeButton } from "./settings/RageshakeButton";
|
import { RageshakeButton } from "./settings/RageshakeButton";
|
||||||
|
import { useUrlParams } from "./UrlParams";
|
||||||
|
|
||||||
interface FullScreenViewProps {
|
interface FullScreenViewProps {
|
||||||
className?: string;
|
className?: string;
|
||||||
@@ -37,12 +38,11 @@ export const FullScreenView: FC<FullScreenViewProps> = ({
|
|||||||
className,
|
className,
|
||||||
children,
|
children,
|
||||||
}) => {
|
}) => {
|
||||||
|
const { hideHeader } = useUrlParams();
|
||||||
return (
|
return (
|
||||||
<div className={classNames(styles.page, className)}>
|
<div className={classNames(styles.page, className)}>
|
||||||
<Header>
|
<Header>
|
||||||
<LeftNav>
|
<LeftNav>{!hideHeader && <HeaderLogo />}</LeftNav>
|
||||||
<HeaderLogo />
|
|
||||||
</LeftNav>
|
|
||||||
<RightNav />
|
<RightNav />
|
||||||
</Header>
|
</Header>
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
@@ -58,6 +58,7 @@ interface ErrorViewProps {
|
|||||||
|
|
||||||
export const ErrorView: FC<ErrorViewProps> = ({ error }) => {
|
export const ErrorView: FC<ErrorViewProps> = ({ error }) => {
|
||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
|
const { confineToRoom } = useUrlParams();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -78,25 +79,26 @@ export const ErrorView: FC<ErrorViewProps> = ({ error }) => {
|
|||||||
: error.message}
|
: error.message}
|
||||||
</p>
|
</p>
|
||||||
<RageshakeButton description={`***Error View***: ${error.message}`} />
|
<RageshakeButton description={`***Error View***: ${error.message}`} />
|
||||||
{location.pathname === "/" ? (
|
{!confineToRoom &&
|
||||||
<Button
|
(location.pathname === "/" ? (
|
||||||
size="lg"
|
<Button
|
||||||
variant="default"
|
size="lg"
|
||||||
className={styles.homeLink}
|
variant="default"
|
||||||
onPress={onReload}
|
className={styles.homeLink}
|
||||||
>
|
onPress={onReload}
|
||||||
{t("return_home_button")}
|
>
|
||||||
</Button>
|
{t("return_home_button")}
|
||||||
) : (
|
</Button>
|
||||||
<LinkButton
|
) : (
|
||||||
size="lg"
|
<LinkButton
|
||||||
variant="default"
|
size="lg"
|
||||||
className={styles.homeLink}
|
variant="default"
|
||||||
to="/"
|
className={styles.homeLink}
|
||||||
>
|
to="/"
|
||||||
{t("return_home_button")}
|
>
|
||||||
</LinkButton>
|
{t("return_home_button")}
|
||||||
)}
|
</LinkButton>
|
||||||
|
))}
|
||||||
</FullScreenView>
|
</FullScreenView>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -109,6 +109,7 @@ limitations under the License.
|
|||||||
|
|
||||||
.participantsLine {
|
.participantsLine {
|
||||||
grid-area: participants;
|
grid-area: participants;
|
||||||
|
color: var(--cpd-color-text-secondary);
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: var(--cpd-space-1-5x);
|
gap: var(--cpd-space-1-5x);
|
||||||
|
|||||||
@@ -117,7 +117,7 @@ interface RoomHeaderInfoProps {
|
|||||||
name: string;
|
name: string;
|
||||||
avatarUrl: string | null;
|
avatarUrl: string | null;
|
||||||
encrypted: boolean;
|
encrypted: boolean;
|
||||||
participantCount: number;
|
participantCount: number | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const RoomHeaderInfo: FC<RoomHeaderInfoProps> = ({
|
export const RoomHeaderInfo: FC<RoomHeaderInfoProps> = ({
|
||||||
@@ -150,7 +150,7 @@ export const RoomHeaderInfo: FC<RoomHeaderInfoProps> = ({
|
|||||||
</Heading>
|
</Heading>
|
||||||
<EncryptionLock encrypted={encrypted} />
|
<EncryptionLock encrypted={encrypted} />
|
||||||
</div>
|
</div>
|
||||||
{participantCount > 0 && (
|
{(participantCount ?? 0) > 0 && (
|
||||||
<div className={styles.participantsLine}>
|
<div className={styles.participantsLine}>
|
||||||
<UserProfileIcon
|
<UserProfileIcon
|
||||||
width={20}
|
width={20}
|
||||||
@@ -158,7 +158,7 @@ export const RoomHeaderInfo: FC<RoomHeaderInfoProps> = ({
|
|||||||
aria-label={t("header_participants_label")}
|
aria-label={t("header_participants_label")}
|
||||||
/>
|
/>
|
||||||
<Text as="span" size="sm" weight="medium">
|
<Text as="span" size="sm" weight="medium">
|
||||||
{t("participant_count", { count: participantCount })}
|
{t("participant_count", { count: participantCount ?? 0 })}
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -76,7 +76,7 @@ export const Toast: FC<Props> = ({
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open && autoDismiss !== undefined) {
|
if (open && autoDismiss !== undefined) {
|
||||||
const timeout = setTimeout(onDismiss, autoDismiss);
|
const timeout = setTimeout(onDismiss, autoDismiss);
|
||||||
return () => clearTimeout(timeout);
|
return (): void => clearTimeout(timeout);
|
||||||
}
|
}
|
||||||
}, [open, autoDismiss, onDismiss]);
|
}, [open, autoDismiss, onDismiss]);
|
||||||
|
|
||||||
|
|||||||
@@ -16,10 +16,11 @@ limitations under the License.
|
|||||||
|
|
||||||
import { useMemo } from "react";
|
import { useMemo } from "react";
|
||||||
import { useLocation } from "react-router-dom";
|
import { useLocation } from "react-router-dom";
|
||||||
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
|
||||||
import { Config } from "./config/Config";
|
import { Config } from "./config/Config";
|
||||||
|
import { EncryptionSystem } from "./e2ee/sharedKeyManagement";
|
||||||
export const PASSWORD_STRING = "password=";
|
import { E2eeType } from "./e2ee/e2eeType";
|
||||||
|
|
||||||
interface RoomIdentifier {
|
interface RoomIdentifier {
|
||||||
roomAlias: string | null;
|
roomAlias: string | null;
|
||||||
@@ -328,3 +329,32 @@ export const useRoomIdentifier = (): RoomIdentifier => {
|
|||||||
[pathname, search, hash],
|
[pathname, search, hash],
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export function generateUrlSearchParams(
|
||||||
|
roomId: string,
|
||||||
|
encryptionSystem: EncryptionSystem,
|
||||||
|
viaServers?: string[],
|
||||||
|
): URLSearchParams {
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
// The password shouldn't need URL encoding here (we generate URL-safe ones) but encode
|
||||||
|
// it in case it came from another client that generated a non url-safe one
|
||||||
|
switch (encryptionSystem?.kind) {
|
||||||
|
case E2eeType.SHARED_KEY: {
|
||||||
|
const encodedPassword = encodeURIComponent(encryptionSystem.secret);
|
||||||
|
if (encodedPassword !== encryptionSystem.secret) {
|
||||||
|
logger.info(
|
||||||
|
"Encoded call password used non URL-safe chars: buggy client?",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
params.set("password", encodedPassword);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case E2eeType.PER_PARTICIPANT:
|
||||||
|
params.set("perParticipantE2EE", "true");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
params.set("roomId", roomId);
|
||||||
|
viaServers?.forEach((s) => params.set("viaServers", s));
|
||||||
|
|
||||||
|
return params;
|
||||||
|
}
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ import {
|
|||||||
UndecryptableToDeviceEventTracker,
|
UndecryptableToDeviceEventTracker,
|
||||||
QualitySurveyEventTracker,
|
QualitySurveyEventTracker,
|
||||||
CallDisconnectedEventTracker,
|
CallDisconnectedEventTracker,
|
||||||
|
CallConnectDurationTracker,
|
||||||
} from "./PosthogEvents";
|
} from "./PosthogEvents";
|
||||||
import { Config } from "../config/Config";
|
import { Config } from "../config/Config";
|
||||||
import { getUrlParams } from "../UrlParams";
|
import { getUrlParams } from "../UrlParams";
|
||||||
@@ -444,4 +445,5 @@ export class PosthogAnalytics {
|
|||||||
public eventUndecryptableToDevice = new UndecryptableToDeviceEventTracker();
|
public eventUndecryptableToDevice = new UndecryptableToDeviceEventTracker();
|
||||||
public eventQualitySurvey = new QualitySurveyEventTracker();
|
public eventQualitySurvey = new QualitySurveyEventTracker();
|
||||||
public eventCallDisconnected = new CallDisconnectedEventTracker();
|
public eventCallDisconnected = new CallDisconnectedEventTracker();
|
||||||
|
public eventCallConnectDuration = new CallConnectDurationTracker();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { DisconnectReason } from "livekit-client";
|
import { DisconnectReason } from "livekit-client";
|
||||||
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
IPosthogEvent,
|
IPosthogEvent,
|
||||||
@@ -201,3 +202,38 @@ export class CallDisconnectedEventTracker {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface CallConnectDuration extends IPosthogEvent {
|
||||||
|
eventName: "CallConnectDuration";
|
||||||
|
totalDuration: number;
|
||||||
|
websocketDuration: number;
|
||||||
|
peerConnectionDuration: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class CallConnectDurationTracker {
|
||||||
|
private connectStart = 0;
|
||||||
|
private websocketConnected = 0;
|
||||||
|
public cacheConnectStart(): void {
|
||||||
|
this.connectStart = Date.now();
|
||||||
|
}
|
||||||
|
public cacheWsConnect(): void {
|
||||||
|
this.websocketConnected = Date.now();
|
||||||
|
}
|
||||||
|
|
||||||
|
public track(options = { log: false }): void {
|
||||||
|
const now = Date.now();
|
||||||
|
const totalDuration = now - this.connectStart;
|
||||||
|
const websocketDuration = this.websocketConnected - this.connectStart;
|
||||||
|
const peerConnectionDuration = now - this.websocketConnected;
|
||||||
|
PosthogAnalytics.instance.trackEvent<CallConnectDuration>({
|
||||||
|
eventName: "CallConnectDuration",
|
||||||
|
totalDuration,
|
||||||
|
websocketDuration,
|
||||||
|
peerConnectionDuration,
|
||||||
|
});
|
||||||
|
if (options.log)
|
||||||
|
logger.log(
|
||||||
|
`Time to connect:\ntotal: ${totalDuration}ms\npeerConnection: ${websocketDuration}ms\nwebsocket: ${peerConnectionDuration}ms`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -82,7 +82,12 @@ export const LoginPage: FC = () => {
|
|||||||
},
|
},
|
||||||
[login, location, history, homeserver, setClient],
|
[login, location, history, homeserver, setClient],
|
||||||
);
|
);
|
||||||
|
// we need to limit the length of the homserver name to not cover the whole loginview input with the string.
|
||||||
|
let shortendHomeserverName = Config.defaultServerName()?.slice(0, 25);
|
||||||
|
shortendHomeserverName =
|
||||||
|
shortendHomeserverName?.length !== Config.defaultServerName()?.length
|
||||||
|
? shortendHomeserverName + "..."
|
||||||
|
: shortendHomeserverName;
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
@@ -102,7 +107,7 @@ export const LoginPage: FC = () => {
|
|||||||
autoCorrect="off"
|
autoCorrect="off"
|
||||||
autoCapitalize="none"
|
autoCapitalize="none"
|
||||||
prefix="@"
|
prefix="@"
|
||||||
suffix={`:${Config.defaultServerName()}`}
|
suffix={`:${shortendHomeserverName}`}
|
||||||
data-testid="login_username"
|
data-testid="login_username"
|
||||||
/>
|
/>
|
||||||
</FieldRow>
|
</FieldRow>
|
||||||
|
|||||||
@@ -16,7 +16,11 @@ limitations under the License.
|
|||||||
|
|
||||||
import { useCallback } from "react";
|
import { useCallback } from "react";
|
||||||
import { InteractiveAuth } from "matrix-js-sdk/src/interactive-auth";
|
import { InteractiveAuth } from "matrix-js-sdk/src/interactive-auth";
|
||||||
import { createClient, MatrixClient } from "matrix-js-sdk/src/matrix";
|
import {
|
||||||
|
createClient,
|
||||||
|
LoginResponse,
|
||||||
|
MatrixClient,
|
||||||
|
} from "matrix-js-sdk/src/matrix";
|
||||||
|
|
||||||
import { initClient } from "../matrix-utils";
|
import { initClient } from "../matrix-utils";
|
||||||
import { Session } from "../ClientContext";
|
import { Session } from "../ClientContext";
|
||||||
@@ -37,7 +41,7 @@ export function useInteractiveLogin(): (
|
|||||||
|
|
||||||
const interactiveAuth = new InteractiveAuth({
|
const interactiveAuth = new InteractiveAuth({
|
||||||
matrixClient: authClient,
|
matrixClient: authClient,
|
||||||
doRequest: () =>
|
doRequest: (): Promise<LoginResponse> =>
|
||||||
authClient.login("m.login.password", {
|
authClient.login("m.login.password", {
|
||||||
identifier: {
|
identifier: {
|
||||||
type: "m.id.user",
|
type: "m.id.user",
|
||||||
|
|||||||
@@ -16,11 +16,16 @@ limitations under the License.
|
|||||||
|
|
||||||
import { useState, useEffect, useCallback, useRef } from "react";
|
import { useState, useEffect, useCallback, useRef } from "react";
|
||||||
import { InteractiveAuth } from "matrix-js-sdk/src/interactive-auth";
|
import { InteractiveAuth } from "matrix-js-sdk/src/interactive-auth";
|
||||||
import { createClient, MatrixClient } from "matrix-js-sdk/src/matrix";
|
import {
|
||||||
|
createClient,
|
||||||
|
MatrixClient,
|
||||||
|
RegisterResponse,
|
||||||
|
} from "matrix-js-sdk/src/matrix";
|
||||||
|
|
||||||
import { initClient } from "../matrix-utils";
|
import { initClient } from "../matrix-utils";
|
||||||
import { Session } from "../ClientContext";
|
import { Session } from "../ClientContext";
|
||||||
import { Config } from "../config/Config";
|
import { Config } from "../config/Config";
|
||||||
|
import { widget } from "../widget";
|
||||||
|
|
||||||
export const useInteractiveRegistration = (): {
|
export const useInteractiveRegistration = (): {
|
||||||
privacyPolicyUrl?: string;
|
privacyPolicyUrl?: string;
|
||||||
@@ -48,6 +53,8 @@ export const useInteractiveRegistration = (): {
|
|||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
if (widget) return;
|
||||||
|
// An empty registerRequest is used to get the privacy policy and recaptcha key.
|
||||||
authClient.current!.registerRequest({}).catch((error) => {
|
authClient.current!.registerRequest({}).catch((error) => {
|
||||||
setPrivacyPolicyUrl(
|
setPrivacyPolicyUrl(
|
||||||
error.data?.params["m.login.terms"]?.policies?.privacy_policy?.en?.url,
|
error.data?.params["m.login.terms"]?.policies?.privacy_policy?.en?.url,
|
||||||
@@ -66,7 +73,7 @@ export const useInteractiveRegistration = (): {
|
|||||||
): Promise<[MatrixClient, Session]> => {
|
): Promise<[MatrixClient, Session]> => {
|
||||||
const interactiveAuth = new InteractiveAuth({
|
const interactiveAuth = new InteractiveAuth({
|
||||||
matrixClient: authClient.current!,
|
matrixClient: authClient.current!,
|
||||||
doRequest: (auth) =>
|
doRequest: (auth): Promise<RegisterResponse> =>
|
||||||
authClient.current!.registerRequest({
|
authClient.current!.registerRequest({
|
||||||
username,
|
username,
|
||||||
password,
|
password,
|
||||||
|
|||||||
@@ -55,16 +55,31 @@ export interface ConfigOptions {
|
|||||||
|
|
||||||
// Describes the LiveKit configuration to be used.
|
// Describes the LiveKit configuration to be used.
|
||||||
livekit?: {
|
livekit?: {
|
||||||
// The link to the service that returns a livekit url and token to use it
|
// The link to the service that returns a livekit url and token to use it.
|
||||||
|
// This is a fallback link in case the homeserver in use does not advertise
|
||||||
|
// a livekit service url in the client well-known.
|
||||||
|
// The well known needs to be formatted like so:
|
||||||
|
// {"type":"livekit", "livekit_service_url":"https://livekit.example.com"}
|
||||||
|
// and stored under the key: "livekit_focus"
|
||||||
livekit_service_url: string;
|
livekit_service_url: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Allow to join a group calls without audio and video.
|
* TEMPORARY experimental features.
|
||||||
* TEMPORARY: Is a feature that's not proved and experimental
|
|
||||||
*/
|
*/
|
||||||
features?: {
|
features?: {
|
||||||
feature_group_calls_without_video_and_audio: boolean;
|
/**
|
||||||
|
* Allow to join group calls without audio and video.
|
||||||
|
*/
|
||||||
|
feature_group_calls_without_video_and_audio?: boolean;
|
||||||
|
/**
|
||||||
|
* Send device-specific call session membership state events instead of
|
||||||
|
* legacy user-specific call membership state events.
|
||||||
|
* This setting has no effect when the user joins an active call with
|
||||||
|
* legacy state events. For compatibility, Element Call will always join
|
||||||
|
* active legacy calls with legacy state events.
|
||||||
|
*/
|
||||||
|
feature_use_device_session_member_events?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -15,12 +15,11 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { useEffect, useMemo } from "react";
|
import { useEffect, useMemo } from "react";
|
||||||
import { Room } from "matrix-js-sdk";
|
|
||||||
|
|
||||||
import { setLocalStorageItem, useLocalStorage } from "../useLocalStorage";
|
import { setLocalStorageItem, useLocalStorage } from "../useLocalStorage";
|
||||||
import { useClient } from "../ClientContext";
|
|
||||||
import { UrlParams, getUrlParams, useUrlParams } from "../UrlParams";
|
import { UrlParams, getUrlParams, useUrlParams } from "../UrlParams";
|
||||||
import { widget } from "../widget";
|
import { E2eeType } from "./e2eeType";
|
||||||
|
import { useClient } from "../ClientContext";
|
||||||
|
|
||||||
export function saveKeyForRoom(roomId: string, password: string): void {
|
export function saveKeyForRoom(roomId: string, password: string): void {
|
||||||
setLocalStorageItem(getRoomSharedKeyLocalStorageKey(roomId), password);
|
setLocalStorageItem(getRoomSharedKeyLocalStorageKey(roomId), password);
|
||||||
@@ -68,30 +67,37 @@ const useKeyFromUrl = (): [string, string] | [undefined, undefined] => {
|
|||||||
: [undefined, undefined];
|
: [undefined, undefined];
|
||||||
};
|
};
|
||||||
|
|
||||||
export const useRoomSharedKey = (roomId: string): string | undefined => {
|
export type Unencrypted = { kind: E2eeType.NONE };
|
||||||
|
export type SharedSecret = { kind: E2eeType.SHARED_KEY; secret: string };
|
||||||
|
export type PerParticipantE2EE = { kind: E2eeType.PER_PARTICIPANT };
|
||||||
|
export type EncryptionSystem = Unencrypted | SharedSecret | PerParticipantE2EE;
|
||||||
|
|
||||||
|
export function useRoomEncryptionSystem(roomId: string): EncryptionSystem {
|
||||||
|
const { client } = useClient();
|
||||||
|
|
||||||
// make sure we've extracted the key from the URL first
|
// make sure we've extracted the key from the URL first
|
||||||
// (and we still need to take the value it returns because
|
// (and we still need to take the value it returns because
|
||||||
// the effect won't run in time for it to save to localstorage in
|
// the effect won't run in time for it to save to localstorage in
|
||||||
// time for us to read it out again).
|
// time for us to read it out again).
|
||||||
const [urlRoomId, passwordFormUrl] = useKeyFromUrl();
|
const [urlRoomId, passwordFromUrl] = useKeyFromUrl();
|
||||||
|
|
||||||
const storedPassword = useInternalRoomSharedKey(roomId);
|
const storedPassword = useInternalRoomSharedKey(roomId);
|
||||||
|
const room = client?.getRoom(roomId);
|
||||||
if (storedPassword) return storedPassword;
|
const e2eeSystem = <EncryptionSystem>useMemo(() => {
|
||||||
if (urlRoomId === roomId) return passwordFormUrl;
|
if (!room) return { kind: E2eeType.NONE };
|
||||||
return undefined;
|
if (storedPassword)
|
||||||
};
|
return {
|
||||||
|
kind: E2eeType.SHARED_KEY,
|
||||||
export const useIsRoomE2EE = (roomId: string): boolean | null => {
|
secret: storedPassword,
|
||||||
const { client } = useClient();
|
};
|
||||||
const room = useMemo(() => client?.getRoom(roomId), [roomId, client]);
|
if (urlRoomId === roomId)
|
||||||
|
return {
|
||||||
return useMemo(() => !room || isRoomE2EE(room), [room]);
|
kind: E2eeType.SHARED_KEY,
|
||||||
};
|
secret: passwordFromUrl,
|
||||||
|
};
|
||||||
export function isRoomE2EE(room: Room): boolean {
|
if (room.hasEncryptionStateEvent()) {
|
||||||
// For now, rooms in widget mode are never considered encrypted.
|
return { kind: E2eeType.PER_PARTICIPANT };
|
||||||
// In the future, when widget mode gains encryption support, then perhaps we
|
}
|
||||||
// should inspect the e2eEnabled URL parameter here?
|
return { kind: E2eeType.NONE };
|
||||||
return widget === null && !room.getCanonicalAlias();
|
}, [passwordFromUrl, room, roomId, storedPassword, urlRoomId]);
|
||||||
|
return e2eeSystem;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ import styles from "./CallList.module.css";
|
|||||||
import { getAbsoluteRoomUrl, getRelativeRoomUrl } from "../matrix-utils";
|
import { getAbsoluteRoomUrl, getRelativeRoomUrl } from "../matrix-utils";
|
||||||
import { Body } from "../typography/Typography";
|
import { Body } from "../typography/Typography";
|
||||||
import { GroupCallRoom } from "./useGroupCallRooms";
|
import { GroupCallRoom } from "./useGroupCallRooms";
|
||||||
import { useRoomSharedKey } from "../e2ee/sharedKeyManagement";
|
import { useRoomEncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
|
|
||||||
interface CallListProps {
|
interface CallListProps {
|
||||||
rooms: GroupCallRoom[];
|
rooms: GroupCallRoom[];
|
||||||
@@ -66,16 +66,11 @@ interface CallTileProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const CallTile: FC<CallTileProps> = ({ name, avatarUrl, room }) => {
|
const CallTile: FC<CallTileProps> = ({ name, avatarUrl, room }) => {
|
||||||
const roomSharedKey = useRoomSharedKey(room.roomId);
|
const roomEncryptionSystem = useRoomEncryptionSystem(room.roomId);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={styles.callTile}>
|
<div className={styles.callTile}>
|
||||||
<Link
|
<Link
|
||||||
to={getRelativeRoomUrl(
|
to={getRelativeRoomUrl(room.roomId, roomEncryptionSystem, room.name)}
|
||||||
room.roomId,
|
|
||||||
room.name,
|
|
||||||
roomSharedKey ?? undefined,
|
|
||||||
)}
|
|
||||||
className={styles.callTileLink}
|
className={styles.callTileLink}
|
||||||
>
|
>
|
||||||
<Avatar id={room.roomId} name={name} size={Size.LG} src={avatarUrl} />
|
<Avatar id={room.roomId} name={name} size={Size.LG} src={avatarUrl} />
|
||||||
@@ -89,11 +84,8 @@ const CallTile: FC<CallTileProps> = ({ name, avatarUrl, room }) => {
|
|||||||
<CopyButton
|
<CopyButton
|
||||||
className={styles.copyButton}
|
className={styles.copyButton}
|
||||||
variant="icon"
|
variant="icon"
|
||||||
value={getAbsoluteRoomUrl(
|
// Todo add the viaServers to the created link
|
||||||
room.roomId,
|
value={getAbsoluteRoomUrl(room.roomId, roomEncryptionSystem, room.name)}
|
||||||
room.name,
|
|
||||||
roomSharedKey ?? undefined,
|
|
||||||
)}
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -78,12 +78,14 @@ export const RegisteredView: FC<Props> = ({ client }) => {
|
|||||||
roomName,
|
roomName,
|
||||||
E2eeType.SHARED_KEY,
|
E2eeType.SHARED_KEY,
|
||||||
);
|
);
|
||||||
|
if (!createRoomResult.password)
|
||||||
|
throw new Error("Failed to create room with shared secret");
|
||||||
|
|
||||||
history.push(
|
history.push(
|
||||||
getRelativeRoomUrl(
|
getRelativeRoomUrl(
|
||||||
createRoomResult.roomId,
|
createRoomResult.roomId,
|
||||||
|
{ kind: E2eeType.SHARED_KEY, secret: createRoomResult.password },
|
||||||
roomName,
|
roomName,
|
||||||
createRoomResult.password,
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,13 +116,15 @@ export const UnauthenticatedView: FC = () => {
|
|||||||
if (!setClient) {
|
if (!setClient) {
|
||||||
throw new Error("setClient is undefined");
|
throw new Error("setClient is undefined");
|
||||||
}
|
}
|
||||||
|
if (!createRoomResult.password)
|
||||||
|
throw new Error("Failed to create room with shared secret");
|
||||||
|
|
||||||
setClient({ client, session });
|
setClient({ client, session });
|
||||||
history.push(
|
history.push(
|
||||||
getRelativeRoomUrl(
|
getRelativeRoomUrl(
|
||||||
createRoomResult.roomId,
|
createRoomResult.roomId,
|
||||||
|
{ kind: E2eeType.SHARED_KEY, secret: createRoomResult.password },
|
||||||
roomName,
|
roomName,
|
||||||
createRoomResult.password,
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,20 +15,22 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { MatrixClient } from "matrix-js-sdk/src/client";
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
||||||
import { GroupCall } from "matrix-js-sdk/src/webrtc/groupCall";
|
import { Room, RoomEvent } from "matrix-js-sdk/src/models/room";
|
||||||
import { Room } from "matrix-js-sdk/src/models/room";
|
|
||||||
import { RoomMember } from "matrix-js-sdk/src/models/room-member";
|
import { RoomMember } from "matrix-js-sdk/src/models/room-member";
|
||||||
import { GroupCallEventHandlerEvent } from "matrix-js-sdk/src/webrtc/groupCallEventHandler";
|
|
||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
|
import { EventTimeline, EventType, JoinRule } from "matrix-js-sdk";
|
||||||
|
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
||||||
|
import { MatrixRTCSessionManagerEvents } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSessionManager";
|
||||||
|
import { KnownMembership } from "matrix-js-sdk/src/types";
|
||||||
|
|
||||||
import { getKeyForRoom, isRoomE2EE } from "../e2ee/sharedKeyManagement";
|
import { getKeyForRoom } from "../e2ee/sharedKeyManagement";
|
||||||
|
|
||||||
export interface GroupCallRoom {
|
export interface GroupCallRoom {
|
||||||
roomAlias?: string;
|
roomAlias?: string;
|
||||||
roomName: string;
|
roomName: string;
|
||||||
avatarUrl: string;
|
avatarUrl: string;
|
||||||
room: Room;
|
room: Room;
|
||||||
groupCall: GroupCall;
|
session: MatrixRTCSession;
|
||||||
participants: RoomMember[];
|
participants: RoomMember[];
|
||||||
}
|
}
|
||||||
const tsCache: { [index: string]: number } = {};
|
const tsCache: { [index: string]: number } = {};
|
||||||
@@ -46,7 +48,7 @@ function getLastTs(client: MatrixClient, r: Room): number {
|
|||||||
|
|
||||||
const myUserId = client.getUserId()!;
|
const myUserId = client.getUserId()!;
|
||||||
|
|
||||||
if (r.getMyMembership() !== "join") {
|
if (r.getMyMembership() !== KnownMembership.Join) {
|
||||||
const membershipEvent = r.currentState.getStateEvents(
|
const membershipEvent = r.currentState.getStateEvents(
|
||||||
"m.room.member",
|
"m.room.member",
|
||||||
myUserId,
|
myUserId,
|
||||||
@@ -80,38 +82,77 @@ function sortRooms(client: MatrixClient, rooms: Room[]): Room[] {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function roomIsJoinable(room: Room): boolean {
|
const roomIsJoinable = (room: Room): boolean => {
|
||||||
if (isRoomE2EE(room)) {
|
if (!room.hasEncryptionStateEvent() && !getKeyForRoom(room.roomId)) {
|
||||||
return Boolean(getKeyForRoom(room.roomId));
|
// if we have an non encrypted room (no encryption state event) we need a locally stored shared key.
|
||||||
} else {
|
// in case this key also does not exists we cannot join the room.
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
// otherwise we can always join rooms because we will automatically decide if we want to use perParticipant or password
|
||||||
|
switch (room.getJoinRule()) {
|
||||||
|
case JoinRule.Public:
|
||||||
|
return true;
|
||||||
|
case JoinRule.Knock:
|
||||||
|
switch (room.getMyMembership()) {
|
||||||
|
case KnownMembership.Join:
|
||||||
|
case KnownMembership.Knock:
|
||||||
|
return true;
|
||||||
|
case KnownMembership.Invite:
|
||||||
|
return (
|
||||||
|
room
|
||||||
|
.getLiveTimeline()
|
||||||
|
.getState(EventTimeline.FORWARDS)
|
||||||
|
?.getStateEvents(EventType.RoomMember, room.myUserId)
|
||||||
|
?.getPrevContent().membership === JoinRule.Knock
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
// TODO: check JoinRule.Restricted and return true if join condition is satisfied
|
||||||
|
default:
|
||||||
|
return room.getMyMembership() === KnownMembership.Join;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const roomHasCallMembershipEvents = (room: Room): boolean => {
|
||||||
|
switch (room.getMyMembership()) {
|
||||||
|
case KnownMembership.Join:
|
||||||
|
return !!room
|
||||||
|
.getLiveTimeline()
|
||||||
|
.getState(EventTimeline.FORWARDS)
|
||||||
|
?.events?.get(EventType.GroupCallMemberPrefix);
|
||||||
|
case KnownMembership.Knock:
|
||||||
|
// Assume that a room you've knocked on is able to hold calls
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export function useGroupCallRooms(client: MatrixClient): GroupCallRoom[] {
|
export function useGroupCallRooms(client: MatrixClient): GroupCallRoom[] {
|
||||||
const [rooms, setRooms] = useState<GroupCallRoom[]>([]);
|
const [rooms, setRooms] = useState<GroupCallRoom[]>([]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
function updateRooms(): void {
|
function updateRooms(): void {
|
||||||
if (!client.groupCallEventHandler) {
|
// We want to show all rooms that historically had a call and which we are (can become) part of.
|
||||||
return;
|
const rooms = client
|
||||||
}
|
.getRooms()
|
||||||
|
.filter(roomHasCallMembershipEvents)
|
||||||
const groupCalls = client.groupCallEventHandler.groupCalls.values();
|
|
||||||
const rooms = Array.from(groupCalls)
|
|
||||||
.map((groupCall) => groupCall.room)
|
|
||||||
.filter(roomIsJoinable);
|
.filter(roomIsJoinable);
|
||||||
const sortedRooms = sortRooms(client, rooms);
|
const sortedRooms = sortRooms(client, rooms);
|
||||||
const items = sortedRooms.map((room) => {
|
const items = sortedRooms.map((room) => {
|
||||||
const groupCall = client.getGroupCallForRoom(room.roomId)!;
|
const session = client.matrixRTC.getRoomSession(room);
|
||||||
|
session.memberships;
|
||||||
return {
|
return {
|
||||||
roomAlias: room.getCanonicalAlias() ?? undefined,
|
roomAlias: room.getCanonicalAlias() ?? undefined,
|
||||||
roomName: room.name,
|
roomName: room.name,
|
||||||
avatarUrl: room.getMxcAvatarUrl()!,
|
avatarUrl: room.getMxcAvatarUrl()!,
|
||||||
room,
|
room,
|
||||||
groupCall,
|
session,
|
||||||
participants: [...groupCall!.participants.keys()],
|
participants: session.memberships
|
||||||
|
.filter((m) => m.sender)
|
||||||
|
.map((m) => room.getMember(m.sender!))
|
||||||
|
.filter((m) => m) as RoomMember[],
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -120,15 +161,17 @@ export function useGroupCallRooms(client: MatrixClient): GroupCallRoom[] {
|
|||||||
|
|
||||||
updateRooms();
|
updateRooms();
|
||||||
|
|
||||||
client.on(GroupCallEventHandlerEvent.Incoming, updateRooms);
|
client.matrixRTC.on(
|
||||||
client.on(GroupCallEventHandlerEvent.Participants, updateRooms);
|
MatrixRTCSessionManagerEvents.SessionStarted,
|
||||||
|
updateRooms,
|
||||||
return () => {
|
);
|
||||||
client.removeListener(GroupCallEventHandlerEvent.Incoming, updateRooms);
|
client.on(RoomEvent.MyMembership, updateRooms);
|
||||||
client.removeListener(
|
return (): void => {
|
||||||
GroupCallEventHandlerEvent.Participants,
|
client.matrixRTC.off(
|
||||||
|
MatrixRTCSessionManagerEvents.SessionStarted,
|
||||||
updateRooms,
|
updateRooms,
|
||||||
);
|
);
|
||||||
|
client.off(RoomEvent.MyMembership, updateRooms);
|
||||||
};
|
};
|
||||||
}, [client]);
|
}, [client]);
|
||||||
|
|
||||||
|
|||||||
@@ -156,6 +156,12 @@ body {
|
|||||||
margin: 0;
|
margin: 0;
|
||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
-webkit-tap-highlight-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We use this to not render the page at all until we know the theme.*/
|
||||||
|
.no-theme {
|
||||||
|
opacity: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
html,
|
html,
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { BrowserTracing } from "@sentry/browser";
|
|
||||||
import i18n from "i18next";
|
import i18n from "i18next";
|
||||||
import { initReactI18next } from "react-i18next";
|
import { initReactI18next } from "react-i18next";
|
||||||
import LanguageDetector from "i18next-browser-languagedetector";
|
import LanguageDetector from "i18next-browser-languagedetector";
|
||||||
@@ -160,10 +159,7 @@ export class Initializer {
|
|||||||
dsn: Config.get().sentry?.DSN,
|
dsn: Config.get().sentry?.DSN,
|
||||||
environment: Config.get().sentry?.environment,
|
environment: Config.get().sentry?.environment,
|
||||||
integrations: [
|
integrations: [
|
||||||
new BrowserTracing({
|
Sentry.reactRouterV5BrowserTracingIntegration({ history }),
|
||||||
routingInstrumentation:
|
|
||||||
Sentry.reactRouterV5Instrumentation(history),
|
|
||||||
}),
|
|
||||||
],
|
],
|
||||||
tracesSampleRate: 1.0,
|
tracesSampleRate: 1.0,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright 2023 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 { Focus } from "matrix-js-sdk/src/matrixrtc/focus";
|
|
||||||
|
|
||||||
export interface LivekitFocus extends Focus {
|
|
||||||
type: "livekit";
|
|
||||||
livekit_service_url: string;
|
|
||||||
livekit_alias: string;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ function useObservableState<T>(
|
|||||||
// observable state doesn't run in SSR
|
// observable state doesn't run in SSR
|
||||||
if (typeof window === "undefined" || !observable) return;
|
if (typeof window === "undefined" || !observable) return;
|
||||||
const subscription = observable.subscribe(setState);
|
const subscription = observable.subscribe(setState);
|
||||||
return () => subscription.unsubscribe();
|
return (): void => subscription.unsubscribe();
|
||||||
}, [observable]);
|
}, [observable]);
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,9 +18,9 @@ import { IOpenIDToken, MatrixClient } from "matrix-js-sdk";
|
|||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
import { LivekitFocus } from "matrix-js-sdk/src/matrixrtc/LivekitFocus";
|
||||||
|
|
||||||
import { LivekitFocus } from "./LivekitFocus";
|
import { useActiveLivekitFocus } from "../room/useActiveFocus";
|
||||||
import { useActiveFocus } from "../room/useActiveFocus";
|
|
||||||
|
|
||||||
export interface SFUConfig {
|
export interface SFUConfig {
|
||||||
url: string;
|
url: string;
|
||||||
@@ -46,7 +46,7 @@ export function useOpenIDSFU(
|
|||||||
): SFUConfig | undefined {
|
): SFUConfig | undefined {
|
||||||
const [sfuConfig, setSFUConfig] = useState<SFUConfig | undefined>(undefined);
|
const [sfuConfig, setSFUConfig] = useState<SFUConfig | undefined>(undefined);
|
||||||
|
|
||||||
const activeFocus = useActiveFocus(rtcSession);
|
const activeFocus = useActiveLivekitFocus(rtcSession);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async (): Promise<void> => {
|
(async (): Promise<void> => {
|
||||||
|
|||||||
@@ -59,4 +59,5 @@ export const defaultLiveKitOptions: RoomOptions = {
|
|||||||
stopLocalTrackOnUnpublish: true,
|
stopLocalTrackOnUnpublish: true,
|
||||||
reconnectPolicy: new DefaultReconnectPolicy(),
|
reconnectPolicy: new DefaultReconnectPolicy(),
|
||||||
disconnectOnPageLeave: true,
|
disconnectOnPageLeave: true,
|
||||||
|
webAudioMix: false,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -27,6 +27,14 @@ import { logger } from "matrix-js-sdk/src/logger";
|
|||||||
import * as Sentry from "@sentry/react";
|
import * as Sentry from "@sentry/react";
|
||||||
|
|
||||||
import { SFUConfig, sfuConfigEquals } from "./openIDSFU";
|
import { SFUConfig, sfuConfigEquals } from "./openIDSFU";
|
||||||
|
import { PosthogAnalytics } from "../analytics/PosthogAnalytics";
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface Window {
|
||||||
|
peerConnectionTimeout?: number;
|
||||||
|
websocketTimeout?: number;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Additional values for states that a call can be in, beyond what livekit
|
* Additional values for states that a call can be in, beyond what livekit
|
||||||
@@ -124,7 +132,21 @@ async function connectAndPublish(
|
|||||||
micTrack: LocalTrack | undefined,
|
micTrack: LocalTrack | undefined,
|
||||||
screenshareTracks: MediaStreamTrack[],
|
screenshareTracks: MediaStreamTrack[],
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
await livekitRoom!.connect(sfuConfig!.url, sfuConfig!.jwt);
|
const tracker = PosthogAnalytics.instance.eventCallConnectDuration;
|
||||||
|
// Track call connect duration
|
||||||
|
tracker.cacheConnectStart();
|
||||||
|
livekitRoom.once(RoomEvent.SignalConnected, tracker.cacheWsConnect);
|
||||||
|
|
||||||
|
await livekitRoom!.connect(sfuConfig!.url, sfuConfig!.jwt, {
|
||||||
|
// Due to stability issues on Firefox we are testing the effect of different
|
||||||
|
// timeouts, and allow these values to be set through the console
|
||||||
|
peerConnectionTimeout: window.peerConnectionTimeout ?? 45000,
|
||||||
|
websocketTimeout: window.websocketTimeout ?? 45000,
|
||||||
|
});
|
||||||
|
|
||||||
|
// remove listener in case the connect promise rejects before `SignalConnected` is emitted.
|
||||||
|
livekitRoom.off(RoomEvent.SignalConnected, tracker.cacheWsConnect);
|
||||||
|
tracker.track({ log: true });
|
||||||
|
|
||||||
if (micTrack) {
|
if (micTrack) {
|
||||||
logger.info(`Publishing precreated mic track`);
|
logger.info(`Publishing precreated mic track`);
|
||||||
@@ -170,7 +192,7 @@ export function useECConnectionState(
|
|||||||
livekitRoom.on(RoomEvent.ConnectionStateChanged, onConnStateChanged);
|
livekitRoom.on(RoomEvent.ConnectionStateChanged, onConnStateChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
if (oldRoom)
|
if (oldRoom)
|
||||||
oldRoom.off(RoomEvent.ConnectionStateChanged, onConnStateChanged);
|
oldRoom.off(RoomEvent.ConnectionStateChanged, onConnStateChanged);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -41,11 +41,7 @@ import {
|
|||||||
} from "./useECConnectionState";
|
} from "./useECConnectionState";
|
||||||
import { MatrixKeyProvider } from "../e2ee/matrixKeyProvider";
|
import { MatrixKeyProvider } from "../e2ee/matrixKeyProvider";
|
||||||
import { E2eeType } from "../e2ee/e2eeType";
|
import { E2eeType } from "../e2ee/e2eeType";
|
||||||
|
import { EncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
export type E2EEConfig = {
|
|
||||||
mode: E2eeType;
|
|
||||||
sharedKey?: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
interface UseLivekitResult {
|
interface UseLivekitResult {
|
||||||
livekitRoom?: Room;
|
livekitRoom?: Room;
|
||||||
@@ -56,41 +52,35 @@ export function useLiveKit(
|
|||||||
rtcSession: MatrixRTCSession,
|
rtcSession: MatrixRTCSession,
|
||||||
muteStates: MuteStates,
|
muteStates: MuteStates,
|
||||||
sfuConfig: SFUConfig | undefined,
|
sfuConfig: SFUConfig | undefined,
|
||||||
e2eeConfig: E2EEConfig,
|
e2eeSystem: EncryptionSystem,
|
||||||
): UseLivekitResult {
|
): UseLivekitResult {
|
||||||
const e2eeOptions = useMemo((): E2EEOptions | undefined => {
|
const e2eeOptions = useMemo((): E2EEOptions | undefined => {
|
||||||
if (e2eeConfig.mode === E2eeType.NONE) return undefined;
|
if (e2eeSystem.kind === E2eeType.NONE) return undefined;
|
||||||
|
|
||||||
if (e2eeConfig.mode === E2eeType.PER_PARTICIPANT) {
|
if (e2eeSystem.kind === E2eeType.PER_PARTICIPANT) {
|
||||||
return {
|
return {
|
||||||
keyProvider: new MatrixKeyProvider(),
|
keyProvider: new MatrixKeyProvider(),
|
||||||
worker: new E2EEWorker(),
|
worker: new E2EEWorker(),
|
||||||
};
|
};
|
||||||
} else if (
|
} else if (e2eeSystem.kind === E2eeType.SHARED_KEY && e2eeSystem.secret) {
|
||||||
e2eeConfig.mode === E2eeType.SHARED_KEY &&
|
|
||||||
e2eeConfig.sharedKey
|
|
||||||
) {
|
|
||||||
return {
|
return {
|
||||||
keyProvider: new ExternalE2EEKeyProvider(),
|
keyProvider: new ExternalE2EEKeyProvider(),
|
||||||
worker: new E2EEWorker(),
|
worker: new E2EEWorker(),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}, [e2eeConfig]);
|
}, [e2eeSystem]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (e2eeConfig.mode === E2eeType.NONE || !e2eeOptions) return;
|
if (e2eeSystem.kind === E2eeType.NONE || !e2eeOptions) return;
|
||||||
|
|
||||||
if (e2eeConfig.mode === E2eeType.PER_PARTICIPANT) {
|
if (e2eeSystem.kind === E2eeType.PER_PARTICIPANT) {
|
||||||
(e2eeOptions.keyProvider as MatrixKeyProvider).setRTCSession(rtcSession);
|
(e2eeOptions.keyProvider as MatrixKeyProvider).setRTCSession(rtcSession);
|
||||||
} else if (
|
} else if (e2eeSystem.kind === E2eeType.SHARED_KEY && e2eeSystem.secret) {
|
||||||
e2eeConfig.mode === E2eeType.SHARED_KEY &&
|
|
||||||
e2eeConfig.sharedKey
|
|
||||||
) {
|
|
||||||
(e2eeOptions.keyProvider as ExternalE2EEKeyProvider).setKey(
|
(e2eeOptions.keyProvider as ExternalE2EEKeyProvider).setKey(
|
||||||
e2eeConfig.sharedKey,
|
e2eeSystem.secret,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}, [e2eeOptions, e2eeConfig, rtcSession]);
|
}, [e2eeOptions, e2eeSystem, rtcSession]);
|
||||||
|
|
||||||
const initialMuteStates = useRef<MuteStates>(muteStates);
|
const initialMuteStates = useRef<MuteStates>(muteStates);
|
||||||
const devices = useMediaDevices();
|
const devices = useMediaDevices();
|
||||||
@@ -131,9 +121,9 @@ export function useLiveKit(
|
|||||||
// useEffect() with an argument that references itself, if E2EE is enabled
|
// useEffect() with an argument that references itself, if E2EE is enabled
|
||||||
const room = useMemo(() => {
|
const room = useMemo(() => {
|
||||||
const r = new Room(roomOptions);
|
const r = new Room(roomOptions);
|
||||||
r.setE2EEEnabled(e2eeConfig.mode !== E2eeType.NONE);
|
r.setE2EEEnabled(e2eeSystem.kind !== E2eeType.NONE);
|
||||||
return r;
|
return r;
|
||||||
}, [roomOptions, e2eeConfig]);
|
}, [roomOptions, e2eeSystem]);
|
||||||
|
|
||||||
const connectionState = useECConnectionState(
|
const connectionState = useECConnectionState(
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -19,25 +19,25 @@ import { MemoryStore } from "matrix-js-sdk/src/store/memory";
|
|||||||
import { IndexedDBCryptoStore } from "matrix-js-sdk/src/crypto/store/indexeddb-crypto-store";
|
import { IndexedDBCryptoStore } from "matrix-js-sdk/src/crypto/store/indexeddb-crypto-store";
|
||||||
import { LocalStorageCryptoStore } from "matrix-js-sdk/src/crypto/store/localStorage-crypto-store";
|
import { LocalStorageCryptoStore } from "matrix-js-sdk/src/crypto/store/localStorage-crypto-store";
|
||||||
import { MemoryCryptoStore } from "matrix-js-sdk/src/crypto/store/memory-crypto-store";
|
import { MemoryCryptoStore } from "matrix-js-sdk/src/crypto/store/memory-crypto-store";
|
||||||
import { createClient, ICreateClientOpts } from "matrix-js-sdk/src/matrix";
|
import {
|
||||||
|
createClient,
|
||||||
|
ICreateClientOpts,
|
||||||
|
Preset,
|
||||||
|
Visibility,
|
||||||
|
} from "matrix-js-sdk/src/matrix";
|
||||||
import { ClientEvent } from "matrix-js-sdk/src/client";
|
import { ClientEvent } from "matrix-js-sdk/src/client";
|
||||||
import { Visibility, Preset } from "matrix-js-sdk/src/@types/partials";
|
|
||||||
import { ISyncStateData, SyncState } from "matrix-js-sdk/src/sync";
|
import { ISyncStateData, SyncState } from "matrix-js-sdk/src/sync";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
import {
|
|
||||||
GroupCallIntent,
|
|
||||||
GroupCallType,
|
|
||||||
} from "matrix-js-sdk/src/webrtc/groupCall";
|
|
||||||
import { secureRandomBase64Url } from "matrix-js-sdk/src/randomstring";
|
import { secureRandomBase64Url } from "matrix-js-sdk/src/randomstring";
|
||||||
|
|
||||||
import type { MatrixClient } from "matrix-js-sdk/src/client";
|
import type { MatrixClient } from "matrix-js-sdk/src/client";
|
||||||
import type { Room } from "matrix-js-sdk/src/models/room";
|
import type { Room } from "matrix-js-sdk/src/models/room";
|
||||||
import IndexedDBWorker from "./IndexedDBWorker?worker";
|
import IndexedDBWorker from "./IndexedDBWorker?worker";
|
||||||
import { getUrlParams, PASSWORD_STRING } from "./UrlParams";
|
import { generateUrlSearchParams, getUrlParams } from "./UrlParams";
|
||||||
import { loadOlm } from "./olm";
|
import { loadOlm } from "./olm";
|
||||||
import { Config } from "./config/Config";
|
import { Config } from "./config/Config";
|
||||||
import { E2eeType } from "./e2ee/e2eeType";
|
import { E2eeType } from "./e2ee/e2eeType";
|
||||||
import { saveKeyForRoom } from "./e2ee/sharedKeyManagement";
|
import { EncryptionSystem, saveKeyForRoom } from "./e2ee/sharedKeyManagement";
|
||||||
|
|
||||||
export const fallbackICEServerAllowed =
|
export const fallbackICEServerAllowed =
|
||||||
import.meta.env.VITE_FALLBACK_STUN_ALLOWED === "true";
|
import.meta.env.VITE_FALLBACK_STUN_ALLOWED === "true";
|
||||||
@@ -55,7 +55,13 @@ const SYNC_STORE_NAME = "element-call-sync";
|
|||||||
// (It's a good opportunity to make the database names consistent.)
|
// (It's a good opportunity to make the database names consistent.)
|
||||||
const CRYPTO_STORE_NAME = "element-call-crypto";
|
const CRYPTO_STORE_NAME = "element-call-crypto";
|
||||||
|
|
||||||
function waitForSync(client: MatrixClient): Promise<void> {
|
async function waitForSync(client: MatrixClient): Promise<void> {
|
||||||
|
// If there is a saved sync, the client will fire an additional sync event
|
||||||
|
// for restoring it before it runs the first network sync.
|
||||||
|
// However, the sync we want to wait for is the network sync,
|
||||||
|
// as the saved sync may be missing some state.
|
||||||
|
// Thus, don't resolve on the first sync when we know it's for the saved sync.
|
||||||
|
let waitForSavedSync = !!(await client.store.getSavedSyncToken());
|
||||||
return new Promise<void>((resolve, reject) => {
|
return new Promise<void>((resolve, reject) => {
|
||||||
const onSync = (
|
const onSync = (
|
||||||
state: SyncState,
|
state: SyncState,
|
||||||
@@ -63,8 +69,12 @@ function waitForSync(client: MatrixClient): Promise<void> {
|
|||||||
data?: ISyncStateData,
|
data?: ISyncStateData,
|
||||||
): void => {
|
): void => {
|
||||||
if (state === "PREPARED") {
|
if (state === "PREPARED") {
|
||||||
client.removeListener(ClientEvent.Sync, onSync);
|
if (waitForSavedSync) {
|
||||||
resolve();
|
waitForSavedSync = false;
|
||||||
|
} else {
|
||||||
|
client.removeListener(ClientEvent.Sync, onSync);
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
} else if (state === "ERROR") {
|
} else if (state === "ERROR") {
|
||||||
client.removeListener(ClientEvent.Sync, onSync);
|
client.removeListener(ClientEvent.Sync, onSync);
|
||||||
reject(data?.error);
|
reject(data?.error);
|
||||||
@@ -190,8 +200,14 @@ export async function initClient(
|
|||||||
|
|
||||||
await client.initCrypto();
|
await client.initCrypto();
|
||||||
client.setGlobalErrorOnUnknownDevices(false);
|
client.setGlobalErrorOnUnknownDevices(false);
|
||||||
await client.startClient();
|
// Once startClient is called, syncs are run asynchronously.
|
||||||
await waitForSync(client);
|
// Also, sync completion is communicated only via events.
|
||||||
|
// So, apply the event listener *before* starting the client.
|
||||||
|
// Otherwise, a sync may complete before the listener gets applied,
|
||||||
|
// and we will miss it.
|
||||||
|
const syncPromise = waitForSync(client);
|
||||||
|
await client.startClient({ clientWellKnownPollPeriod: 60 * 10 });
|
||||||
|
await syncPromise;
|
||||||
|
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
@@ -338,16 +354,6 @@ export async function createRoom(
|
|||||||
|
|
||||||
const result = await createPromise;
|
const result = await createPromise;
|
||||||
|
|
||||||
logger.log(`Creating group call in ${result.room_id}`);
|
|
||||||
|
|
||||||
await client.createGroupCall(
|
|
||||||
result.room_id,
|
|
||||||
GroupCallType.Video,
|
|
||||||
false,
|
|
||||||
GroupCallIntent.Room,
|
|
||||||
true,
|
|
||||||
);
|
|
||||||
|
|
||||||
let password;
|
let password;
|
||||||
if (e2ee == E2eeType.SHARED_KEY) {
|
if (e2ee == E2eeType.SHARED_KEY) {
|
||||||
password = secureRandomBase64Url(16);
|
password = secureRandomBase64Url(16);
|
||||||
@@ -365,39 +371,35 @@ export async function createRoom(
|
|||||||
* Returns an absolute URL to that will load Element Call with the given room
|
* Returns an absolute URL to that will load Element Call with the given room
|
||||||
* @param roomId ID of the room
|
* @param roomId ID of the room
|
||||||
* @param roomName Name of the room
|
* @param roomName Name of the room
|
||||||
* @param password e2e key for the room
|
* @param encryptionSystem what encryption (or EncryptionSystem.Unencrypted) the room uses
|
||||||
*/
|
*/
|
||||||
export function getAbsoluteRoomUrl(
|
export function getAbsoluteRoomUrl(
|
||||||
roomId: string,
|
roomId: string,
|
||||||
|
encryptionSystem: EncryptionSystem,
|
||||||
roomName?: string,
|
roomName?: string,
|
||||||
password?: string,
|
viaServers?: string[],
|
||||||
): string {
|
): string {
|
||||||
return `${window.location.protocol}//${
|
return `${window.location.protocol}//${
|
||||||
window.location.host
|
window.location.host
|
||||||
}${getRelativeRoomUrl(roomId, roomName, password)}`;
|
}${getRelativeRoomUrl(roomId, encryptionSystem, roomName, viaServers)}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a relative URL to that will load Element Call with the given room
|
* Returns a relative URL to that will load Element Call with the given room
|
||||||
* @param roomId ID of the room
|
* @param roomId ID of the room
|
||||||
* @param roomName Name of the room
|
* @param roomName Name of the room
|
||||||
* @param password e2e key for the room
|
* @param encryptionSystem what encryption (or EncryptionSystem.Unencrypted) the room uses
|
||||||
*/
|
*/
|
||||||
export function getRelativeRoomUrl(
|
export function getRelativeRoomUrl(
|
||||||
roomId: string,
|
roomId: string,
|
||||||
|
encryptionSystem: EncryptionSystem,
|
||||||
roomName?: string,
|
roomName?: string,
|
||||||
password?: string,
|
viaServers?: string[],
|
||||||
): string {
|
): string {
|
||||||
// The password shouldn't need URL encoding here (we generate URL-safe ones) but encode
|
const roomPart = roomName
|
||||||
// it in case it came from another client that generated a non url-safe one
|
? "/" + roomAliasLocalpartFromRoomName(roomName)
|
||||||
const encodedPassword = password ? encodeURIComponent(password) : undefined;
|
: "";
|
||||||
if (password && encodedPassword !== password) {
|
return `/room/#${roomPart}?${generateUrlSearchParams(roomId, encryptionSystem, viaServers).toString()}`;
|
||||||
logger.info("Encoded call password used non URL-safe chars: buggy client?");
|
|
||||||
}
|
|
||||||
|
|
||||||
return `/room/#${
|
|
||||||
roomName ? "/" + roomAliasLocalpartFromRoomName(roomName) : ""
|
|
||||||
}?roomId=${roomId}${password ? "&" + PASSWORD_STRING + encodedPassword : ""}`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getAvatarUrl(
|
export function getAvatarUrl(
|
||||||
|
|||||||
@@ -40,7 +40,6 @@ import {
|
|||||||
SummaryStatsReport,
|
SummaryStatsReport,
|
||||||
CallFeedReport,
|
CallFeedReport,
|
||||||
} from "matrix-js-sdk/src/webrtc/stats/statsReport";
|
} from "matrix-js-sdk/src/webrtc/stats/statsReport";
|
||||||
import { setSpan } from "@opentelemetry/api/build/esm/trace/context-utils";
|
|
||||||
|
|
||||||
import { ElementCallOpenTelemetry } from "./otel";
|
import { ElementCallOpenTelemetry } from "./otel";
|
||||||
import { ObjectFlattener } from "./ObjectFlattener";
|
import { ObjectFlattener } from "./ObjectFlattener";
|
||||||
@@ -446,7 +445,7 @@ export class OTelGroupCallMembership {
|
|||||||
const type = OTelStatsReportType.SummaryReport;
|
const type = OTelStatsReportType.SummaryReport;
|
||||||
const data = ObjectFlattener.flattenSummaryStatsReportObject(statsReport);
|
const data = ObjectFlattener.flattenSummaryStatsReportObject(statsReport);
|
||||||
if (this.statsReportSpan.span === undefined && this.callMembershipSpan) {
|
if (this.statsReportSpan.span === undefined && this.callMembershipSpan) {
|
||||||
const ctx = setSpan(
|
const ctx = opentelemetry.trace.setSpan(
|
||||||
opentelemetry.context.active(),
|
opentelemetry.context.active(),
|
||||||
this.callMembershipSpan,
|
this.callMembershipSpan,
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -82,7 +82,7 @@ export function useProfile(client: MatrixClient | undefined): UseProfile {
|
|||||||
user?.on(UserEvent.AvatarUrl, onChangeUser);
|
user?.on(UserEvent.AvatarUrl, onChangeUser);
|
||||||
}
|
}
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
if (user) {
|
if (user) {
|
||||||
user.removeListener(UserEvent.DisplayName, onChangeUser);
|
user.removeListener(UserEvent.DisplayName, onChangeUser);
|
||||||
user.removeListener(UserEvent.AvatarUrl, onChangeUser);
|
user.removeListener(UserEvent.AvatarUrl, onChangeUser);
|
||||||
@@ -118,7 +118,7 @@ export function useProfile(client: MatrixClient | undefined): UseProfile {
|
|||||||
displayName,
|
displayName,
|
||||||
avatarUrl: removeAvatar
|
avatarUrl: removeAvatar
|
||||||
? undefined
|
? undefined
|
||||||
: mxcAvatarUrl ?? prev.avatarUrl,
|
: (mxcAvatarUrl ?? prev.avatarUrl),
|
||||||
loading: false,
|
loading: false,
|
||||||
success: true,
|
success: true,
|
||||||
}));
|
}));
|
||||||
|
|||||||
@@ -21,13 +21,14 @@ import PopOutIcon from "@vector-im/compound-design-tokens/icons/pop-out.svg?reac
|
|||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
|
||||||
import { Modal } from "../Modal";
|
import { Modal } from "../Modal";
|
||||||
import { useIsRoomE2EE, useRoomSharedKey } from "../e2ee/sharedKeyManagement";
|
import { useRoomEncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
import { getAbsoluteRoomUrl } from "../matrix-utils";
|
import { getAbsoluteRoomUrl } from "../matrix-utils";
|
||||||
import styles from "./AppSelectionModal.module.css";
|
import styles from "./AppSelectionModal.module.css";
|
||||||
import { editFragmentQuery } from "../UrlParams";
|
import { editFragmentQuery } from "../UrlParams";
|
||||||
|
import { E2eeType } from "../e2ee/e2eeType";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
roomId: string | null;
|
roomId: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const AppSelectionModal: FC<Props> = ({ roomId }) => {
|
export const AppSelectionModal: FC<Props> = ({ roomId }) => {
|
||||||
@@ -42,10 +43,9 @@ export const AppSelectionModal: FC<Props> = ({ roomId }) => {
|
|||||||
},
|
},
|
||||||
[setOpen],
|
[setOpen],
|
||||||
);
|
);
|
||||||
|
const e2eeSystem = useRoomEncryptionSystem(roomId);
|
||||||
|
|
||||||
const roomSharedKey = useRoomSharedKey(roomId ?? "");
|
if (e2eeSystem.kind === E2eeType.NONE) {
|
||||||
const roomIsEncrypted = useIsRoomE2EE(roomId ?? "");
|
|
||||||
if (roomIsEncrypted && roomSharedKey === undefined) {
|
|
||||||
logger.error(
|
logger.error(
|
||||||
"Generating app redirect URL for encrypted room but don't have key available!",
|
"Generating app redirect URL for encrypted room but don't have key available!",
|
||||||
);
|
);
|
||||||
@@ -60,7 +60,7 @@ export const AppSelectionModal: FC<Props> = ({ roomId }) => {
|
|||||||
const url = new URL(
|
const url = new URL(
|
||||||
roomId === null
|
roomId === null
|
||||||
? window.location.href
|
? window.location.href
|
||||||
: getAbsoluteRoomUrl(roomId, undefined, roomSharedKey ?? undefined),
|
: getAbsoluteRoomUrl(roomId, e2eeSystem),
|
||||||
);
|
);
|
||||||
// Edit the URL to prevent the app selection prompt from appearing a second
|
// Edit the URL to prevent the app selection prompt from appearing a second
|
||||||
// time within the app, and to keep the user confined to the current room
|
// time within the app, and to keep the user confined to the current room
|
||||||
@@ -73,7 +73,7 @@ export const AppSelectionModal: FC<Props> = ({ roomId }) => {
|
|||||||
const result = new URL("io.element.call:/");
|
const result = new URL("io.element.call:/");
|
||||||
result.searchParams.set("url", url.toString());
|
result.searchParams.set("url", url.toString());
|
||||||
return result.toString();
|
return result.toString();
|
||||||
}, [roomId, roomSharedKey]);
|
}, [e2eeSystem, roomId]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
|
|||||||
@@ -14,22 +14,25 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ReactNode, useCallback } from "react";
|
import { useCallback } from "react";
|
||||||
import { MatrixClient } from "matrix-js-sdk/src/client";
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
|
||||||
import { MatrixError } from "matrix-js-sdk";
|
import { MatrixError } from "matrix-js-sdk";
|
||||||
import { useHistory } from "react-router-dom";
|
import { useHistory } from "react-router-dom";
|
||||||
import { Heading, Link, Text } from "@vector-im/compound-web";
|
import { Heading, Link, Text } from "@vector-im/compound-web";
|
||||||
|
|
||||||
import { useLoadGroupCall } from "./useLoadGroupCall";
|
import {
|
||||||
|
useLoadGroupCall,
|
||||||
|
GroupCallStatus,
|
||||||
|
CallTerminatedMessage,
|
||||||
|
} from "./useLoadGroupCall";
|
||||||
import { ErrorView, FullScreenView } from "../FullScreenView";
|
import { ErrorView, FullScreenView } from "../FullScreenView";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
client: MatrixClient;
|
client: MatrixClient;
|
||||||
roomIdOrAlias: string;
|
roomIdOrAlias: string;
|
||||||
viaServers: string[];
|
viaServers: string[];
|
||||||
children: (rtcSession: MatrixRTCSession) => ReactNode;
|
children: (groupCallState: GroupCallStatus) => JSX.Element;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function GroupCallLoader({
|
export function GroupCallLoader({
|
||||||
@@ -51,20 +54,22 @@ export function GroupCallLoader({
|
|||||||
);
|
);
|
||||||
|
|
||||||
switch (groupCallState.kind) {
|
switch (groupCallState.kind) {
|
||||||
|
case "loaded":
|
||||||
|
case "waitForInvite":
|
||||||
|
case "canKnock":
|
||||||
|
return children(groupCallState);
|
||||||
case "loading":
|
case "loading":
|
||||||
return (
|
return (
|
||||||
<FullScreenView>
|
<FullScreenView>
|
||||||
<h1>{t("common.loading")}</h1>
|
<h1>{t("common.loading")}</h1>
|
||||||
</FullScreenView>
|
</FullScreenView>
|
||||||
);
|
);
|
||||||
case "loaded":
|
|
||||||
return <>{children(groupCallState.rtcSession)}</>;
|
|
||||||
case "failed":
|
case "failed":
|
||||||
if ((groupCallState.error as MatrixError).errcode === "M_NOT_FOUND") {
|
if ((groupCallState.error as MatrixError).errcode === "M_NOT_FOUND") {
|
||||||
return (
|
return (
|
||||||
<FullScreenView>
|
<FullScreenView>
|
||||||
<Heading>{t("group_call_loader_failed_heading")}</Heading>
|
<Heading>{t("group_call_loader.failed_heading")}</Heading>
|
||||||
<Text>{t("group_call_loader_failed_text")}</Text>
|
<Text>{t("group_call_loader.failed_text")}</Text>
|
||||||
{/* XXX: A 'create it for me' button would be the obvious UX here. Two screens already have
|
{/* XXX: A 'create it for me' button would be the obvious UX here. Two screens already have
|
||||||
dupes of this flow, let's make a common component and put it here. */}
|
dupes of this flow, let's make a common component and put it here. */}
|
||||||
<Link href="/" onClick={onHomeClick}>
|
<Link href="/" onClick={onHomeClick}>
|
||||||
@@ -72,6 +77,22 @@ export function GroupCallLoader({
|
|||||||
</Link>
|
</Link>
|
||||||
</FullScreenView>
|
</FullScreenView>
|
||||||
);
|
);
|
||||||
|
} else if (groupCallState.error instanceof CallTerminatedMessage) {
|
||||||
|
return (
|
||||||
|
<FullScreenView>
|
||||||
|
<Heading>{groupCallState.error.message}</Heading>
|
||||||
|
<Text>{groupCallState.error.messageBody}</Text>
|
||||||
|
{groupCallState.error.reason && (
|
||||||
|
<>
|
||||||
|
{t("group_call_loader.reason")}:
|
||||||
|
<Text size="sm">"{groupCallState.error.reason}"</Text>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<Link href="/" onClick={onHomeClick}>
|
||||||
|
{t("common.home")}
|
||||||
|
</Link>
|
||||||
|
</FullScreenView>
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
return <ErrorView error={groupCallState.error} />;
|
return <ErrorView error={groupCallState.error} />;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,7 +17,10 @@ limitations under the License.
|
|||||||
import { FC, useCallback, useEffect, useMemo, useRef, useState } from "react";
|
import { FC, useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
import { useHistory } from "react-router-dom";
|
import { useHistory } from "react-router-dom";
|
||||||
import { MatrixClient } from "matrix-js-sdk/src/client";
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
||||||
import { Room, isE2EESupported } from "livekit-client";
|
import {
|
||||||
|
Room,
|
||||||
|
isE2EESupported as isE2EESupportedBrowser,
|
||||||
|
} from "livekit-client";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
||||||
import { JoinRule } from "matrix-js-sdk/src/matrix";
|
import { JoinRule } from "matrix-js-sdk/src/matrix";
|
||||||
@@ -26,7 +29,7 @@ import { useTranslation } from "react-i18next";
|
|||||||
|
|
||||||
import type { IWidgetApiRequest } from "matrix-widget-api";
|
import type { IWidgetApiRequest } from "matrix-widget-api";
|
||||||
import { widget, ElementWidgetActions, JoinCallData } from "../widget";
|
import { widget, ElementWidgetActions, JoinCallData } from "../widget";
|
||||||
import { ErrorView, FullScreenView } from "../FullScreenView";
|
import { FullScreenView } from "../FullScreenView";
|
||||||
import { LobbyView } from "./LobbyView";
|
import { LobbyView } from "./LobbyView";
|
||||||
import { MatrixInfo } from "./VideoPreview";
|
import { MatrixInfo } from "./VideoPreview";
|
||||||
import { CallEndedView } from "./CallEndedView";
|
import { CallEndedView } from "./CallEndedView";
|
||||||
@@ -34,17 +37,16 @@ import { PosthogAnalytics } from "../analytics/PosthogAnalytics";
|
|||||||
import { useProfile } from "../profile/useProfile";
|
import { useProfile } from "../profile/useProfile";
|
||||||
import { findDeviceByName } from "../media-utils";
|
import { findDeviceByName } from "../media-utils";
|
||||||
import { ActiveCall } from "./InCallView";
|
import { ActiveCall } from "./InCallView";
|
||||||
import { MuteStates, useMuteStates } from "./MuteStates";
|
import { MUTE_PARTICIPANT_COUNT, MuteStates } from "./MuteStates";
|
||||||
import { useMediaDevices, MediaDevices } from "../livekit/MediaDevicesContext";
|
import { useMediaDevices, MediaDevices } from "../livekit/MediaDevicesContext";
|
||||||
import { useMatrixRTCSessionMemberships } from "../useMatrixRTCSessionMemberships";
|
import { useMatrixRTCSessionMemberships } from "../useMatrixRTCSessionMemberships";
|
||||||
import { enterRTCSession, leaveRTCSession } from "../rtcSessionHelpers";
|
import { enterRTCSession, leaveRTCSession } from "../rtcSessionHelpers";
|
||||||
import { useMatrixRTCSessionJoinState } from "../useMatrixRTCSessionJoinState";
|
import { useMatrixRTCSessionJoinState } from "../useMatrixRTCSessionJoinState";
|
||||||
import { useIsRoomE2EE, useRoomSharedKey } from "../e2ee/sharedKeyManagement";
|
import { useRoomEncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
import { useRoomAvatar } from "./useRoomAvatar";
|
import { useRoomAvatar } from "./useRoomAvatar";
|
||||||
import { useRoomName } from "./useRoomName";
|
import { useRoomName } from "./useRoomName";
|
||||||
import { useJoinRule } from "./useJoinRule";
|
import { useJoinRule } from "./useJoinRule";
|
||||||
import { InviteModal } from "./InviteModal";
|
import { InviteModal } from "./InviteModal";
|
||||||
import { E2EEConfig } from "../livekit/useLiveKit";
|
|
||||||
import { useUrlParams } from "../UrlParams";
|
import { useUrlParams } from "../UrlParams";
|
||||||
import { E2eeType } from "../e2ee/e2eeType";
|
import { E2eeType } from "../e2ee/e2eeType";
|
||||||
|
|
||||||
@@ -62,6 +64,7 @@ interface Props {
|
|||||||
skipLobby: boolean;
|
skipLobby: boolean;
|
||||||
hideHeader: boolean;
|
hideHeader: boolean;
|
||||||
rtcSession: MatrixRTCSession;
|
rtcSession: MatrixRTCSession;
|
||||||
|
muteStates: MuteStates;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const GroupCallView: FC<Props> = ({
|
export const GroupCallView: FC<Props> = ({
|
||||||
@@ -72,13 +75,21 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
skipLobby,
|
skipLobby,
|
||||||
hideHeader,
|
hideHeader,
|
||||||
rtcSession,
|
rtcSession,
|
||||||
|
muteStates,
|
||||||
}) => {
|
}) => {
|
||||||
const memberships = useMatrixRTCSessionMemberships(rtcSession);
|
const memberships = useMatrixRTCSessionMemberships(rtcSession);
|
||||||
const isJoined = useMatrixRTCSessionJoinState(rtcSession);
|
const isJoined = useMatrixRTCSessionJoinState(rtcSession);
|
||||||
|
|
||||||
|
// This should use `useEffectEvent` (only available in experimental versions)
|
||||||
|
useEffect(() => {
|
||||||
|
if (memberships.length >= MUTE_PARTICIPANT_COUNT)
|
||||||
|
muteStates.audio.setEnabled?.(false);
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
window.rtcSession = rtcSession;
|
window.rtcSession = rtcSession;
|
||||||
return () => {
|
return (): void => {
|
||||||
delete window.rtcSession;
|
delete window.rtcSession;
|
||||||
};
|
};
|
||||||
}, [rtcSession]);
|
}, [rtcSession]);
|
||||||
@@ -86,10 +97,8 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
const { displayName, avatarUrl } = useProfile(client);
|
const { displayName, avatarUrl } = useProfile(client);
|
||||||
const roomName = useRoomName(rtcSession.room);
|
const roomName = useRoomName(rtcSession.room);
|
||||||
const roomAvatar = useRoomAvatar(rtcSession.room);
|
const roomAvatar = useRoomAvatar(rtcSession.room);
|
||||||
const e2eeSharedKey = useRoomSharedKey(rtcSession.room.roomId);
|
|
||||||
const { perParticipantE2EE, returnToLobby } = useUrlParams();
|
const { perParticipantE2EE, returnToLobby } = useUrlParams();
|
||||||
const roomEncrypted =
|
const e2eeSystem = useRoomEncryptionSystem(rtcSession.room.roomId);
|
||||||
useIsRoomE2EE(rtcSession.room.roomId) || perParticipantE2EE;
|
|
||||||
|
|
||||||
const matrixInfo = useMemo((): MatrixInfo => {
|
const matrixInfo = useMemo((): MatrixInfo => {
|
||||||
return {
|
return {
|
||||||
@@ -100,16 +109,16 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
roomName,
|
roomName,
|
||||||
roomAlias: rtcSession.room.getCanonicalAlias(),
|
roomAlias: rtcSession.room.getCanonicalAlias(),
|
||||||
roomAvatar,
|
roomAvatar,
|
||||||
roomEncrypted,
|
e2eeSystem,
|
||||||
};
|
};
|
||||||
}, [
|
}, [
|
||||||
|
client,
|
||||||
displayName,
|
displayName,
|
||||||
avatarUrl,
|
avatarUrl,
|
||||||
rtcSession,
|
rtcSession.room,
|
||||||
roomName,
|
roomName,
|
||||||
roomAvatar,
|
roomAvatar,
|
||||||
roomEncrypted,
|
e2eeSystem,
|
||||||
client,
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// Count each member only once, regardless of how many devices they use
|
// Count each member only once, regardless of how many devices they use
|
||||||
@@ -122,20 +131,9 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
const latestDevices = useRef<MediaDevices>();
|
const latestDevices = useRef<MediaDevices>();
|
||||||
latestDevices.current = deviceContext;
|
latestDevices.current = deviceContext;
|
||||||
|
|
||||||
const muteStates = useMuteStates(memberships.length);
|
|
||||||
const latestMuteStates = useRef<MuteStates>();
|
const latestMuteStates = useRef<MuteStates>();
|
||||||
latestMuteStates.current = muteStates;
|
latestMuteStates.current = muteStates;
|
||||||
|
|
||||||
const e2eeConfig = useMemo((): E2EEConfig => {
|
|
||||||
if (perParticipantE2EE) {
|
|
||||||
return { mode: E2eeType.PER_PARTICIPANT };
|
|
||||||
} else if (e2eeSharedKey) {
|
|
||||||
return { mode: E2eeType.SHARED_KEY, sharedKey: e2eeSharedKey };
|
|
||||||
} else {
|
|
||||||
return { mode: E2eeType.NONE };
|
|
||||||
}
|
|
||||||
}, [perParticipantE2EE, e2eeSharedKey]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const defaultDeviceSetup = async (
|
const defaultDeviceSetup = async (
|
||||||
requestedDeviceData: JoinCallData,
|
requestedDeviceData: JoinCallData,
|
||||||
@@ -193,11 +191,11 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
ev: CustomEvent<IWidgetApiRequest>,
|
ev: CustomEvent<IWidgetApiRequest>,
|
||||||
): Promise<void> => {
|
): Promise<void> => {
|
||||||
defaultDeviceSetup(ev.detail.data as unknown as JoinCallData);
|
defaultDeviceSetup(ev.detail.data as unknown as JoinCallData);
|
||||||
enterRTCSession(rtcSession, perParticipantE2EE);
|
await enterRTCSession(rtcSession, perParticipantE2EE);
|
||||||
await widget!.api.transport.reply(ev.detail, {});
|
await widget!.api.transport.reply(ev.detail, {});
|
||||||
};
|
};
|
||||||
widget.lazyActions.on(ElementWidgetActions.JoinCall, onJoin);
|
widget.lazyActions.on(ElementWidgetActions.JoinCall, onJoin);
|
||||||
return () => {
|
return (): void => {
|
||||||
widget!.lazyActions.off(ElementWidgetActions.JoinCall, onJoin);
|
widget!.lazyActions.off(ElementWidgetActions.JoinCall, onJoin);
|
||||||
};
|
};
|
||||||
} else if (widget && !preload && skipLobby) {
|
} else if (widget && !preload && skipLobby) {
|
||||||
@@ -252,7 +250,7 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
await leaveRTCSession(rtcSession);
|
await leaveRTCSession(rtcSession);
|
||||||
};
|
};
|
||||||
widget.lazyActions.once(ElementWidgetActions.HangupCall, onHangup);
|
widget.lazyActions.once(ElementWidgetActions.HangupCall, onHangup);
|
||||||
return () => {
|
return (): void => {
|
||||||
widget!.lazyActions.off(ElementWidgetActions.HangupCall, onHangup);
|
widget!.lazyActions.off(ElementWidgetActions.HangupCall, onHangup);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -288,17 +286,8 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
if (roomEncrypted && !perParticipantE2EE && !e2eeSharedKey) {
|
if (!isE2EESupportedBrowser() && e2eeSystem.kind !== E2eeType.NONE) {
|
||||||
return (
|
// If we have a encryption system but the browser does not support it.
|
||||||
<ErrorView
|
|
||||||
error={
|
|
||||||
new Error(
|
|
||||||
"No E2EE key provided: please make sure the URL you're using to join this call has been retrieved using the in-app button.",
|
|
||||||
)
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
} else if (!isE2EESupported() && roomEncrypted) {
|
|
||||||
return (
|
return (
|
||||||
<FullScreenView>
|
<FullScreenView>
|
||||||
<Heading>{t("browser_media_e2ee_unsupported_heading")}</Heading>
|
<Heading>{t("browser_media_e2ee_unsupported_heading")}</Heading>
|
||||||
@@ -324,7 +313,7 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
client={client}
|
client={client}
|
||||||
matrixInfo={matrixInfo}
|
matrixInfo={matrixInfo}
|
||||||
muteStates={muteStates}
|
muteStates={muteStates}
|
||||||
onEnter={(): void => enterRTCSession(rtcSession, perParticipantE2EE)}
|
onEnter={() => void enterRTCSession(rtcSession, perParticipantE2EE)}
|
||||||
confineToRoom={confineToRoom}
|
confineToRoom={confineToRoom}
|
||||||
hideHeader={hideHeader}
|
hideHeader={hideHeader}
|
||||||
participantCount={participantCount}
|
participantCount={participantCount}
|
||||||
@@ -345,7 +334,7 @@ export const GroupCallView: FC<Props> = ({
|
|||||||
onLeave={onLeave}
|
onLeave={onLeave}
|
||||||
hideHeader={hideHeader}
|
hideHeader={hideHeader}
|
||||||
muteStates={muteStates}
|
muteStates={muteStates}
|
||||||
e2eeConfig={e2eeConfig}
|
e2eeSystem={e2eeSystem}
|
||||||
//otelGroupCallMembership={otelGroupCallMembership}
|
//otelGroupCallMembership={otelGroupCallMembership}
|
||||||
onShareClick={onShareClick}
|
onShareClick={onShareClick}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -85,13 +85,13 @@ limitations under the License.
|
|||||||
|
|
||||||
@media (min-height: 400px) {
|
@media (min-height: 400px) {
|
||||||
.footer {
|
.footer {
|
||||||
padding-block: var(--cpd-space-10x);
|
padding-block: var(--cpd-space-8x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-height: 800px) {
|
@media (min-height: 800px) {
|
||||||
.footer {
|
.footer {
|
||||||
padding-block: var(--cpd-space-15x);
|
padding-block: var(--cpd-space-10x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -63,7 +63,7 @@ import { OTelGroupCallMembership } from "../otel/OTelGroupCallMembership";
|
|||||||
import { SettingsModal, defaultSettingsTab } from "../settings/SettingsModal";
|
import { SettingsModal, defaultSettingsTab } from "../settings/SettingsModal";
|
||||||
import { useRageshakeRequestModal } from "../settings/submit-rageshake";
|
import { useRageshakeRequestModal } from "../settings/submit-rageshake";
|
||||||
import { RageshakeRequestModal } from "./RageshakeRequestModal";
|
import { RageshakeRequestModal } from "./RageshakeRequestModal";
|
||||||
import { E2EEConfig, useLiveKit } from "../livekit/useLiveKit";
|
import { useLiveKit } from "../livekit/useLiveKit";
|
||||||
import { useFullscreen } from "./useFullscreen";
|
import { useFullscreen } from "./useFullscreen";
|
||||||
import { useLayoutStates } from "../video-grid/Layout";
|
import { useLayoutStates } from "../video-grid/Layout";
|
||||||
import { useWakeLock } from "../useWakeLock";
|
import { useWakeLock } from "../useWakeLock";
|
||||||
@@ -76,13 +76,15 @@ import { ECConnectionState } from "../livekit/useECConnectionState";
|
|||||||
import { useOpenIDSFU } from "../livekit/openIDSFU";
|
import { useOpenIDSFU } from "../livekit/openIDSFU";
|
||||||
import { useCallViewModel } from "../state/CallViewModel";
|
import { useCallViewModel } from "../state/CallViewModel";
|
||||||
import { subscribe } from "../state/subscribe";
|
import { subscribe } from "../state/subscribe";
|
||||||
|
import { EncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
|
import { E2eeType } from "../e2ee/e2eeType";
|
||||||
|
|
||||||
const canScreenshare = "getDisplayMedia" in (navigator.mediaDevices ?? {});
|
const canScreenshare = "getDisplayMedia" in (navigator.mediaDevices ?? {});
|
||||||
const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
|
const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
|
||||||
|
|
||||||
export interface ActiveCallProps
|
export interface ActiveCallProps
|
||||||
extends Omit<InCallViewProps, "livekitRoom" | "connState"> {
|
extends Omit<InCallViewProps, "livekitRoom" | "connState"> {
|
||||||
e2eeConfig: E2EEConfig;
|
e2eeSystem: EncryptionSystem;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ActiveCall: FC<ActiveCallProps> = (props) => {
|
export const ActiveCall: FC<ActiveCallProps> = (props) => {
|
||||||
@@ -91,11 +93,11 @@ export const ActiveCall: FC<ActiveCallProps> = (props) => {
|
|||||||
props.rtcSession,
|
props.rtcSession,
|
||||||
props.muteStates,
|
props.muteStates,
|
||||||
sfuConfig,
|
sfuConfig,
|
||||||
props.e2eeConfig,
|
props.e2eeSystem,
|
||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
return () => {
|
return (): void => {
|
||||||
livekitRoom?.disconnect();
|
livekitRoom?.disconnect();
|
||||||
};
|
};
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
@@ -218,7 +220,7 @@ export const InCallView: FC<InCallViewProps> = subscribe(
|
|||||||
onSpotlightLayout,
|
onSpotlightLayout,
|
||||||
);
|
);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
widget!.lazyActions.off(
|
widget!.lazyActions.off(
|
||||||
ElementWidgetActions.TileLayout,
|
ElementWidgetActions.TileLayout,
|
||||||
onTileLayout,
|
onTileLayout,
|
||||||
@@ -238,7 +240,7 @@ export const InCallView: FC<InCallViewProps> = subscribe(
|
|||||||
const vm = useCallViewModel(
|
const vm = useCallViewModel(
|
||||||
rtcSession.room,
|
rtcSession.room,
|
||||||
livekitRoom,
|
livekitRoom,
|
||||||
matrixInfo.roomEncrypted,
|
matrixInfo.e2eeSystem.kind !== E2eeType.NONE,
|
||||||
connState,
|
connState,
|
||||||
);
|
);
|
||||||
const items = useStateObservable(vm.tiles);
|
const items = useStateObservable(vm.tiles);
|
||||||
@@ -252,7 +254,7 @@ export const InCallView: FC<InCallViewProps> = subscribe(
|
|||||||
() =>
|
() =>
|
||||||
fullscreenItem ??
|
fullscreenItem ??
|
||||||
(noControls
|
(noControls
|
||||||
? items.find((item) => item.isSpeaker) ?? items.at(0) ?? null
|
? (items.find((item) => item.isSpeaker) ?? items.at(0) ?? null)
|
||||||
: null),
|
: null),
|
||||||
[fullscreenItem, noControls, items],
|
[fullscreenItem, noControls, items],
|
||||||
);
|
);
|
||||||
@@ -432,7 +434,7 @@ export const InCallView: FC<InCallViewProps> = subscribe(
|
|||||||
id={matrixInfo.roomId}
|
id={matrixInfo.roomId}
|
||||||
name={matrixInfo.roomName}
|
name={matrixInfo.roomName}
|
||||||
avatarUrl={matrixInfo.roomAvatar}
|
avatarUrl={matrixInfo.roomAvatar}
|
||||||
encrypted={matrixInfo.roomEncrypted}
|
encrypted={matrixInfo.e2eeSystem.kind !== E2eeType.NONE}
|
||||||
participantCount={participantCount}
|
participantCount={participantCount}
|
||||||
/>
|
/>
|
||||||
</LeftNav>
|
</LeftNav>
|
||||||
|
|||||||
@@ -25,8 +25,8 @@ import useClipboard from "react-use-clipboard";
|
|||||||
import { Modal } from "../Modal";
|
import { Modal } from "../Modal";
|
||||||
import { getAbsoluteRoomUrl } from "../matrix-utils";
|
import { getAbsoluteRoomUrl } from "../matrix-utils";
|
||||||
import styles from "./InviteModal.module.css";
|
import styles from "./InviteModal.module.css";
|
||||||
import { useRoomSharedKey } from "../e2ee/sharedKeyManagement";
|
|
||||||
import { Toast } from "../Toast";
|
import { Toast } from "../Toast";
|
||||||
|
import { useRoomEncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
room: Room;
|
room: Room;
|
||||||
@@ -36,11 +36,11 @@ interface Props {
|
|||||||
|
|
||||||
export const InviteModal: FC<Props> = ({ room, open, onDismiss }) => {
|
export const InviteModal: FC<Props> = ({ room, open, onDismiss }) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const roomSharedKey = useRoomSharedKey(room.roomId);
|
const e2eeSystem = useRoomEncryptionSystem(room.roomId);
|
||||||
|
|
||||||
const url = useMemo(
|
const url = useMemo(
|
||||||
() =>
|
() => getAbsoluteRoomUrl(room.roomId, e2eeSystem, room.name),
|
||||||
getAbsoluteRoomUrl(room.roomId, room.name, roomSharedKey ?? undefined),
|
[e2eeSystem, room.name, room.roomId],
|
||||||
[room, roomSharedKey],
|
|
||||||
);
|
);
|
||||||
const [, setCopied] = useClipboard(url);
|
const [, setCopied] = useClipboard(url);
|
||||||
const [toastOpen, setToastOpen] = useState(false);
|
const [toastOpen, setToastOpen] = useState(false);
|
||||||
|
|||||||
@@ -25,6 +25,18 @@ limitations under the License.
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.wait {
|
||||||
|
color: var(--cpd-color-text-primary) !important;
|
||||||
|
background-color: var(--cpd-color-bg-canvas-default) !important;
|
||||||
|
/* relative colors are only supported on chromium based browsers */
|
||||||
|
background-color: rgb(
|
||||||
|
from var(--cpd-color-bg-canvas-default) r g b / 0.5
|
||||||
|
) !important;
|
||||||
|
}
|
||||||
|
.wait > svg {
|
||||||
|
color: var(--cpd-color-theme-primary) !important;
|
||||||
|
}
|
||||||
|
|
||||||
@media (max-width: 500px) {
|
@media (max-width: 500px) {
|
||||||
.join {
|
.join {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|||||||
@@ -21,8 +21,8 @@ import { Button, Link } from "@vector-im/compound-web";
|
|||||||
import classNames from "classnames";
|
import classNames from "classnames";
|
||||||
import { useHistory } from "react-router-dom";
|
import { useHistory } from "react-router-dom";
|
||||||
|
|
||||||
import styles from "./LobbyView.module.css";
|
|
||||||
import inCallStyles from "./InCallView.module.css";
|
import inCallStyles from "./InCallView.module.css";
|
||||||
|
import styles from "./LobbyView.module.css";
|
||||||
import { Header, LeftNav, RightNav, RoomHeaderInfo } from "../Header";
|
import { Header, LeftNav, RightNav, RoomHeaderInfo } from "../Header";
|
||||||
import { useLocationNavigation } from "../useLocationNavigation";
|
import { useLocationNavigation } from "../useLocationNavigation";
|
||||||
import { MatrixInfo, VideoPreview } from "./VideoPreview";
|
import { MatrixInfo, VideoPreview } from "./VideoPreview";
|
||||||
@@ -36,16 +36,19 @@ import {
|
|||||||
} from "../button/Button";
|
} from "../button/Button";
|
||||||
import { SettingsModal, defaultSettingsTab } from "../settings/SettingsModal";
|
import { SettingsModal, defaultSettingsTab } from "../settings/SettingsModal";
|
||||||
import { useMediaQuery } from "../useMediaQuery";
|
import { useMediaQuery } from "../useMediaQuery";
|
||||||
|
import { E2eeType } from "../e2ee/e2eeType";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
client: MatrixClient;
|
client: MatrixClient;
|
||||||
matrixInfo: MatrixInfo;
|
matrixInfo: MatrixInfo;
|
||||||
muteStates: MuteStates;
|
muteStates: MuteStates;
|
||||||
onEnter: () => void;
|
onEnter: () => void;
|
||||||
|
enterLabel?: JSX.Element | string;
|
||||||
confineToRoom: boolean;
|
confineToRoom: boolean;
|
||||||
hideHeader: boolean;
|
hideHeader: boolean;
|
||||||
participantCount: number;
|
participantCount: number | null;
|
||||||
onShareClick: (() => void) | null;
|
onShareClick: (() => void) | null;
|
||||||
|
waitingForInvite?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const LobbyView: FC<Props> = ({
|
export const LobbyView: FC<Props> = ({
|
||||||
@@ -53,10 +56,12 @@ export const LobbyView: FC<Props> = ({
|
|||||||
matrixInfo,
|
matrixInfo,
|
||||||
muteStates,
|
muteStates,
|
||||||
onEnter,
|
onEnter,
|
||||||
|
enterLabel,
|
||||||
confineToRoom,
|
confineToRoom,
|
||||||
hideHeader,
|
hideHeader,
|
||||||
participantCount,
|
participantCount,
|
||||||
onShareClick,
|
onShareClick,
|
||||||
|
waitingForInvite,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
useLocationNavigation();
|
useLocationNavigation();
|
||||||
@@ -104,7 +109,7 @@ export const LobbyView: FC<Props> = ({
|
|||||||
id={matrixInfo.roomId}
|
id={matrixInfo.roomId}
|
||||||
name={matrixInfo.roomName}
|
name={matrixInfo.roomName}
|
||||||
avatarUrl={matrixInfo.roomAvatar}
|
avatarUrl={matrixInfo.roomAvatar}
|
||||||
encrypted={matrixInfo.roomEncrypted}
|
encrypted={matrixInfo.e2eeSystem.kind !== E2eeType.NONE}
|
||||||
participantCount={participantCount}
|
participantCount={participantCount}
|
||||||
/>
|
/>
|
||||||
</LeftNav>
|
</LeftNav>
|
||||||
@@ -116,12 +121,16 @@ export const LobbyView: FC<Props> = ({
|
|||||||
<div className={styles.content}>
|
<div className={styles.content}>
|
||||||
<VideoPreview matrixInfo={matrixInfo} muteStates={muteStates}>
|
<VideoPreview matrixInfo={matrixInfo} muteStates={muteStates}>
|
||||||
<Button
|
<Button
|
||||||
className={styles.join}
|
className={classNames(styles.join, {
|
||||||
size="lg"
|
[styles.wait]: waitingForInvite,
|
||||||
onClick={onEnter}
|
})}
|
||||||
|
size={waitingForInvite ? "sm" : "lg"}
|
||||||
|
onClick={() => {
|
||||||
|
if (!waitingForInvite) onEnter();
|
||||||
|
}}
|
||||||
data-testid="lobby_joinCall"
|
data-testid="lobby_joinCall"
|
||||||
>
|
>
|
||||||
{t("lobby.join_button")}
|
{enterLabel ?? t("lobby.join_button")}
|
||||||
</Button>
|
</Button>
|
||||||
</VideoPreview>
|
</VideoPreview>
|
||||||
{!recentsButtonInFooter && recentsButton}
|
{!recentsButtonInFooter && recentsButton}
|
||||||
|
|||||||
@@ -20,10 +20,10 @@ import { MediaDevice, useMediaDevices } from "../livekit/MediaDevicesContext";
|
|||||||
import { useReactiveState } from "../useReactiveState";
|
import { useReactiveState } from "../useReactiveState";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If there already is this many participants in the call, we automatically mute
|
* If there already are this many participants in the call, we automatically mute
|
||||||
* the user
|
* the user.
|
||||||
*/
|
*/
|
||||||
const MUTE_PARTICIPANT_COUNT = 8;
|
export const MUTE_PARTICIPANT_COUNT = 8;
|
||||||
|
|
||||||
interface DeviceAvailable {
|
interface DeviceAvailable {
|
||||||
enabled: boolean;
|
enabled: boolean;
|
||||||
@@ -51,26 +51,27 @@ function useMuteState(
|
|||||||
device: MediaDevice,
|
device: MediaDevice,
|
||||||
enabledByDefault: () => boolean,
|
enabledByDefault: () => boolean,
|
||||||
): MuteState {
|
): MuteState {
|
||||||
const [enabled, setEnabled] = useReactiveState<boolean>(
|
const [enabled, setEnabled] = useReactiveState<boolean | undefined>(
|
||||||
(prev) => device.available.length > 0 && (prev ?? enabledByDefault()),
|
(prev) =>
|
||||||
|
device.available.length > 0 ? (prev ?? enabledByDefault()) : undefined,
|
||||||
[device],
|
[device],
|
||||||
);
|
);
|
||||||
return useMemo(
|
return useMemo(
|
||||||
() =>
|
() =>
|
||||||
device.available.length === 0
|
device.available.length === 0
|
||||||
? deviceUnavailable
|
? deviceUnavailable
|
||||||
: { enabled, setEnabled },
|
: {
|
||||||
|
enabled: enabled ?? false,
|
||||||
|
setEnabled: setEnabled as Dispatch<SetStateAction<boolean>>,
|
||||||
|
},
|
||||||
[device, enabled, setEnabled],
|
[device, enabled, setEnabled],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useMuteStates(participantCount: number): MuteStates {
|
export function useMuteStates(): MuteStates {
|
||||||
const devices = useMediaDevices();
|
const devices = useMediaDevices();
|
||||||
|
|
||||||
const audio = useMuteState(
|
const audio = useMuteState(devices.audioInput, () => true);
|
||||||
devices.audioInput,
|
|
||||||
() => participantCount <= MUTE_PARTICIPANT_COUNT,
|
|
||||||
);
|
|
||||||
const video = useMuteState(devices.videoInput, () => true);
|
const video = useMuteState(devices.videoInput, () => true);
|
||||||
|
|
||||||
return useMemo(() => ({ audio, video }), [audio, video]);
|
return useMemo(() => ({ audio, video }), [audio, video]);
|
||||||
|
|||||||
@@ -15,8 +15,9 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { FC, useEffect, useState, useCallback, ReactNode } from "react";
|
import { FC, useEffect, useState, useCallback, ReactNode } from "react";
|
||||||
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import CheckIcon from "@vector-im/compound-design-tokens/icons/check.svg?react";
|
||||||
|
|
||||||
import { useClientLegacy } from "../ClientContext";
|
import { useClientLegacy } from "../ClientContext";
|
||||||
import { ErrorView, LoadingView } from "../FullScreenView";
|
import { ErrorView, LoadingView } from "../FullScreenView";
|
||||||
@@ -30,6 +31,11 @@ import { HomePage } from "../home/HomePage";
|
|||||||
import { platform } from "../Platform";
|
import { platform } from "../Platform";
|
||||||
import { AppSelectionModal } from "./AppSelectionModal";
|
import { AppSelectionModal } from "./AppSelectionModal";
|
||||||
import { widget } from "../widget";
|
import { widget } from "../widget";
|
||||||
|
import { GroupCallStatus } from "./useLoadGroupCall";
|
||||||
|
import { LobbyView } from "./LobbyView";
|
||||||
|
import { E2eeType } from "../e2ee/e2eeType";
|
||||||
|
import { useProfile } from "../profile/useProfile";
|
||||||
|
import { useMuteStates } from "./MuteStates";
|
||||||
|
|
||||||
export const RoomPage: FC = () => {
|
export const RoomPage: FC = () => {
|
||||||
const {
|
const {
|
||||||
@@ -40,7 +46,7 @@ export const RoomPage: FC = () => {
|
|||||||
displayName,
|
displayName,
|
||||||
skipLobby,
|
skipLobby,
|
||||||
} = useUrlParams();
|
} = useUrlParams();
|
||||||
|
const { t } = useTranslation();
|
||||||
const { roomAlias, roomId, viaServers } = useRoomIdentifier();
|
const { roomAlias, roomId, viaServers } = useRoomIdentifier();
|
||||||
|
|
||||||
const roomIdOrAlias = roomId ?? roomAlias;
|
const roomIdOrAlias = roomId ?? roomAlias;
|
||||||
@@ -48,17 +54,14 @@ export const RoomPage: FC = () => {
|
|||||||
logger.error("No room specified");
|
logger.error("No room specified");
|
||||||
}
|
}
|
||||||
|
|
||||||
const [optInAnalytics, setOptInAnalytics] = useOptInAnalytics();
|
|
||||||
const { registerPasswordlessUser } = useRegisterPasswordlessUser();
|
const { registerPasswordlessUser } = useRegisterPasswordlessUser();
|
||||||
const [isRegistering, setIsRegistering] = useState(false);
|
const [isRegistering, setIsRegistering] = useState(false);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
// During the beta, opt into analytics by default
|
|
||||||
if (optInAnalytics === null && setOptInAnalytics) setOptInAnalytics(true);
|
|
||||||
}, [optInAnalytics, setOptInAnalytics]);
|
|
||||||
|
|
||||||
const { loading, authenticated, client, error, passwordlessUser } =
|
const { loading, authenticated, client, error, passwordlessUser } =
|
||||||
useClientLegacy();
|
useClientLegacy();
|
||||||
|
const { avatarUrl, displayName: userDisplayName } = useProfile(client);
|
||||||
|
|
||||||
|
const muteStates = useMuteStates();
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// If we've finished loading, are not already authed and we've been given a display name as
|
// If we've finished loading, are not already authed and we've been given a display name as
|
||||||
@@ -77,19 +80,87 @@ export const RoomPage: FC = () => {
|
|||||||
registerPasswordlessUser,
|
registerPasswordlessUser,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
const [optInAnalytics, setOptInAnalytics] = useOptInAnalytics();
|
||||||
|
useEffect(() => {
|
||||||
|
// During the beta, opt into analytics by default
|
||||||
|
if (optInAnalytics === null && setOptInAnalytics) setOptInAnalytics(true);
|
||||||
|
}, [optInAnalytics, setOptInAnalytics]);
|
||||||
|
|
||||||
const groupCallView = useCallback(
|
const groupCallView = useCallback(
|
||||||
(rtcSession: MatrixRTCSession) => (
|
(groupCallState: GroupCallStatus): JSX.Element => {
|
||||||
<GroupCallView
|
switch (groupCallState.kind) {
|
||||||
client={client!}
|
case "loaded":
|
||||||
rtcSession={rtcSession}
|
return (
|
||||||
isPasswordlessUser={passwordlessUser}
|
<GroupCallView
|
||||||
confineToRoom={confineToRoom}
|
client={client!}
|
||||||
preload={preload}
|
rtcSession={groupCallState.rtcSession}
|
||||||
skipLobby={skipLobby}
|
isPasswordlessUser={passwordlessUser}
|
||||||
hideHeader={hideHeader}
|
confineToRoom={confineToRoom}
|
||||||
/>
|
preload={preload}
|
||||||
),
|
skipLobby={skipLobby}
|
||||||
[client, passwordlessUser, confineToRoom, preload, hideHeader, skipLobby],
|
hideHeader={hideHeader}
|
||||||
|
muteStates={muteStates}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
case "waitForInvite":
|
||||||
|
case "canKnock": {
|
||||||
|
const knock =
|
||||||
|
groupCallState.kind === "canKnock" ? groupCallState.knock : null;
|
||||||
|
const label: string | JSX.Element =
|
||||||
|
groupCallState.kind === "canKnock" ? (
|
||||||
|
t("lobby.ask_to_join")
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{t("lobby.waiting_for_invite")}
|
||||||
|
<CheckIcon />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
return (
|
||||||
|
<LobbyView
|
||||||
|
client={client!}
|
||||||
|
matrixInfo={{
|
||||||
|
userId: client!.getUserId() ?? "",
|
||||||
|
displayName: userDisplayName ?? "",
|
||||||
|
avatarUrl: avatarUrl ?? "",
|
||||||
|
roomAlias: null,
|
||||||
|
roomId: groupCallState.roomSummary.room_id,
|
||||||
|
roomName: groupCallState.roomSummary.name ?? "",
|
||||||
|
roomAvatar: groupCallState.roomSummary.avatar_url ?? null,
|
||||||
|
e2eeSystem: {
|
||||||
|
kind: groupCallState.roomSummary[
|
||||||
|
"im.nheko.summary.encryption"
|
||||||
|
]
|
||||||
|
? E2eeType.PER_PARTICIPANT
|
||||||
|
: E2eeType.NONE,
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
onEnter={(): void => knock?.()}
|
||||||
|
enterLabel={label}
|
||||||
|
waitingForInvite={groupCallState.kind === "waitForInvite"}
|
||||||
|
confineToRoom={confineToRoom}
|
||||||
|
hideHeader={hideHeader}
|
||||||
|
participantCount={null}
|
||||||
|
muteStates={muteStates}
|
||||||
|
onShareClick={null}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return <> </>;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[
|
||||||
|
client,
|
||||||
|
passwordlessUser,
|
||||||
|
confineToRoom,
|
||||||
|
preload,
|
||||||
|
skipLobby,
|
||||||
|
hideHeader,
|
||||||
|
muteStates,
|
||||||
|
t,
|
||||||
|
userDisplayName,
|
||||||
|
avatarUrl,
|
||||||
|
],
|
||||||
);
|
);
|
||||||
|
|
||||||
let content: ReactNode;
|
let content: ReactNode;
|
||||||
@@ -118,9 +189,9 @@ export const RoomPage: FC = () => {
|
|||||||
<>
|
<>
|
||||||
{content}
|
{content}
|
||||||
{/* On Android and iOS, show a prompt to launch the mobile app. */}
|
{/* On Android and iOS, show a prompt to launch the mobile app. */}
|
||||||
{appPrompt && (platform === "android" || platform === "ios") && (
|
{appPrompt &&
|
||||||
<AppSelectionModal roomId={roomId} />
|
(platform === "android" || platform === "ios") &&
|
||||||
)}
|
roomId && <AppSelectionModal roomId={roomId} />}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { useEffect, useMemo, useRef, FC, ReactNode } from "react";
|
import { useEffect, useMemo, useRef, FC, ReactNode, useCallback } from "react";
|
||||||
import useMeasure from "react-use-measure";
|
import useMeasure from "react-use-measure";
|
||||||
import { ResizeObserver } from "@juggle/resize-observer";
|
import { ResizeObserver } from "@juggle/resize-observer";
|
||||||
import { usePreviewTracks } from "@livekit/components-react";
|
import { usePreviewTracks } from "@livekit/components-react";
|
||||||
@@ -32,6 +32,7 @@ import styles from "./VideoPreview.module.css";
|
|||||||
import { useMediaDevices } from "../livekit/MediaDevicesContext";
|
import { useMediaDevices } from "../livekit/MediaDevicesContext";
|
||||||
import { MuteStates } from "./MuteStates";
|
import { MuteStates } from "./MuteStates";
|
||||||
import { useMediaQuery } from "../useMediaQuery";
|
import { useMediaQuery } from "../useMediaQuery";
|
||||||
|
import { EncryptionSystem } from "../e2ee/sharedKeyManagement";
|
||||||
|
|
||||||
export type MatrixInfo = {
|
export type MatrixInfo = {
|
||||||
userId: string;
|
userId: string;
|
||||||
@@ -41,7 +42,7 @@ export type MatrixInfo = {
|
|||||||
roomName: string;
|
roomName: string;
|
||||||
roomAlias: string | null;
|
roomAlias: string | null;
|
||||||
roomAvatar: string | null;
|
roomAvatar: string | null;
|
||||||
roomEncrypted: boolean;
|
e2eeSystem: EncryptionSystem;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -67,8 +68,8 @@ export const VideoPreview: FC<Props> = ({
|
|||||||
deviceId: devices.audioInput.selectedId,
|
deviceId: devices.audioInput.selectedId,
|
||||||
};
|
};
|
||||||
|
|
||||||
const tracks = usePreviewTracks(
|
const localTrackOptions = useMemo(
|
||||||
{
|
() => ({
|
||||||
// The only reason we request audio here is to get the audio permission
|
// The only reason we request audio here is to get the audio permission
|
||||||
// request over with at the same time. But changing the audio settings
|
// request over with at the same time. But changing the audio settings
|
||||||
// shouldn't cause this hook to recreate the track, which is why we
|
// shouldn't cause this hook to recreate the track, which is why we
|
||||||
@@ -79,13 +80,21 @@ export const VideoPreview: FC<Props> = ({
|
|||||||
video: muteStates.video.enabled && {
|
video: muteStates.video.enabled && {
|
||||||
deviceId: devices.videoInput.selectedId,
|
deviceId: devices.videoInput.selectedId,
|
||||||
},
|
},
|
||||||
},
|
}),
|
||||||
(error) => {
|
[devices.videoInput.selectedId, muteStates.video.enabled],
|
||||||
|
);
|
||||||
|
|
||||||
|
const onError = useCallback(
|
||||||
|
(error: Error) => {
|
||||||
logger.error("Error while creating preview Tracks:", error);
|
logger.error("Error while creating preview Tracks:", error);
|
||||||
muteStates.audio.setEnabled?.(false);
|
muteStates.audio.setEnabled?.(false);
|
||||||
muteStates.video.setEnabled?.(false);
|
muteStates.video.setEnabled?.(false);
|
||||||
},
|
},
|
||||||
|
[muteStates.audio, muteStates.video],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const tracks = usePreviewTracks(localTrackOptions, onError);
|
||||||
|
|
||||||
const videoTrack = useMemo(
|
const videoTrack = useMemo(
|
||||||
() =>
|
() =>
|
||||||
tracks?.find((t) => t.kind === Track.Kind.Video) as
|
tracks?.find((t) => t.kind === Track.Kind.Video) as
|
||||||
@@ -101,7 +110,7 @@ export const VideoPreview: FC<Props> = ({
|
|||||||
if (videoEl.current) {
|
if (videoEl.current) {
|
||||||
videoTrack?.attach(videoEl.current);
|
videoTrack?.attach(videoEl.current);
|
||||||
}
|
}
|
||||||
return () => {
|
return (): void => {
|
||||||
videoTrack?.detach();
|
videoTrack?.detach();
|
||||||
};
|
};
|
||||||
}, [videoTrack]);
|
}, [videoTrack]);
|
||||||
|
|||||||
@@ -21,33 +21,28 @@ import {
|
|||||||
import { useCallback, useEffect, useState } from "react";
|
import { useCallback, useEffect, useState } from "react";
|
||||||
import { deepCompare } from "matrix-js-sdk/src/utils";
|
import { deepCompare } from "matrix-js-sdk/src/utils";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
import {
|
||||||
import { LivekitFocus } from "../livekit/LivekitFocus";
|
LivekitFocus,
|
||||||
|
isLivekitFocus,
|
||||||
function getActiveFocus(
|
} from "matrix-js-sdk/src/matrixrtc/LivekitFocus";
|
||||||
rtcSession: MatrixRTCSession,
|
|
||||||
): LivekitFocus | undefined {
|
|
||||||
const oldestMembership = rtcSession.getOldestMembership();
|
|
||||||
const focus = oldestMembership?.getActiveFoci()[0] as LivekitFocus;
|
|
||||||
|
|
||||||
return focus;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the currently active (livekit) focus for a MatrixRTC session
|
* Gets the currently active (livekit) focus for a MatrixRTC session
|
||||||
* This logic is specific to livekit foci where the whole call must use one
|
* This logic is specific to livekit foci where the whole call must use one
|
||||||
* and the same focus.
|
* and the same focus.
|
||||||
*/
|
*/
|
||||||
export function useActiveFocus(
|
export function useActiveLivekitFocus(
|
||||||
rtcSession: MatrixRTCSession,
|
rtcSession: MatrixRTCSession,
|
||||||
): LivekitFocus | undefined {
|
): LivekitFocus | undefined {
|
||||||
const [activeFocus, setActiveFocus] = useState(() =>
|
const [activeFocus, setActiveFocus] = useState(() => {
|
||||||
getActiveFocus(rtcSession),
|
const f = rtcSession.getActiveFocus();
|
||||||
);
|
// Only handle foci with type="livekit" for now.
|
||||||
|
return !!f && isLivekitFocus(f) ? f : undefined;
|
||||||
|
});
|
||||||
|
|
||||||
const onMembershipsChanged = useCallback(() => {
|
const onMembershipsChanged = useCallback(() => {
|
||||||
const newActiveFocus = getActiveFocus(rtcSession);
|
const newActiveFocus = rtcSession.getActiveFocus();
|
||||||
|
if (!!newActiveFocus && !isLivekitFocus(newActiveFocus)) return;
|
||||||
if (!deepCompare(activeFocus, newActiveFocus)) {
|
if (!deepCompare(activeFocus, newActiveFocus)) {
|
||||||
const oldestMembership = rtcSession.getOldestMembership();
|
const oldestMembership = rtcSession.getOldestMembership();
|
||||||
logger.warn(
|
logger.warn(
|
||||||
@@ -64,7 +59,7 @@ export function useActiveFocus(
|
|||||||
onMembershipsChanged,
|
onMembershipsChanged,
|
||||||
);
|
);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
rtcSession.off(
|
rtcSession.off(
|
||||||
MatrixRTCSessionEvent.MembershipsChanged,
|
MatrixRTCSessionEvent.MembershipsChanged,
|
||||||
onMembershipsChanged,
|
onMembershipsChanged,
|
||||||
|
|||||||
@@ -65,7 +65,7 @@ export function useFullscreen<T>(items: TileDescriptor<T>[]): {
|
|||||||
(prevItem) =>
|
(prevItem) =>
|
||||||
prevItem == null
|
prevItem == null
|
||||||
? null
|
? null
|
||||||
: items.find((i) => i.id === prevItem.id) ?? null,
|
: (items.find((i) => i.id === prevItem.id) ?? null),
|
||||||
[items],
|
[items],
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -79,7 +79,7 @@ export function useFullscreen<T>(items: TileDescriptor<T>[]): {
|
|||||||
(itemId: string) => {
|
(itemId: string) => {
|
||||||
setFullscreenItem(
|
setFullscreenItem(
|
||||||
latestFullscreenItem.current === null
|
latestFullscreenItem.current === null
|
||||||
? latestItems.current.find((i) => i.id === itemId) ?? null
|
? (latestItems.current.find((i) => i.id === itemId) ?? null)
|
||||||
: null,
|
: null,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -14,15 +14,22 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect, useRef, useCallback } from "react";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
import { ClientEvent, MatrixClient } from "matrix-js-sdk/src/client";
|
import { EventType } from "matrix-js-sdk/src/@types/event";
|
||||||
|
import {
|
||||||
|
ClientEvent,
|
||||||
|
MatrixClient,
|
||||||
|
RoomSummary,
|
||||||
|
} from "matrix-js-sdk/src/client";
|
||||||
import { SyncState } from "matrix-js-sdk/src/sync";
|
import { SyncState } from "matrix-js-sdk/src/sync";
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
||||||
|
import { RoomEvent, Room } from "matrix-js-sdk/src/models/room";
|
||||||
|
import { KnownMembership } from "matrix-js-sdk/src/types";
|
||||||
|
import { JoinRule } from "matrix-js-sdk";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
import type { Room } from "matrix-js-sdk/src/models/room";
|
import { widget } from "../widget";
|
||||||
import type { GroupCall } from "matrix-js-sdk/src/webrtc/groupCall";
|
|
||||||
|
|
||||||
export type GroupCallLoaded = {
|
export type GroupCallLoaded = {
|
||||||
kind: "loaded";
|
kind: "loaded";
|
||||||
@@ -38,14 +45,48 @@ export type GroupCallLoading = {
|
|||||||
kind: "loading";
|
kind: "loading";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type GroupCallWaitForInvite = {
|
||||||
|
kind: "waitForInvite";
|
||||||
|
roomSummary: RoomSummary;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type GroupCallCanKnock = {
|
||||||
|
kind: "canKnock";
|
||||||
|
roomSummary: RoomSummary;
|
||||||
|
knock: () => void;
|
||||||
|
};
|
||||||
|
|
||||||
export type GroupCallStatus =
|
export type GroupCallStatus =
|
||||||
| GroupCallLoaded
|
| GroupCallLoaded
|
||||||
| GroupCallLoadFailed
|
| GroupCallLoadFailed
|
||||||
| GroupCallLoading;
|
| GroupCallLoading
|
||||||
|
| GroupCallWaitForInvite
|
||||||
|
| GroupCallCanKnock;
|
||||||
|
|
||||||
export interface GroupCallLoadState {
|
export class CallTerminatedMessage extends Error {
|
||||||
error?: Error;
|
/**
|
||||||
groupCall?: GroupCall;
|
* @param messageBody The message explaining the kind of termination (kick, ban, knock reject, etc.) (translated)
|
||||||
|
*/
|
||||||
|
public messageBody: string;
|
||||||
|
/**
|
||||||
|
* @param reason The user provided reason for the termination (kick/ban)
|
||||||
|
*/
|
||||||
|
public reason?: string;
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param messageTitle The title of the call ended screen message (translated)
|
||||||
|
* @param messageBody The message explaining the kind of termination (kick, ban, knock reject, etc.) (translated)
|
||||||
|
* @param reason The user provided reason for the termination (kick/ban)
|
||||||
|
*/
|
||||||
|
public constructor(
|
||||||
|
messageTitle: string,
|
||||||
|
messageBody: string,
|
||||||
|
reason?: string,
|
||||||
|
) {
|
||||||
|
super(messageTitle);
|
||||||
|
this.messageBody = messageBody;
|
||||||
|
this.reason = reason;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useLoadGroupCall = (
|
export const useLoadGroupCall = (
|
||||||
@@ -53,36 +94,190 @@ export const useLoadGroupCall = (
|
|||||||
roomIdOrAlias: string,
|
roomIdOrAlias: string,
|
||||||
viaServers: string[],
|
viaServers: string[],
|
||||||
): GroupCallStatus => {
|
): GroupCallStatus => {
|
||||||
const { t } = useTranslation();
|
|
||||||
const [state, setState] = useState<GroupCallStatus>({ kind: "loading" });
|
const [state, setState] = useState<GroupCallStatus>({ kind: "loading" });
|
||||||
|
const activeRoom = useRef<Room>();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const bannedError = useCallback(
|
||||||
|
(): CallTerminatedMessage =>
|
||||||
|
new CallTerminatedMessage(
|
||||||
|
t("group_call_loader.banned_heading"),
|
||||||
|
t("group_call_loader.banned_body"),
|
||||||
|
leaveReason(),
|
||||||
|
),
|
||||||
|
[t],
|
||||||
|
);
|
||||||
|
const knockRejectError = useCallback(
|
||||||
|
(): CallTerminatedMessage =>
|
||||||
|
new CallTerminatedMessage(
|
||||||
|
t("group_call_loader.knock_reject_heading"),
|
||||||
|
t("group_call_loader.knock_reject_body"),
|
||||||
|
leaveReason(),
|
||||||
|
),
|
||||||
|
[t],
|
||||||
|
);
|
||||||
|
const removeNoticeError = useCallback(
|
||||||
|
(): CallTerminatedMessage =>
|
||||||
|
new CallTerminatedMessage(
|
||||||
|
t("group_call_loader.call_ended_heading"),
|
||||||
|
t("group_call_loader.call_ended_body"),
|
||||||
|
leaveReason(),
|
||||||
|
),
|
||||||
|
[t],
|
||||||
|
);
|
||||||
|
|
||||||
|
const leaveReason = (): string =>
|
||||||
|
activeRoom.current?.currentState
|
||||||
|
.getStateEvents(EventType.RoomMember, activeRoom.current?.myUserId)
|
||||||
|
?.getContent().reason;
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
const getRoomByAlias = async (alias: string): Promise<Room> => {
|
||||||
|
// We lowercase the localpart when we create the room, so we must lowercase
|
||||||
|
// it here too (we just do the whole alias). We can't do the same to room IDs
|
||||||
|
// though.
|
||||||
|
// Also, we explicitly look up the room alias here. We previously just tried to
|
||||||
|
// join anyway but the js-sdk recreates the room if you pass the alias for a
|
||||||
|
// room you're already joined to (which it probably ought not to).
|
||||||
|
let room: Room | null = null;
|
||||||
|
const lookupResult = await client.getRoomIdForAlias(alias.toLowerCase());
|
||||||
|
logger.info(`${alias} resolved to ${lookupResult.room_id}`);
|
||||||
|
room = client.getRoom(lookupResult.room_id);
|
||||||
|
if (!room) {
|
||||||
|
logger.info(`Room ${lookupResult.room_id} not found, joining.`);
|
||||||
|
room = await client.joinRoom(lookupResult.room_id, {
|
||||||
|
viaServers: lookupResult.servers,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
logger.info(`Already in room ${lookupResult.room_id}, not rejoining.`);
|
||||||
|
}
|
||||||
|
return room;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getRoomByKnocking = async (
|
||||||
|
roomId: string,
|
||||||
|
viaServers: string[],
|
||||||
|
onKnockSent: () => void,
|
||||||
|
): Promise<Room> => {
|
||||||
|
let joinedRoom: Room | null = null;
|
||||||
|
await client.knockRoom(roomId, { viaServers });
|
||||||
|
onKnockSent();
|
||||||
|
const invitePromise = new Promise<void>((resolve, reject) => {
|
||||||
|
client.on(
|
||||||
|
RoomEvent.MyMembership,
|
||||||
|
async (room, membership, prevMembership) => {
|
||||||
|
if (roomId !== room.roomId) return;
|
||||||
|
activeRoom.current = room;
|
||||||
|
if (
|
||||||
|
membership === KnownMembership.Invite &&
|
||||||
|
prevMembership === KnownMembership.Knock
|
||||||
|
) {
|
||||||
|
await client.joinRoom(room.roomId, { viaServers });
|
||||||
|
joinedRoom = room;
|
||||||
|
logger.log("Auto-joined %s", room.roomId);
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
if (membership === KnownMembership.Ban) reject(bannedError());
|
||||||
|
if (membership === KnownMembership.Leave)
|
||||||
|
reject(knockRejectError());
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
await invitePromise;
|
||||||
|
if (!joinedRoom) {
|
||||||
|
throw new Error("Failed to join room after knocking.");
|
||||||
|
}
|
||||||
|
return joinedRoom;
|
||||||
|
};
|
||||||
|
|
||||||
const fetchOrCreateRoom = async (): Promise<Room> => {
|
const fetchOrCreateRoom = async (): Promise<Room> => {
|
||||||
let room: Room | null = null;
|
let room: Room | null = null;
|
||||||
if (roomIdOrAlias[0] === "#") {
|
if (roomIdOrAlias[0] === "#") {
|
||||||
// We lowercase the localpart when we create the room, so we must lowercase
|
const alias = roomIdOrAlias;
|
||||||
// it here too (we just do the whole alias). We can't do the same to room IDs
|
// The call uses a room alias
|
||||||
// though.
|
room = await getRoomByAlias(alias);
|
||||||
// Also, we explicitly look up the room alias here. We previously just tried to
|
activeRoom.current = room;
|
||||||
// join anyway but the js-sdk recreates the room if you pass the alias for a
|
} else {
|
||||||
// room you're already joined to (which it probably ought not to).
|
// The call uses a room_id
|
||||||
const lookupResult = await client.getRoomIdForAlias(
|
const roomId = roomIdOrAlias;
|
||||||
roomIdOrAlias.toLowerCase(),
|
|
||||||
);
|
// first try if the room already exists
|
||||||
logger.info(`${roomIdOrAlias} resolved to ${lookupResult.room_id}`);
|
// - in widget mode
|
||||||
room = client.getRoom(lookupResult.room_id);
|
// - in SPA mode if the user already joined the room
|
||||||
if (!room) {
|
room = client.getRoom(roomId);
|
||||||
logger.info(`Room ${lookupResult.room_id} not found, joining.`);
|
activeRoom.current = room ?? undefined;
|
||||||
room = await client.joinRoom(lookupResult.room_id, {
|
const membership = room?.getMyMembership();
|
||||||
viaServers: lookupResult.servers,
|
if (membership === KnownMembership.Join) {
|
||||||
|
// room already joined so we are done here already.
|
||||||
|
return room!;
|
||||||
|
}
|
||||||
|
if (widget)
|
||||||
|
// in widget mode we never should reach this point. (getRoom should return the room.)
|
||||||
|
throw new Error(
|
||||||
|
"Room not found. The widget-api did not pass over the relevant room events/information.",
|
||||||
|
);
|
||||||
|
|
||||||
|
if (membership === KnownMembership.Ban) {
|
||||||
|
throw bannedError();
|
||||||
|
} else if (membership === KnownMembership.Invite) {
|
||||||
|
room = await client.joinRoom(roomId, {
|
||||||
|
viaServers,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
logger.info(
|
// If the room does not exist we first search for it with viaServers
|
||||||
`Already in room ${lookupResult.room_id}, not rejoining.`,
|
let roomSummary: RoomSummary | undefined = undefined;
|
||||||
);
|
try {
|
||||||
|
roomSummary = await client.getRoomSummary(roomId, viaServers);
|
||||||
|
} catch (error) {
|
||||||
|
// If the room summary endpoint is not supported we let it be undefined and treat this case like
|
||||||
|
// `JoinRule.Public`.
|
||||||
|
// This is how the logic was done before: "we expect any room id passed to EC
|
||||||
|
// to be for a public call" Which is definitely not ideal but worth a try if fetching
|
||||||
|
// the summary crashes.
|
||||||
|
logger.warn(
|
||||||
|
`Could not load room summary to decide whether we want to join or knock.
|
||||||
|
EC will fallback to join as if this would be a public room.
|
||||||
|
Reach out to your homeserver admin to ask them about supporting the \`/summary\` endpoint (im.nheko.summary):`,
|
||||||
|
error,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
roomSummary === undefined ||
|
||||||
|
roomSummary.join_rule === JoinRule.Public
|
||||||
|
) {
|
||||||
|
room = await client.joinRoom(roomId, {
|
||||||
|
viaServers,
|
||||||
|
});
|
||||||
|
} else if (roomSummary.join_rule === JoinRule.Knock) {
|
||||||
|
// bind room summary in this scope so we have it stored in a binding of type `RoomSummary`
|
||||||
|
// instead of `RoomSummary | undefined`. Because we use it in a promise the linter does not accept
|
||||||
|
// the type check from the if condition above.
|
||||||
|
const _roomSummary = roomSummary;
|
||||||
|
let knock: () => void = () => {};
|
||||||
|
const userPressedAskToJoinPromise: Promise<void> = new Promise(
|
||||||
|
(resolve) => {
|
||||||
|
if (_roomSummary.membership !== KnownMembership.Knock) {
|
||||||
|
knock = resolve;
|
||||||
|
} else {
|
||||||
|
// resolve immediately if the user already knocked
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
setState({ kind: "canKnock", roomSummary: _roomSummary, knock });
|
||||||
|
await userPressedAskToJoinPromise;
|
||||||
|
room = await getRoomByKnocking(
|
||||||
|
roomSummary.room_id,
|
||||||
|
viaServers,
|
||||||
|
() =>
|
||||||
|
setState({ kind: "waitForInvite", roomSummary: _roomSummary }),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
throw new Error(
|
||||||
|
`Room ${roomSummary.room_id} is not joinable. This likely means, that the conference owner has changed the room settings to private.`,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
room = await client.joinRoom(roomIdOrAlias, { viaServers });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(
|
logger.info(
|
||||||
@@ -95,6 +290,7 @@ export const useLoadGroupCall = (
|
|||||||
|
|
||||||
const fetchOrCreateGroupCall = async (): Promise<MatrixRTCSession> => {
|
const fetchOrCreateGroupCall = async (): Promise<MatrixRTCSession> => {
|
||||||
const room = await fetchOrCreateRoom();
|
const room = await fetchOrCreateRoom();
|
||||||
|
activeRoom.current = room;
|
||||||
logger.debug(`Fetched / joined room ${roomIdOrAlias}`);
|
logger.debug(`Fetched / joined room ${roomIdOrAlias}`);
|
||||||
|
|
||||||
const rtcSession = client.matrixRTC.getRoomSession(room);
|
const rtcSession = client.matrixRTC.getRoomSession(room);
|
||||||
@@ -119,11 +315,33 @@ export const useLoadGroupCall = (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
waitForClientSyncing()
|
const observeMyMembership = async (): Promise<void> => {
|
||||||
.then(fetchOrCreateGroupCall)
|
await new Promise((_, reject) => {
|
||||||
.then((rtcSession) => setState({ kind: "loaded", rtcSession }))
|
client.on(RoomEvent.MyMembership, async (_, membership) => {
|
||||||
.catch((error) => setState({ kind: "failed", error }));
|
if (membership === KnownMembership.Leave) reject(removeNoticeError());
|
||||||
}, [client, roomIdOrAlias, viaServers, t]);
|
if (membership === KnownMembership.Ban) reject(bannedError());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
if (state.kind === "loading") {
|
||||||
|
logger.log("Start loading group call");
|
||||||
|
waitForClientSyncing()
|
||||||
|
.then(fetchOrCreateGroupCall)
|
||||||
|
.then((rtcSession) => setState({ kind: "loaded", rtcSession }))
|
||||||
|
.then(observeMyMembership)
|
||||||
|
.catch((error) => setState({ kind: "failed", error }));
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
bannedError,
|
||||||
|
client,
|
||||||
|
knockRejectError,
|
||||||
|
removeNoticeError,
|
||||||
|
roomIdOrAlias,
|
||||||
|
state,
|
||||||
|
t,
|
||||||
|
viaServers,
|
||||||
|
]);
|
||||||
|
|
||||||
return state;
|
return state;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ export function usePageUnload(callback: () => void): void {
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
window.addEventListener("beforeunload", onBeforeUnload);
|
window.addEventListener("beforeunload", onBeforeUnload);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
window.removeEventListener("pagehide", onBeforeUnload);
|
window.removeEventListener("pagehide", onBeforeUnload);
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
|||||||
@@ -15,29 +15,90 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
import { MatrixRTCSession } from "matrix-js-sdk/src/matrixrtc/MatrixRTCSession";
|
||||||
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
import {
|
||||||
|
LivekitFocus,
|
||||||
|
LivekitFocusActive,
|
||||||
|
isLivekitFocus,
|
||||||
|
isLivekitFocusConfig,
|
||||||
|
} from "matrix-js-sdk/src/matrixrtc/LivekitFocus";
|
||||||
|
|
||||||
import { PosthogAnalytics } from "./analytics/PosthogAnalytics";
|
import { PosthogAnalytics } from "./analytics/PosthogAnalytics";
|
||||||
import { LivekitFocus } from "./livekit/LivekitFocus";
|
|
||||||
import { Config } from "./config/Config";
|
import { Config } from "./config/Config";
|
||||||
import { ElementWidgetActions, WidgetHelpers, widget } from "./widget";
|
import { ElementWidgetActions, WidgetHelpers, widget } from "./widget";
|
||||||
|
|
||||||
function makeFocus(livekitAlias: string): LivekitFocus {
|
const FOCI_WK_KEY = "org.matrix.msc4143.rtc_foci";
|
||||||
const urlFromConf = Config.get().livekit!.livekit_service_url;
|
|
||||||
if (!urlFromConf) {
|
|
||||||
throw new Error("No livekit_service_url is configured!");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
export function makeActiveFocus(): LivekitFocusActive {
|
||||||
return {
|
return {
|
||||||
type: "livekit",
|
type: "livekit",
|
||||||
livekit_service_url: urlFromConf,
|
focus_selection: "oldest_membership",
|
||||||
livekit_alias: livekitAlias,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function enterRTCSession(
|
async function makePreferredLivekitFoci(
|
||||||
|
rtcSession: MatrixRTCSession,
|
||||||
|
livekitAlias: string,
|
||||||
|
): Promise<LivekitFocus[]> {
|
||||||
|
logger.log("Start building foci_preferred list: ", rtcSession.room.roomId);
|
||||||
|
|
||||||
|
const preferredFoci: LivekitFocus[] = [];
|
||||||
|
|
||||||
|
// Make the Focus from the running rtc session the highest priority one
|
||||||
|
// This minimizes how often we need to switch foci during a call.
|
||||||
|
const focusInUse = rtcSession.getFocusInUse();
|
||||||
|
if (focusInUse && isLivekitFocus(focusInUse)) {
|
||||||
|
logger.log("Adding livekit focus from oldest member: ", focusInUse);
|
||||||
|
preferredFoci.push(focusInUse);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prioritize the client well known over the configured sfu.
|
||||||
|
const wellKnownFoci =
|
||||||
|
rtcSession.room.client.getClientWellKnown()?.[FOCI_WK_KEY];
|
||||||
|
if (Array.isArray(wellKnownFoci)) {
|
||||||
|
preferredFoci.push(
|
||||||
|
...wellKnownFoci
|
||||||
|
.filter((f) => !!f)
|
||||||
|
.filter(isLivekitFocusConfig)
|
||||||
|
.map((wellKnownFocus) => {
|
||||||
|
logger.log("Adding livekit focus from well known: ", wellKnownFocus);
|
||||||
|
return { ...wellKnownFocus, livekit_alias: livekitAlias };
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const urlFromConf = Config.get().livekit?.livekit_service_url;
|
||||||
|
if (urlFromConf) {
|
||||||
|
const focusFormConf: LivekitFocus = {
|
||||||
|
type: "livekit",
|
||||||
|
livekit_service_url: urlFromConf,
|
||||||
|
livekit_alias: livekitAlias,
|
||||||
|
};
|
||||||
|
logger.log("Adding livekit focus from config: ", focusFormConf);
|
||||||
|
preferredFoci.push(focusFormConf);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preferredFoci.length === 0)
|
||||||
|
throw new Error(
|
||||||
|
`No livekit_service_url is configured so we could not create a focus.
|
||||||
|
Currently we skip computing a focus based on other users in the room.`,
|
||||||
|
);
|
||||||
|
|
||||||
|
return preferredFoci;
|
||||||
|
|
||||||
|
// TODO: we want to do something like this:
|
||||||
|
//
|
||||||
|
// const focusOtherMembers = await focusFromOtherMembers(
|
||||||
|
// rtcSession,
|
||||||
|
// livekitAlias,
|
||||||
|
// );
|
||||||
|
// if (focusOtherMembers) preferredFoci.push(focusOtherMembers);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function enterRTCSession(
|
||||||
rtcSession: MatrixRTCSession,
|
rtcSession: MatrixRTCSession,
|
||||||
encryptMedia: boolean,
|
encryptMedia: boolean,
|
||||||
): void {
|
): Promise<void> {
|
||||||
PosthogAnalytics.instance.eventCallEnded.cacheStartCall(new Date());
|
PosthogAnalytics.instance.eventCallEnded.cacheStartCall(new Date());
|
||||||
PosthogAnalytics.instance.eventCallStarted.track(rtcSession.room.roomId);
|
PosthogAnalytics.instance.eventCallStarted.track(rtcSession.room.roomId);
|
||||||
|
|
||||||
@@ -47,8 +108,18 @@ export function enterRTCSession(
|
|||||||
|
|
||||||
// right now we assume everything is a room-scoped call
|
// right now we assume everything is a room-scoped call
|
||||||
const livekitAlias = rtcSession.room.roomId;
|
const livekitAlias = rtcSession.room.roomId;
|
||||||
|
const useDeviceSessionMemberEvents =
|
||||||
rtcSession.joinRoomSession([makeFocus(livekitAlias)], encryptMedia);
|
Config.get().features?.feature_use_device_session_member_events;
|
||||||
|
rtcSession.joinRoomSession(
|
||||||
|
await makePreferredLivekitFoci(rtcSession, livekitAlias),
|
||||||
|
makeActiveFocus(),
|
||||||
|
{
|
||||||
|
manageMediaKeys: encryptMedia,
|
||||||
|
...(useDeviceSessionMemberEvents !== undefined && {
|
||||||
|
useLegacyMemberEvents: !useDeviceSessionMemberEvents,
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const widgetPostHangupProcedure = async (
|
const widgetPostHangupProcedure = async (
|
||||||
|
|||||||
@@ -47,18 +47,18 @@ export const ProfileSettingsTab: FC<Props> = ({ client }) => {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const form = formRef.current!;
|
const form = formRef.current!;
|
||||||
// Auto-save when the user dismisses this component
|
// Auto-save when the user dismisses this component
|
||||||
return () => {
|
return (): void => {
|
||||||
if (formChanged.current) {
|
if (formChanged.current) {
|
||||||
const data = new FormData(form);
|
const data = new FormData(form);
|
||||||
const displayNameDataEntry = data.get("displayName");
|
const displayNameDataEntry = data.get("displayName");
|
||||||
const avatar = data.get("avatar");
|
const avatar = data.get("avatar");
|
||||||
|
|
||||||
const avatarSize =
|
const avatarSize =
|
||||||
typeof avatar == "string" ? avatar.length : avatar?.size ?? 0;
|
typeof avatar == "string" ? avatar.length : (avatar?.size ?? 0);
|
||||||
const displayName =
|
const displayName =
|
||||||
typeof displayNameDataEntry == "string"
|
typeof displayNameDataEntry == "string"
|
||||||
? displayNameDataEntry
|
? displayNameDataEntry
|
||||||
: displayNameDataEntry?.name ?? null;
|
: (displayNameDataEntry?.name ?? null);
|
||||||
|
|
||||||
if (!displayName) {
|
if (!displayName) {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -298,13 +298,13 @@ export function useRageshakeRequest(): (
|
|||||||
|
|
||||||
const sendRageshakeRequest = useCallback(
|
const sendRageshakeRequest = useCallback(
|
||||||
(roomId: string, rageshakeRequestId: string) => {
|
(roomId: string, rageshakeRequestId: string) => {
|
||||||
|
// @ts-expect-error - org.matrix.rageshake_request is not part of `keyof TimelineEvents` but it is okay to sent a custom event.
|
||||||
client!.sendEvent(roomId, "org.matrix.rageshake_request", {
|
client!.sendEvent(roomId, "org.matrix.rageshake_request", {
|
||||||
request_id: rageshakeRequestId,
|
request_id: rageshakeRequestId,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
[client],
|
[client],
|
||||||
);
|
);
|
||||||
|
|
||||||
return sendRageshakeRequest;
|
return sendRageshakeRequest;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -334,7 +334,7 @@ export function useRageshakeRequestModal(
|
|||||||
|
|
||||||
client.on(ClientEvent.Event, onEvent);
|
client.on(ClientEvent.Event, onEvent);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
client.removeListener(ClientEvent.Event, onEvent);
|
client.removeListener(ClientEvent.Event, onEvent);
|
||||||
};
|
};
|
||||||
}, [setOpen, roomId, client]);
|
}, [setOpen, roomId, client]);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright 2023 New Vector Ltd
|
Copyright 2023-2024 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,25 +16,41 @@ limitations under the License.
|
|||||||
|
|
||||||
import {
|
import {
|
||||||
connectedParticipantsObserver,
|
connectedParticipantsObserver,
|
||||||
|
observeParticipantEvents,
|
||||||
observeParticipantMedia,
|
observeParticipantMedia,
|
||||||
} from "@livekit/components-core";
|
} from "@livekit/components-core";
|
||||||
import { Room as LivekitRoom, RemoteParticipant } from "livekit-client";
|
import {
|
||||||
|
Room as LivekitRoom,
|
||||||
|
LocalParticipant,
|
||||||
|
ParticipantEvent,
|
||||||
|
RemoteParticipant,
|
||||||
|
} from "livekit-client";
|
||||||
import { Room as MatrixRoom, RoomMember } from "matrix-js-sdk/src/matrix";
|
import { Room as MatrixRoom, RoomMember } from "matrix-js-sdk/src/matrix";
|
||||||
import { useEffect, useRef } from "react";
|
import { useEffect, useRef } from "react";
|
||||||
import {
|
import {
|
||||||
|
BehaviorSubject,
|
||||||
EMPTY,
|
EMPTY,
|
||||||
Observable,
|
Observable,
|
||||||
|
audit,
|
||||||
combineLatest,
|
combineLatest,
|
||||||
concat,
|
concat,
|
||||||
|
distinctUntilChanged,
|
||||||
|
filter,
|
||||||
|
map,
|
||||||
|
merge,
|
||||||
mergeAll,
|
mergeAll,
|
||||||
of,
|
of,
|
||||||
sample,
|
sample,
|
||||||
scan,
|
scan,
|
||||||
|
shareReplay,
|
||||||
startWith,
|
startWith,
|
||||||
takeUntil,
|
switchAll,
|
||||||
|
switchMap,
|
||||||
|
throttleTime,
|
||||||
|
timer,
|
||||||
zip,
|
zip,
|
||||||
} from "rxjs";
|
} from "rxjs";
|
||||||
import { state } from "@react-rxjs/core";
|
import { StateObservable, state } from "@react-rxjs/core";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
|
||||||
import { ViewModel } from "./ViewModel";
|
import { ViewModel } from "./ViewModel";
|
||||||
@@ -45,14 +61,21 @@ import {
|
|||||||
} from "../livekit/useECConnectionState";
|
} from "../livekit/useECConnectionState";
|
||||||
import { usePrevious } from "../usePrevious";
|
import { usePrevious } from "../usePrevious";
|
||||||
import {
|
import {
|
||||||
TileViewModel,
|
MediaViewModel,
|
||||||
UserMediaTileViewModel,
|
UserMediaViewModel,
|
||||||
ScreenShareTileViewModel,
|
ScreenShareViewModel,
|
||||||
} from "./TileViewModel";
|
} from "./MediaViewModel";
|
||||||
import { finalizeValue } from "../observable-utils";
|
import { finalizeValue } from "../observable-utils";
|
||||||
|
import { ObservableScope } from "./ObservableScope";
|
||||||
|
|
||||||
|
// How long we wait after a focus switch before showing the real participant
|
||||||
|
// list again
|
||||||
|
const POST_FOCUS_PARTICIPANT_UPDATE_DELAY_MS = 3000;
|
||||||
|
|
||||||
// Represents something that should get a tile on the layout,
|
// Represents something that should get a tile on the layout,
|
||||||
// ie. a user's video feed or a screen share feed.
|
// ie. a user's video feed or a screen share feed.
|
||||||
|
// TODO: This exposes too much information to the view layer, let's keep this
|
||||||
|
// information internal to the view model and switch to using Tile<T> instead
|
||||||
export interface TileDescriptor<T> {
|
export interface TileDescriptor<T> {
|
||||||
id: string;
|
id: string;
|
||||||
focused: boolean;
|
focused: boolean;
|
||||||
@@ -65,9 +88,123 @@ export interface TileDescriptor<T> {
|
|||||||
data: T;
|
data: T;
|
||||||
}
|
}
|
||||||
|
|
||||||
// How long we wait after a focus switch before showing the real participant
|
export interface GridLayout {
|
||||||
// list again
|
type: "grid";
|
||||||
const POST_FOCUS_PARTICIPANT_UPDATE_DELAY_MS = 3000;
|
spotlight?: MediaViewModel[];
|
||||||
|
grid: UserMediaViewModel[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SpotlightLayout {
|
||||||
|
type: "spotlight";
|
||||||
|
spotlight: MediaViewModel[];
|
||||||
|
grid: UserMediaViewModel[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FullScreenLayout {
|
||||||
|
type: "full screen";
|
||||||
|
spotlight: MediaViewModel[];
|
||||||
|
pip?: UserMediaViewModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface PipLayout {
|
||||||
|
type: "pip";
|
||||||
|
spotlight: MediaViewModel[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A layout defining the media tiles present on screen and their visual
|
||||||
|
* arrangement.
|
||||||
|
*/
|
||||||
|
export type Layout =
|
||||||
|
| GridLayout
|
||||||
|
| SpotlightLayout
|
||||||
|
| FullScreenLayout
|
||||||
|
| PipLayout;
|
||||||
|
|
||||||
|
export type GridMode = "grid" | "spotlight";
|
||||||
|
|
||||||
|
export type WindowMode = "normal" | "full screen" | "pip";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sorting bins defining the order in which media tiles appear in the layout.
|
||||||
|
*/
|
||||||
|
enum SortingBin {
|
||||||
|
SelfStart,
|
||||||
|
Presenters,
|
||||||
|
Speakers,
|
||||||
|
VideoAndAudio,
|
||||||
|
Video,
|
||||||
|
Audio,
|
||||||
|
NoMedia,
|
||||||
|
SelfEnd,
|
||||||
|
}
|
||||||
|
|
||||||
|
class UserMedia {
|
||||||
|
private readonly scope = new ObservableScope();
|
||||||
|
public readonly vm: UserMediaViewModel;
|
||||||
|
public readonly speaker: Observable<boolean>;
|
||||||
|
public readonly presenter: Observable<boolean>;
|
||||||
|
|
||||||
|
public constructor(
|
||||||
|
public readonly id: string,
|
||||||
|
member: RoomMember | undefined,
|
||||||
|
participant: LocalParticipant | RemoteParticipant,
|
||||||
|
callEncrypted: boolean,
|
||||||
|
) {
|
||||||
|
this.vm = new UserMediaViewModel(id, member, participant, callEncrypted);
|
||||||
|
|
||||||
|
this.speaker = this.vm.speaking.pipeState(
|
||||||
|
// Require 1 s of continuous speaking to become a speaker, and 10 s of
|
||||||
|
// continuous silence to stop being considered a speaker
|
||||||
|
audit((s) =>
|
||||||
|
merge(
|
||||||
|
timer(s ? 1000 : 10000),
|
||||||
|
// If the speaking flag resets to its original value during this time,
|
||||||
|
// end the silencing window to stick with that original value
|
||||||
|
this.vm.speaking.pipe(filter((s1) => s1 !== s)),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
startWith(false),
|
||||||
|
distinctUntilChanged(),
|
||||||
|
this.scope.bind(),
|
||||||
|
// Make this Observable hot so that the timers don't reset when you
|
||||||
|
// resubscribe
|
||||||
|
shareReplay(1),
|
||||||
|
);
|
||||||
|
|
||||||
|
this.presenter = observeParticipantEvents(
|
||||||
|
participant,
|
||||||
|
ParticipantEvent.TrackPublished,
|
||||||
|
ParticipantEvent.TrackUnpublished,
|
||||||
|
ParticipantEvent.LocalTrackPublished,
|
||||||
|
ParticipantEvent.LocalTrackUnpublished,
|
||||||
|
).pipe(map((p) => p.isScreenShareEnabled));
|
||||||
|
}
|
||||||
|
|
||||||
|
public destroy(): void {
|
||||||
|
this.scope.end();
|
||||||
|
this.vm.destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScreenShare {
|
||||||
|
public readonly vm: ScreenShareViewModel;
|
||||||
|
|
||||||
|
public constructor(
|
||||||
|
id: string,
|
||||||
|
member: RoomMember | undefined,
|
||||||
|
participant: LocalParticipant | RemoteParticipant,
|
||||||
|
callEncrypted: boolean,
|
||||||
|
) {
|
||||||
|
this.vm = new ScreenShareViewModel(id, member, participant, callEncrypted);
|
||||||
|
}
|
||||||
|
|
||||||
|
public destroy(): void {
|
||||||
|
this.vm.destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type MediaItem = UserMedia | ScreenShare;
|
||||||
|
|
||||||
function findMatrixMember(
|
function findMatrixMember(
|
||||||
room: MatrixRoom,
|
room: MatrixRoom,
|
||||||
@@ -98,160 +235,346 @@ export class CallViewModel extends ViewModel {
|
|||||||
|
|
||||||
// Lists of participants to "hold" on display, even if LiveKit claims that
|
// Lists of participants to "hold" on display, even if LiveKit claims that
|
||||||
// they've left
|
// they've left
|
||||||
private readonly remoteParticipantHolds = zip(
|
private readonly remoteParticipantHolds: Observable<RemoteParticipant[][]> =
|
||||||
this.connectionState,
|
zip(
|
||||||
this.rawRemoteParticipants.pipe(sample(this.connectionState)),
|
this.connectionState,
|
||||||
(s, ps) => {
|
this.rawRemoteParticipants.pipe(sample(this.connectionState)),
|
||||||
// Whenever we switch focuses, we should retain all the previous
|
(s, ps) => {
|
||||||
// participants for at least POST_FOCUS_PARTICIPANT_UPDATE_DELAY_MS ms to
|
// Whenever we switch focuses, we should retain all the previous
|
||||||
// give their clients time to switch over and avoid jarring layout shifts
|
// participants for at least POST_FOCUS_PARTICIPANT_UPDATE_DELAY_MS ms to
|
||||||
if (s === ECAddonConnectionState.ECSwitchingFocus) {
|
// give their clients time to switch over and avoid jarring layout shifts
|
||||||
return concat(
|
if (s === ECAddonConnectionState.ECSwitchingFocus) {
|
||||||
// Hold these participants
|
return concat(
|
||||||
of({ hold: ps }),
|
// Hold these participants
|
||||||
// Wait for time to pass and the connection state to have changed
|
of({ hold: ps }),
|
||||||
Promise.all([
|
// Wait for time to pass and the connection state to have changed
|
||||||
new Promise<void>((resolve) =>
|
Promise.all([
|
||||||
setTimeout(resolve, POST_FOCUS_PARTICIPANT_UPDATE_DELAY_MS),
|
new Promise<void>((resolve) =>
|
||||||
),
|
setTimeout(resolve, POST_FOCUS_PARTICIPANT_UPDATE_DELAY_MS),
|
||||||
new Promise<void>((resolve) => {
|
),
|
||||||
const subscription = this.connectionState
|
new Promise<void>((resolve) => {
|
||||||
.pipe(takeUntil(this.destroyed))
|
const subscription = this.connectionState
|
||||||
.subscribe((s) => {
|
.pipe(this.scope.bind())
|
||||||
if (s !== ECAddonConnectionState.ECSwitchingFocus) {
|
.subscribe((s) => {
|
||||||
resolve();
|
if (s !== ECAddonConnectionState.ECSwitchingFocus) {
|
||||||
subscription.unsubscribe();
|
resolve();
|
||||||
}
|
subscription.unsubscribe();
|
||||||
});
|
}
|
||||||
}),
|
});
|
||||||
// Then unhold them
|
}),
|
||||||
]).then(() => Promise.resolve({ unhold: ps })),
|
// Then unhold them
|
||||||
);
|
]).then(() => Promise.resolve({ unhold: ps })),
|
||||||
} else {
|
);
|
||||||
return EMPTY;
|
} else {
|
||||||
}
|
return EMPTY;
|
||||||
},
|
}
|
||||||
).pipe(
|
},
|
||||||
mergeAll(),
|
).pipe(
|
||||||
// Aggregate the hold instructions into a single list showing which
|
mergeAll(),
|
||||||
// participants are being held
|
// Aggregate the hold instructions into a single list showing which
|
||||||
scan(
|
// participants are being held
|
||||||
(holds, instruction) =>
|
scan(
|
||||||
"hold" in instruction
|
(holds, instruction) =>
|
||||||
? [instruction.hold, ...holds]
|
"hold" in instruction
|
||||||
: holds.filter((h) => h !== instruction.unhold),
|
? [instruction.hold, ...holds]
|
||||||
[] as RemoteParticipant[][],
|
: holds.filter((h) => h !== instruction.unhold),
|
||||||
|
[] as RemoteParticipant[][],
|
||||||
|
),
|
||||||
|
startWith([]),
|
||||||
|
);
|
||||||
|
|
||||||
|
private readonly remoteParticipants: Observable<RemoteParticipant[]> =
|
||||||
|
combineLatest(
|
||||||
|
[this.rawRemoteParticipants, this.remoteParticipantHolds],
|
||||||
|
(raw, holds) => {
|
||||||
|
const result = [...raw];
|
||||||
|
const resultIds = new Set(result.map((p) => p.identity));
|
||||||
|
|
||||||
|
// Incorporate the held participants into the list
|
||||||
|
for (const hold of holds) {
|
||||||
|
for (const p of hold) {
|
||||||
|
if (!resultIds.has(p.identity)) {
|
||||||
|
result.push(p);
|
||||||
|
resultIds.add(p.identity);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
private readonly mediaItems: StateObservable<MediaItem[]> = state(
|
||||||
|
combineLatest([
|
||||||
|
this.remoteParticipants,
|
||||||
|
observeParticipantMedia(this.livekitRoom.localParticipant),
|
||||||
|
]).pipe(
|
||||||
|
scan(
|
||||||
|
(
|
||||||
|
prevItems,
|
||||||
|
[remoteParticipants, { participant: localParticipant }],
|
||||||
|
) => {
|
||||||
|
let allGhosts = true;
|
||||||
|
|
||||||
|
const newItems = new Map(
|
||||||
|
function* (this: CallViewModel): Iterable<[string, MediaItem]> {
|
||||||
|
for (const p of [localParticipant, ...remoteParticipants]) {
|
||||||
|
const member = findMatrixMember(this.matrixRoom, p.identity);
|
||||||
|
allGhosts &&= member === undefined;
|
||||||
|
// We always start with a local participant with the empty string as
|
||||||
|
// their ID before we're connected, this is fine and we'll be in
|
||||||
|
// "all ghosts" mode.
|
||||||
|
if (p.identity !== "" && member === undefined) {
|
||||||
|
logger.warn(
|
||||||
|
`Ruh, roh! No matrix member found for SFU participant '${p.identity}': creating g-g-g-ghost!`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const userMediaId = p.identity;
|
||||||
|
yield [
|
||||||
|
userMediaId,
|
||||||
|
prevItems.get(userMediaId) ??
|
||||||
|
new UserMedia(userMediaId, member, p, this.encrypted),
|
||||||
|
];
|
||||||
|
|
||||||
|
if (p.isScreenShareEnabled) {
|
||||||
|
const screenShareId = `${userMediaId}:screen-share`;
|
||||||
|
yield [
|
||||||
|
screenShareId,
|
||||||
|
prevItems.get(screenShareId) ??
|
||||||
|
new ScreenShare(screenShareId, member, p, this.encrypted),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.bind(this)(),
|
||||||
|
);
|
||||||
|
|
||||||
|
for (const [id, t] of prevItems) if (!newItems.has(id)) t.destroy();
|
||||||
|
|
||||||
|
// If every item is a ghost, that probably means we're still connecting
|
||||||
|
// and shouldn't bother showing anything yet
|
||||||
|
return allGhosts ? new Map() : newItems;
|
||||||
|
},
|
||||||
|
new Map<string, MediaItem>(),
|
||||||
|
),
|
||||||
|
map((ms) => [...ms.values()]),
|
||||||
|
finalizeValue((ts) => {
|
||||||
|
for (const t of ts) t.destroy();
|
||||||
|
}),
|
||||||
),
|
),
|
||||||
startWith([]),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
private readonly remoteParticipants = combineLatest(
|
private readonly userMedia: Observable<UserMedia[]> = this.mediaItems.pipe(
|
||||||
[this.rawRemoteParticipants, this.remoteParticipantHolds],
|
map((ms) => ms.filter((m): m is UserMedia => m instanceof UserMedia)),
|
||||||
(raw, holds) => {
|
);
|
||||||
const result = [...raw];
|
|
||||||
const resultIds = new Set(result.map((p) => p.identity));
|
|
||||||
|
|
||||||
// Incorporate the held participants into the list
|
private readonly screenShares: Observable<ScreenShare[]> =
|
||||||
for (const hold of holds) {
|
this.mediaItems.pipe(
|
||||||
for (const p of hold) {
|
map((ms) => ms.filter((m): m is ScreenShare => m instanceof ScreenShare)),
|
||||||
if (!resultIds.has(p.identity)) {
|
);
|
||||||
result.push(p);
|
|
||||||
resultIds.add(p.identity);
|
private readonly spotlightSpeaker: Observable<UserMedia | null> =
|
||||||
|
this.userMedia.pipe(
|
||||||
|
switchMap((ms) =>
|
||||||
|
ms.length === 0
|
||||||
|
? of([])
|
||||||
|
: combineLatest(
|
||||||
|
ms.map((m) => m.vm.speaking.pipe(map((s) => [m, s] as const))),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
scan<(readonly [UserMedia, boolean])[], UserMedia | null, null>(
|
||||||
|
(prev, ms) =>
|
||||||
|
// Decide who to spotlight:
|
||||||
|
// If the previous speaker is still speaking, stick with them rather
|
||||||
|
// than switching eagerly to someone else
|
||||||
|
ms.find(([m, s]) => m === prev && s)?.[0] ??
|
||||||
|
// Otherwise, select anyone who is speaking
|
||||||
|
ms.find(([, s]) => s)?.[0] ??
|
||||||
|
// Otherwise, stick with the person who was last speaking
|
||||||
|
prev ??
|
||||||
|
// Otherwise, spotlight the local user
|
||||||
|
ms.find(([m]) => m.vm.local)?.[0] ??
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
),
|
||||||
|
distinctUntilChanged(),
|
||||||
|
throttleTime(800, undefined, { leading: true, trailing: true }),
|
||||||
|
);
|
||||||
|
|
||||||
|
private readonly grid: Observable<UserMediaViewModel[]> = this.userMedia.pipe(
|
||||||
|
switchMap((ms) => {
|
||||||
|
const bins = ms.map((m) =>
|
||||||
|
combineLatest(
|
||||||
|
[m.speaker, m.presenter, m.vm.audioEnabled, m.vm.videoEnabled],
|
||||||
|
(speaker, presenter, audio, video) => {
|
||||||
|
let bin: SortingBin;
|
||||||
|
if (m.vm.local) bin = SortingBin.SelfStart;
|
||||||
|
else if (presenter) bin = SortingBin.Presenters;
|
||||||
|
else if (speaker) bin = SortingBin.Speakers;
|
||||||
|
else if (video)
|
||||||
|
bin = audio ? SortingBin.VideoAndAudio : SortingBin.Video;
|
||||||
|
else bin = audio ? SortingBin.Audio : SortingBin.NoMedia;
|
||||||
|
|
||||||
|
return [m, bin] as const;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
// Sort the media by bin order and generate a tile for each one
|
||||||
|
return bins.length === 0
|
||||||
|
? of([])
|
||||||
|
: combineLatest(bins, (...bins) =>
|
||||||
|
bins.sort(([, bin1], [, bin2]) => bin1 - bin2).map(([m]) => m.vm),
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
private readonly spotlight: Observable<MediaViewModel[]> = combineLatest(
|
||||||
|
[this.screenShares, this.spotlightSpeaker],
|
||||||
|
(screenShares, spotlightSpeaker): MediaViewModel[] =>
|
||||||
|
screenShares.length > 0
|
||||||
|
? screenShares.map((m) => m.vm)
|
||||||
|
: spotlightSpeaker === null
|
||||||
|
? []
|
||||||
|
: [spotlightSpeaker.vm],
|
||||||
|
);
|
||||||
|
|
||||||
|
// TODO: Make this react to changes in window dimensions and screen
|
||||||
|
// orientation
|
||||||
|
private readonly windowMode = of<WindowMode>("normal");
|
||||||
|
|
||||||
|
private readonly _gridMode = new BehaviorSubject<GridMode>("grid");
|
||||||
|
/**
|
||||||
|
* The layout mode of the media tile grid.
|
||||||
|
*/
|
||||||
|
public readonly gridMode = state(this._gridMode);
|
||||||
|
|
||||||
|
public setGridMode(value: GridMode): void {
|
||||||
|
this._gridMode.next(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public readonly layout: StateObservable<Layout> = state(
|
||||||
|
combineLatest([this._gridMode, this.windowMode], (gridMode, windowMode) => {
|
||||||
|
switch (windowMode) {
|
||||||
|
case "full screen":
|
||||||
|
throw new Error("unimplemented");
|
||||||
|
case "pip":
|
||||||
|
throw new Error("unimplemented");
|
||||||
|
case "normal": {
|
||||||
|
switch (gridMode) {
|
||||||
|
case "grid":
|
||||||
|
return combineLatest(
|
||||||
|
[this.grid, this.spotlight, this.screenShares],
|
||||||
|
(grid, spotlight, screenShares): Layout => ({
|
||||||
|
type: "grid",
|
||||||
|
spotlight: screenShares.length > 0 ? spotlight : undefined,
|
||||||
|
grid,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
case "spotlight":
|
||||||
|
return combineLatest(
|
||||||
|
[this.grid, this.spotlight],
|
||||||
|
(grid, spotlight): Layout => ({
|
||||||
|
type: "spotlight",
|
||||||
|
spotlight,
|
||||||
|
grid,
|
||||||
|
}),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}).pipe(switchAll()),
|
||||||
return result;
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The media tiles to be displayed in the call view.
|
* The media tiles to be displayed in the call view.
|
||||||
*/
|
*/
|
||||||
public readonly tiles = state(
|
// TODO: Get rid of this field, replacing it with the 'layout' field above
|
||||||
combineLatest([
|
// which keeps more details of the layout order internal to the view model
|
||||||
this.remoteParticipants,
|
public readonly tiles: StateObservable<TileDescriptor<MediaViewModel>[]> =
|
||||||
observeParticipantMedia(this.livekitRoom.localParticipant),
|
state(
|
||||||
]).pipe(
|
combineLatest([
|
||||||
scan((ts, [remoteParticipants, { participant: localParticipant }]) => {
|
this.remoteParticipants,
|
||||||
const ps = [localParticipant, ...remoteParticipants];
|
observeParticipantMedia(this.livekitRoom.localParticipant),
|
||||||
const tilesById = new Map(ts.map((t) => [t.id, t]));
|
]).pipe(
|
||||||
const now = Date.now();
|
scan((ts, [remoteParticipants, { participant: localParticipant }]) => {
|
||||||
let allGhosts = true;
|
const ps = [localParticipant, ...remoteParticipants];
|
||||||
|
const tilesById = new Map(ts.map((t) => [t.id, t]));
|
||||||
|
const now = Date.now();
|
||||||
|
let allGhosts = true;
|
||||||
|
|
||||||
const newTiles = ps.flatMap((p) => {
|
const newTiles = ps.flatMap((p) => {
|
||||||
const userMediaId = p.identity;
|
const userMediaId = p.identity;
|
||||||
const member = findMatrixMember(this.matrixRoom, userMediaId);
|
const member = findMatrixMember(this.matrixRoom, userMediaId);
|
||||||
allGhosts &&= member === undefined;
|
allGhosts &&= member === undefined;
|
||||||
const spokeRecently =
|
const spokeRecently =
|
||||||
p.lastSpokeAt !== undefined && now - +p.lastSpokeAt <= 10000;
|
p.lastSpokeAt !== undefined && now - +p.lastSpokeAt <= 10000;
|
||||||
|
|
||||||
// We always start with a local participant with the empty string as
|
// We always start with a local participant with the empty string as
|
||||||
// their ID before we're connected, this is fine and we'll be in
|
// their ID before we're connected, this is fine and we'll be in
|
||||||
// "all ghosts" mode.
|
// "all ghosts" mode.
|
||||||
if (userMediaId !== "" && member === undefined) {
|
if (userMediaId !== "" && member === undefined) {
|
||||||
logger.warn(
|
logger.warn(
|
||||||
`Ruh, roh! No matrix member found for SFU participant '${userMediaId}': creating g-g-g-ghost!`,
|
`Ruh, roh! No matrix member found for SFU participant '${userMediaId}': creating g-g-g-ghost!`,
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const userMediaVm =
|
|
||||||
tilesById.get(userMediaId)?.data ??
|
|
||||||
new UserMediaTileViewModel(userMediaId, member, p, this.encrypted);
|
|
||||||
tilesById.delete(userMediaId);
|
|
||||||
|
|
||||||
const userMediaTile: TileDescriptor<TileViewModel> = {
|
|
||||||
id: userMediaId,
|
|
||||||
focused: false,
|
|
||||||
isPresenter: p.isScreenShareEnabled,
|
|
||||||
isSpeaker: (p.isSpeaking || spokeRecently) && !p.isLocal,
|
|
||||||
hasVideo: p.isCameraEnabled,
|
|
||||||
local: p.isLocal,
|
|
||||||
largeBaseSize: false,
|
|
||||||
data: userMediaVm,
|
|
||||||
};
|
|
||||||
|
|
||||||
if (p.isScreenShareEnabled) {
|
|
||||||
const screenShareId = `${userMediaId}:screen-share`;
|
|
||||||
const screenShareVm =
|
|
||||||
tilesById.get(screenShareId)?.data ??
|
|
||||||
new ScreenShareTileViewModel(
|
|
||||||
screenShareId,
|
|
||||||
member,
|
|
||||||
p,
|
|
||||||
this.encrypted,
|
|
||||||
);
|
);
|
||||||
tilesById.delete(screenShareId);
|
}
|
||||||
|
|
||||||
const screenShareTile: TileDescriptor<TileViewModel> = {
|
const userMediaVm =
|
||||||
id: screenShareId,
|
tilesById.get(userMediaId)?.data ??
|
||||||
focused: true,
|
new UserMediaViewModel(userMediaId, member, p, this.encrypted);
|
||||||
isPresenter: false,
|
tilesById.delete(userMediaId);
|
||||||
isSpeaker: false,
|
|
||||||
hasVideo: true,
|
const userMediaTile: TileDescriptor<MediaViewModel> = {
|
||||||
|
id: userMediaId,
|
||||||
|
focused: false,
|
||||||
|
isPresenter: p.isScreenShareEnabled,
|
||||||
|
isSpeaker: (p.isSpeaking || spokeRecently) && !p.isLocal,
|
||||||
|
hasVideo: p.isCameraEnabled,
|
||||||
local: p.isLocal,
|
local: p.isLocal,
|
||||||
largeBaseSize: true,
|
largeBaseSize: false,
|
||||||
placeNear: userMediaId,
|
data: userMediaVm,
|
||||||
data: screenShareVm,
|
|
||||||
};
|
};
|
||||||
return [userMediaTile, screenShareTile];
|
|
||||||
} else {
|
|
||||||
return [userMediaTile];
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Any tiles left in the map are unused and should be destroyed
|
if (p.isScreenShareEnabled) {
|
||||||
for (const t of tilesById.values()) t.data.destroy();
|
const screenShareId = `${userMediaId}:screen-share`;
|
||||||
|
const screenShareVm =
|
||||||
|
tilesById.get(screenShareId)?.data ??
|
||||||
|
new ScreenShareViewModel(
|
||||||
|
screenShareId,
|
||||||
|
member,
|
||||||
|
p,
|
||||||
|
this.encrypted,
|
||||||
|
);
|
||||||
|
tilesById.delete(screenShareId);
|
||||||
|
|
||||||
// If every item is a ghost, that probably means we're still connecting
|
const screenShareTile: TileDescriptor<MediaViewModel> = {
|
||||||
// and shouldn't bother showing anything yet
|
id: screenShareId,
|
||||||
return allGhosts ? [] : newTiles;
|
focused: true,
|
||||||
}, [] as TileDescriptor<TileViewModel>[]),
|
isPresenter: false,
|
||||||
finalizeValue((ts) => {
|
isSpeaker: false,
|
||||||
for (const t of ts) t.data.destroy();
|
hasVideo: true,
|
||||||
}),
|
local: p.isLocal,
|
||||||
),
|
largeBaseSize: true,
|
||||||
);
|
placeNear: userMediaId,
|
||||||
|
data: screenShareVm,
|
||||||
|
};
|
||||||
|
return [userMediaTile, screenShareTile];
|
||||||
|
} else {
|
||||||
|
return [userMediaTile];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Any tiles left in the map are unused and should be destroyed
|
||||||
|
for (const t of tilesById.values()) t.data.destroy();
|
||||||
|
|
||||||
|
// If every item is a ghost, that probably means we're still connecting
|
||||||
|
// and shouldn't bother showing anything yet
|
||||||
|
return allGhosts ? [] : newTiles;
|
||||||
|
}, [] as TileDescriptor<MediaViewModel>[]),
|
||||||
|
finalizeValue((ts) => {
|
||||||
|
for (const t of ts) t.data.destroy();
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
public constructor(
|
public constructor(
|
||||||
// A call is permanently tied to a single Matrix room and LiveKit room
|
// A call is permanently tied to a single Matrix room and LiveKit room
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright 2023 New Vector Ltd
|
Copyright 2023-2024 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.
|
||||||
@@ -43,7 +43,6 @@ import {
|
|||||||
of,
|
of,
|
||||||
startWith,
|
startWith,
|
||||||
switchMap,
|
switchMap,
|
||||||
takeUntil,
|
|
||||||
} from "rxjs";
|
} from "rxjs";
|
||||||
|
|
||||||
import { ViewModel } from "./ViewModel";
|
import { ViewModel } from "./ViewModel";
|
||||||
@@ -64,13 +63,13 @@ function observeTrackReference(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract class BaseTileViewModel extends ViewModel {
|
abstract class BaseMediaViewModel extends ViewModel {
|
||||||
/**
|
/**
|
||||||
* Whether the tile belongs to the local user.
|
* Whether the media belongs to the local user.
|
||||||
*/
|
*/
|
||||||
public readonly local = this.participant.isLocal;
|
public readonly local = this.participant.isLocal;
|
||||||
/**
|
/**
|
||||||
* The LiveKit video track to be shown on this tile.
|
* The LiveKit video track for this media.
|
||||||
*/
|
*/
|
||||||
public readonly video: StateObservable<TrackReferenceOrPlaceholder>;
|
public readonly video: StateObservable<TrackReferenceOrPlaceholder>;
|
||||||
/**
|
/**
|
||||||
@@ -83,7 +82,7 @@ abstract class BaseTileViewModel extends ViewModel {
|
|||||||
// soon as that code is moved into the view models
|
// soon as that code is moved into the view models
|
||||||
public readonly id: string,
|
public readonly id: string,
|
||||||
/**
|
/**
|
||||||
* The Matrix room member to which this tile belongs.
|
* The Matrix room member to which this media belongs.
|
||||||
*/
|
*/
|
||||||
// TODO: Fully separate the data layer from the UI layer by keeping the
|
// TODO: Fully separate the data layer from the UI layer by keeping the
|
||||||
// member object internal
|
// member object internal
|
||||||
@@ -109,14 +108,14 @@ abstract class BaseTileViewModel extends ViewModel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A tile displaying some media.
|
* Some participant's media.
|
||||||
*/
|
*/
|
||||||
export type TileViewModel = UserMediaTileViewModel | ScreenShareTileViewModel;
|
export type MediaViewModel = UserMediaViewModel | ScreenShareViewModel;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A tile displaying some participant's user media.
|
* Some participant's user media.
|
||||||
*/
|
*/
|
||||||
export class UserMediaTileViewModel extends BaseTileViewModel {
|
export class UserMediaViewModel extends BaseMediaViewModel {
|
||||||
/**
|
/**
|
||||||
* Whether the video should be mirrored.
|
* Whether the video should be mirrored.
|
||||||
*/
|
*/
|
||||||
@@ -201,7 +200,7 @@ export class UserMediaTileViewModel extends BaseTileViewModel {
|
|||||||
combineLatest([this._locallyMuted, this._localVolume], (muted, volume) =>
|
combineLatest([this._locallyMuted, this._localVolume], (muted, volume) =>
|
||||||
muted ? 0 : volume,
|
muted ? 0 : volume,
|
||||||
)
|
)
|
||||||
.pipe(takeUntil(this.destroyed))
|
.pipe(this.scope.bind())
|
||||||
.subscribe((volume) => {
|
.subscribe((volume) => {
|
||||||
(this.participant as RemoteParticipant).setVolume(volume);
|
(this.participant as RemoteParticipant).setVolume(volume);
|
||||||
});
|
});
|
||||||
@@ -221,9 +220,9 @@ export class UserMediaTileViewModel extends BaseTileViewModel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A tile displaying some participant's screen share.
|
* Some participant's screen share media.
|
||||||
*/
|
*/
|
||||||
export class ScreenShareTileViewModel extends BaseTileViewModel {
|
export class ScreenShareViewModel extends BaseMediaViewModel {
|
||||||
public constructor(
|
public constructor(
|
||||||
id: string,
|
id: string,
|
||||||
member: RoomMember | undefined,
|
member: RoomMember | undefined,
|
||||||
40
src/state/ObservableScope.ts
Normal file
40
src/state/ObservableScope.ts
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2024 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 { MonoTypeOperatorFunction, Subject, takeUntil } from "rxjs";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A scope which limits the execution lifetime of its bound Observables.
|
||||||
|
*/
|
||||||
|
export class ObservableScope {
|
||||||
|
private readonly ended = new Subject<void>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Binds an Observable to this scope, so that it completes when the scope
|
||||||
|
* ends.
|
||||||
|
*/
|
||||||
|
public bind<T>(): MonoTypeOperatorFunction<T> {
|
||||||
|
return takeUntil(this.ended);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ends the scope, causing any bound Observables to complete.
|
||||||
|
*/
|
||||||
|
public end(): void {
|
||||||
|
this.ended.next();
|
||||||
|
this.ended.complete();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright 2023 New Vector Ltd
|
Copyright 2023-2024 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,20 +14,19 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Subject } from "rxjs";
|
import { ObservableScope } from "./ObservableScope";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An MVVM view model.
|
* An MVVM view model.
|
||||||
*/
|
*/
|
||||||
export abstract class ViewModel {
|
export abstract class ViewModel {
|
||||||
protected readonly destroyed = new Subject<void>();
|
protected readonly scope = new ObservableScope();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Instructs the ViewModel to clean up its resources. If you forget to call
|
* Instructs the ViewModel to clean up its resources. If you forget to call
|
||||||
* this, there may be memory leaks!
|
* this, there may be memory leaks!
|
||||||
*/
|
*/
|
||||||
public destroy(): void {
|
public destroy(): void {
|
||||||
this.destroyed.next();
|
this.scope.end();
|
||||||
this.destroyed.complete();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright 2023 New Vector Ltd
|
Copyright 2023-2024 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,18 +14,17 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { useEffect, useRef } from "react";
|
import { useRef } from "react";
|
||||||
import { BehaviorSubject, Observable } from "rxjs";
|
import { BehaviorSubject, Observable } from "rxjs";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* React hook that creates an Observable from a changing value. The Observable
|
* React hook that creates an Observable from a changing value. The Observable
|
||||||
* replays its current value upon subscription, emits whenever the value
|
* replays its current value upon subscription and emits whenever the value
|
||||||
* changes, and completes when the component is unmounted.
|
* changes.
|
||||||
*/
|
*/
|
||||||
export function useObservable<T>(value: T): Observable<T> {
|
export function useObservable<T>(value: T): Observable<T> {
|
||||||
const subject = useRef<BehaviorSubject<T>>();
|
const subject = useRef<BehaviorSubject<T>>();
|
||||||
subject.current ??= new BehaviorSubject(value);
|
subject.current ??= new BehaviorSubject(value);
|
||||||
if (value !== subject.current.value) subject.current.next(value);
|
if (value !== subject.current.value) subject.current.next(value);
|
||||||
useEffect(() => subject.current!.complete(), []);
|
|
||||||
return subject.current;
|
return subject.current;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ export function useEventTarget<T extends Event>(
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (target) {
|
if (target) {
|
||||||
target.addEventListener(eventType, listener as EventListener, options);
|
target.addEventListener(eventType, listener as EventListener, options);
|
||||||
return () =>
|
return (): void =>
|
||||||
target.removeEventListener(
|
target.removeEventListener(
|
||||||
eventType,
|
eventType,
|
||||||
listener as EventListener,
|
listener as EventListener,
|
||||||
@@ -55,7 +55,7 @@ export function useTypedEventEmitter<
|
|||||||
): void {
|
): void {
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
emitter.on(eventType, listener);
|
emitter.on(eventType, listener);
|
||||||
return () => {
|
return (): void => {
|
||||||
emitter.off(eventType, listener);
|
emitter.off(eventType, listener);
|
||||||
};
|
};
|
||||||
}, [emitter, eventType, listener]);
|
}, [emitter, eventType, listener]);
|
||||||
@@ -72,7 +72,7 @@ export function useEventEmitterThree<
|
|||||||
): void {
|
): void {
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
emitter.on(eventType, listener);
|
emitter.on(eventType, listener);
|
||||||
return () => {
|
return (): void => {
|
||||||
emitter.off(eventType, listener);
|
emitter.off(eventType, listener);
|
||||||
};
|
};
|
||||||
}, [emitter, eventType, listener]);
|
}, [emitter, eventType, listener]);
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ export const useLocalStorage = (
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
localStorageBus.on(key, setValue);
|
localStorageBus.on(key, setValue);
|
||||||
return () => {
|
return (): void => {
|
||||||
localStorageBus.off(key, setValue);
|
localStorageBus.off(key, setValue);
|
||||||
};
|
};
|
||||||
}, [key, setValue]);
|
}, [key, setValue]);
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ export function useLocationNavigation(enabled = false): void {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
if (unblock) {
|
if (unblock) {
|
||||||
|
|||||||
@@ -26,19 +26,22 @@ export function useMatrixRTCSessionJoinState(
|
|||||||
): boolean {
|
): boolean {
|
||||||
const [isJoined, setJoined] = useState(rtcSession.isJoined());
|
const [isJoined, setJoined] = useState(rtcSession.isJoined());
|
||||||
|
|
||||||
const onJoinStateChanged = useCallback(() => {
|
const onJoinStateChanged = useCallback(
|
||||||
logger.info(
|
(isJoined: boolean) => {
|
||||||
`Session in room ${rtcSession.room.roomId} changed to ${
|
logger.info(
|
||||||
rtcSession.isJoined() ? "joined" : "left"
|
`Session in room ${rtcSession.room.roomId} changed to ${
|
||||||
}`,
|
isJoined ? "joined" : "left"
|
||||||
);
|
}`,
|
||||||
setJoined(rtcSession.isJoined());
|
);
|
||||||
}, [rtcSession]);
|
setJoined(isJoined);
|
||||||
|
},
|
||||||
|
[rtcSession],
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
rtcSession.on(MatrixRTCSessionEvent.JoinStateChanged, onJoinStateChanged);
|
rtcSession.on(MatrixRTCSessionEvent.JoinStateChanged, onJoinStateChanged);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
rtcSession.off(
|
rtcSession.off(
|
||||||
MatrixRTCSessionEvent.JoinStateChanged,
|
MatrixRTCSessionEvent.JoinStateChanged,
|
||||||
onJoinStateChanged,
|
onJoinStateChanged,
|
||||||
|
|||||||
@@ -40,7 +40,7 @@ export function useMatrixRTCSessionMemberships(
|
|||||||
onMembershipsChanged,
|
onMembershipsChanged,
|
||||||
);
|
);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
rtcSession.off(
|
rtcSession.off(
|
||||||
MatrixRTCSessionEvent.MembershipsChanged,
|
MatrixRTCSessionEvent.MembershipsChanged,
|
||||||
onMembershipsChanged,
|
onMembershipsChanged,
|
||||||
|
|||||||
@@ -22,10 +22,9 @@ export const useTheme = (): void => {
|
|||||||
const { theme: themeName } = useUrlParams();
|
const { theme: themeName } = useUrlParams();
|
||||||
const previousTheme = useRef<string | null>(document.body.classList.item(0));
|
const previousTheme = useRef<string | null>(document.body.classList.item(0));
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
// Don't update the current theme if the url does not contain a theme prop.
|
// If the url does not contain a theme props we default to "dark".
|
||||||
if (!themeName) return;
|
const theme = themeName?.includes("light") ? "light" : "dark";
|
||||||
const theme = themeName.includes("light") ? "light" : "dark";
|
const themeHighContrast = themeName?.includes("high-contrast") ? "-hc" : "";
|
||||||
const themeHighContrast = themeName.includes("high-contrast") ? "-hc" : "";
|
|
||||||
const themeString = "cpd-theme-" + theme + themeHighContrast;
|
const themeString = "cpd-theme-" + theme + themeHighContrast;
|
||||||
if (themeString !== previousTheme.current) {
|
if (themeString !== previousTheme.current) {
|
||||||
document.body.classList.remove(
|
document.body.classList.remove(
|
||||||
@@ -37,5 +36,6 @@ export const useTheme = (): void => {
|
|||||||
document.body.classList.add(themeString);
|
document.body.classList.add(themeString);
|
||||||
previousTheme.current = themeString;
|
previousTheme.current = themeString;
|
||||||
}
|
}
|
||||||
|
document.body.classList.remove("no-theme");
|
||||||
}, [previousTheme, themeName]);
|
}, [previousTheme, themeName]);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -27,8 +27,8 @@ export function useWakeLock(): void {
|
|||||||
let lock: WakeLockSentinel | null = null;
|
let lock: WakeLockSentinel | null = null;
|
||||||
|
|
||||||
// The lock is automatically released whenever the window goes invisible,
|
// The lock is automatically released whenever the window goes invisible,
|
||||||
// so we need to reacquire it on visiblity changes
|
// so we need to reacquire it on visibility changes
|
||||||
const onVisiblityChange = async (): Promise<void> => {
|
const onVisibilityChange = async (): Promise<void> => {
|
||||||
if (document.visibilityState === "visible") {
|
if (document.visibilityState === "visible") {
|
||||||
try {
|
try {
|
||||||
lock = await navigator.wakeLock.request("screen");
|
lock = await navigator.wakeLock.request("screen");
|
||||||
@@ -44,16 +44,16 @@ export function useWakeLock(): void {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
onVisiblityChange();
|
onVisibilityChange();
|
||||||
document.addEventListener("visiblitychange", onVisiblityChange);
|
document.addEventListener("visibilitychange", onVisibilityChange);
|
||||||
|
|
||||||
return () => {
|
return (): void => {
|
||||||
mounted = false;
|
mounted = false;
|
||||||
if (lock !== null)
|
if (lock !== null)
|
||||||
lock
|
lock
|
||||||
.release()
|
.release()
|
||||||
.catch((e) => logger.warn("Can't release wake lock", e));
|
.catch((e) => logger.warn("Can't release wake lock", e));
|
||||||
document.removeEventListener("visiblitychange", onVisiblityChange);
|
document.removeEventListener("visibilitychange", onVisibilityChange);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user