Compare commits
No commits in common. "dev" and "upstream-sync-202605161749" have entirely different histories.
dev
...
upstream-s
20 changed files with 9957 additions and 4541 deletions
|
|
@ -12,7 +12,7 @@ jobs:
|
||||||
- name: Checkout Local Repository
|
- name: Checkout Local Repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0 # 전체 커밋 히스토리를 가져와 비교 및 머지가 가능하도록 설정
|
||||||
|
|
||||||
- name: Configure Git
|
- name: Configure Git
|
||||||
run: |
|
run: |
|
||||||
|
|
@ -24,22 +24,23 @@ jobs:
|
||||||
git remote add upstream https://github.com/cinnyapp/cinny.git
|
git remote add upstream https://github.com/cinnyapp/cinny.git
|
||||||
git fetch upstream dev
|
git fetch upstream dev
|
||||||
|
|
||||||
- name: Merge Upstream and Check Changes
|
- name: Check for Changes and Push Branch
|
||||||
id: check_changes
|
id: check_changes
|
||||||
run: |
|
run: |
|
||||||
BRANCH_NAME="upstream-sync-$(date +'%Y%m%d%H%M')"
|
if ! git diff --quiet HEAD upstream/dev; then
|
||||||
git checkout -b $BRANCH_NAME origin/dev
|
echo "New changes detected in upstream."
|
||||||
|
|
||||||
git merge upstream/dev --no-edit --allow-unrelated-histories
|
BRANCH_NAME="upstream-sync-$(date +'%Y%m%d%H%M')"
|
||||||
|
|
||||||
if ! git diff --quiet origin/dev HEAD; then
|
|
||||||
echo "New upstream changes merged successfully."
|
|
||||||
echo "branch_name=$BRANCH_NAME" >> $GITHUB_OUTPUT
|
echo "branch_name=$BRANCH_NAME" >> $GITHUB_OUTPUT
|
||||||
echo "has_changes=true" >> $GITHUB_OUTPUT
|
echo "has_changes=true" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
git checkout -b $BRANCH_NAME
|
||||||
|
# --allow-unrelated-histories 플래그를 추가하여 독립된 히스토리 간의 병합을 허용
|
||||||
|
git merge upstream/dev --no-edit --allow-unrelated-histories
|
||||||
|
|
||||||
git push origin $BRANCH_NAME
|
git push origin $BRANCH_NAME
|
||||||
else
|
else
|
||||||
echo "No new upstream changes to apply. Everything is up to date."
|
echo "No changes detected. Everything is up to date."
|
||||||
echo "has_changes=false" >> $GITHUB_OUTPUT
|
echo "has_changes=false" >> $GITHUB_OUTPUT
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|
@ -47,7 +48,7 @@ jobs:
|
||||||
if: ${{ steps.check_changes.outputs.has_changes == 'true' }}
|
if: ${{ steps.check_changes.outputs.has_changes == 'true' }}
|
||||||
run: |
|
run: |
|
||||||
PR_TITLE="chore(deps): sync upstream changes ($(date +'%Y-%m-%d'))"
|
PR_TITLE="chore(deps): sync upstream changes ($(date +'%Y-%m-%d'))"
|
||||||
PR_BODY="Upstream (cinnyapp/cinny)에 새로운 변경사항이 감지되어 내 커스텀 브랜치(dev)에 병합한 후 생성된 PR입니다."
|
PR_BODY="Upstream (cinnyapp/cinny)에 새로운 변경사항이 감지되어 자동으로 생성된 PR입니다."
|
||||||
HEAD_BRANCH="${{ steps.check_changes.outputs.branch_name }}"
|
HEAD_BRANCH="${{ steps.check_changes.outputs.branch_name }}"
|
||||||
BASE_BRANCH="dev"
|
BASE_BRANCH="dev"
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,23 +1,39 @@
|
||||||
name: Production deploy
|
name: Production deploy
|
||||||
|
|
||||||
on:
|
on:
|
||||||
release:
|
workflow_dispatch:
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
deploy-and-tarball:
|
deploy-and-tarball:
|
||||||
name: Netlify deploy and tarball
|
name: Netlify deploy and tarball
|
||||||
|
outputs:
|
||||||
|
version: ${{ steps.vars.outputs.tag }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
- name: Setup node
|
- name: Setup node
|
||||||
uses: actions/setup-node@48b55a011bda9f5d6aeb4c2d9c7362e8dae4041e # v6.4.0
|
uses: actions/setup-node@48b55a011bda9f5d6aeb4c2d9c7362e8dae4041e # v6.4.0
|
||||||
with:
|
with:
|
||||||
node-version-file: '.node-version'
|
node-version-file: ".node-version"
|
||||||
package-manager-cache: false
|
package-manager-cache: false
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: npm ci
|
run: npm ci
|
||||||
|
- name: Run semantic release
|
||||||
|
run: npm run semantic-release
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
GIT_AUTHOR_NAME: ${{ secrets.GIT_AUTHOR_NAME }}
|
||||||
|
GIT_AUTHOR_EMAIL: ${{ secrets.GIT_AUTHOR_EMAIL }}
|
||||||
|
GIT_COMMITTER_NAME: ${{ secrets.GIT_AUTHOR_NAME }}
|
||||||
|
GIT_COMMITTER_EMAIL: ${{ secrets.GIT_AUTHOR_EMAIL }}
|
||||||
|
- name: Get version from tag
|
||||||
|
id: vars
|
||||||
|
run: |
|
||||||
|
TAG=$(git describe --tags --abbrev=0)
|
||||||
|
echo "tag=$TAG" >> $GITHUB_OUTPUT
|
||||||
- name: Build app
|
- name: Build app
|
||||||
env:
|
env:
|
||||||
NODE_OPTIONS: '--max_old_space_size=4096'
|
NODE_OPTIONS: '--max_old_space_size=4096'
|
||||||
|
|
@ -26,7 +42,7 @@ jobs:
|
||||||
uses: nwtgck/actions-netlify@4cbaf4c08f1a7bfa537d6113472ef4424e4eb654 # v3.0.0
|
uses: nwtgck/actions-netlify@4cbaf4c08f1a7bfa537d6113472ef4424e4eb654 # v3.0.0
|
||||||
with:
|
with:
|
||||||
publish-dir: dist
|
publish-dir: dist
|
||||||
deploy-message: 'Prod deploy ${{ github.ref_name }}'
|
deploy-message: 'Prod deploy ${{ steps.vars.outputs.tag }}'
|
||||||
enable-commit-comment: false
|
enable-commit-comment: false
|
||||||
github-token: ${{ secrets.GITHUB_TOKEN }}
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
production-deploy: true
|
production-deploy: true
|
||||||
|
|
@ -36,9 +52,6 @@ jobs:
|
||||||
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
||||||
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID_APP }}
|
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID_APP }}
|
||||||
timeout-minutes: 1
|
timeout-minutes: 1
|
||||||
- name: Get version from tag
|
|
||||||
id: vars
|
|
||||||
run: echo "tag=${GITHUB_REF#refs/*/}" >> $GITHUB_OUTPUT
|
|
||||||
- name: Create tar.gz
|
- name: Create tar.gz
|
||||||
run: tar -czvf cinny-${{ steps.vars.outputs.tag }}.tar.gz dist
|
run: tar -czvf cinny-${{ steps.vars.outputs.tag }}.tar.gz dist
|
||||||
- name: Sign tar.gz
|
- name: Sign tar.gz
|
||||||
|
|
@ -54,12 +67,16 @@ jobs:
|
||||||
- name: Upload tagged release
|
- name: Upload tagged release
|
||||||
uses: softprops/action-gh-release@b4309332981a82ec1c5618f44dd2e27cc8bfbfda # v3.0.0
|
uses: softprops/action-gh-release@b4309332981a82ec1c5618f44dd2e27cc8bfbfda # v3.0.0
|
||||||
with:
|
with:
|
||||||
|
tag_name: ${{ steps.vars.outputs.tag }}
|
||||||
files: |
|
files: |
|
||||||
cinny-${{ steps.vars.outputs.tag }}.tar.gz
|
cinny-${{ steps.vars.outputs.tag }}.tar.gz
|
||||||
cinny-${{ steps.vars.outputs.tag }}.tar.gz.asc
|
cinny-${{ steps.vars.outputs.tag }}.tar.gz.asc
|
||||||
|
|
||||||
publish-image:
|
publish-image:
|
||||||
name: Push Docker image to Docker Hub, GHCR
|
name: Push Docker image to Docker Hub, GHCR
|
||||||
|
needs: deploy-and-tarball
|
||||||
|
env:
|
||||||
|
VERSION: ${{ needs.deploy-and-tarball.outputs.version }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
|
|
@ -67,6 +84,8 @@ jobs:
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
- name: Set up QEMU
|
- name: Set up QEMU
|
||||||
uses: docker/setup-qemu-action@c7c53464625b32c7a7e944ae62b3e17d2b600130 # v3.7.0
|
uses: docker/setup-qemu-action@c7c53464625b32c7a7e944ae62b3e17d2b600130 # v3.7.0
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
|
|
@ -89,6 +108,9 @@ jobs:
|
||||||
images: |
|
images: |
|
||||||
${{ secrets.DOCKER_USERNAME }}/cinny
|
${{ secrets.DOCKER_USERNAME }}/cinny
|
||||||
ghcr.io/${{ github.repository }}
|
ghcr.io/${{ github.repository }}
|
||||||
|
tags: |
|
||||||
|
type=raw,value=${{ env.VERSION }}
|
||||||
|
type=raw,value=latest
|
||||||
- name: Build and push Docker image
|
- name: Build and push Docker image
|
||||||
uses: docker/build-push-action@bcafcacb16a39f128d818304e6c9c0c18556b85f # v7.1.0
|
uses: docker/build-push-action@bcafcacb16a39f128d818304e6c9c0c18556b85f # v7.1.0
|
||||||
with:
|
with:
|
||||||
|
|
@ -96,4 +118,4 @@ jobs:
|
||||||
platforms: linux/amd64,linux/arm64
|
platforms: linux/amd64,linux/arm64
|
||||||
push: true
|
push: true
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
5438
package-lock.json
generated
5438
package-lock.json
generated
File diff suppressed because it is too large
Load diff
38
package.json
38
package.json
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "cinny",
|
"name": "cinny",
|
||||||
"version": "4.12.2",
|
"version": "4.12.1",
|
||||||
"description": "Yet another matrix client",
|
"description": "Yet another matrix client",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|
@ -18,7 +18,7 @@
|
||||||
"typecheck": "tsc --noEmit",
|
"typecheck": "tsc --noEmit",
|
||||||
"prepare": "husky install",
|
"prepare": "husky install",
|
||||||
"commit": "git-cz",
|
"commit": "git-cz",
|
||||||
"bump": "node scripts/update-version.js"
|
"semantic-release": "semantic-release"
|
||||||
},
|
},
|
||||||
"lint-staged": {
|
"lint-staged": {
|
||||||
"*.{ts,tsx,js,jsx}": "eslint",
|
"*.{ts,tsx,js,jsx}": "eslint",
|
||||||
|
|
@ -29,6 +29,35 @@
|
||||||
"path": "./node_modules/cz-conventional-changelog"
|
"path": "./node_modules/cz-conventional-changelog"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"release": {
|
||||||
|
"branches": [
|
||||||
|
"dev"
|
||||||
|
],
|
||||||
|
"plugins": [
|
||||||
|
"@semantic-release/commit-analyzer",
|
||||||
|
"@semantic-release/release-notes-generator",
|
||||||
|
[
|
||||||
|
"@semantic-release/exec",
|
||||||
|
{
|
||||||
|
"prepareCmd": "node scripts/update-version.js ${nextRelease.version}"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"@semantic-release/git",
|
||||||
|
{
|
||||||
|
"assets": [
|
||||||
|
"package.json",
|
||||||
|
"package-lock.json",
|
||||||
|
"src/app/features/settings/about/About.tsx",
|
||||||
|
"src/app/pages/auth/AuthFooter.tsx",
|
||||||
|
"src/app/pages/client/WelcomePage.tsx"
|
||||||
|
],
|
||||||
|
"message": "chore(release): ${nextRelease.version} [skip ci]"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"@semantic-release/github"
|
||||||
|
]
|
||||||
|
},
|
||||||
"keywords": [],
|
"keywords": [],
|
||||||
"author": "Ajay Bura",
|
"author": "Ajay Bura",
|
||||||
"license": "AGPL-3.0-only",
|
"license": "AGPL-3.0-only",
|
||||||
|
|
@ -67,7 +96,7 @@
|
||||||
"jotai": "2.6.0",
|
"jotai": "2.6.0",
|
||||||
"linkify-react": "4.3.2",
|
"linkify-react": "4.3.2",
|
||||||
"linkifyjs": "4.3.2",
|
"linkifyjs": "4.3.2",
|
||||||
"matrix-js-sdk": "41.5.0",
|
"matrix-js-sdk": "38.2.0",
|
||||||
"matrix-widget-api": "1.16.1",
|
"matrix-widget-api": "1.16.1",
|
||||||
"millify": "6.1.0",
|
"millify": "6.1.0",
|
||||||
"pdfjs-dist": "4.2.67",
|
"pdfjs-dist": "4.2.67",
|
||||||
|
|
@ -94,6 +123,8 @@
|
||||||
"@esbuild-plugins/node-globals-polyfill": "0.2.3",
|
"@esbuild-plugins/node-globals-polyfill": "0.2.3",
|
||||||
"@rollup/plugin-inject": "5.0.3",
|
"@rollup/plugin-inject": "5.0.3",
|
||||||
"@rollup/plugin-wasm": "6.1.1",
|
"@rollup/plugin-wasm": "6.1.1",
|
||||||
|
"@semantic-release/exec": "7.1.0",
|
||||||
|
"@semantic-release/git": "10.0.1",
|
||||||
"@types/chroma-js": "3.1.1",
|
"@types/chroma-js": "3.1.1",
|
||||||
"@types/file-saver": "2.0.5",
|
"@types/file-saver": "2.0.5",
|
||||||
"@types/is-hotkey": "0.1.10",
|
"@types/is-hotkey": "0.1.10",
|
||||||
|
|
@ -119,6 +150,7 @@
|
||||||
"husky": "9.1.7",
|
"husky": "9.1.7",
|
||||||
"lint-staged": "16.3.2",
|
"lint-staged": "16.3.2",
|
||||||
"prettier": "2.8.1",
|
"prettier": "2.8.1",
|
||||||
|
"semantic-release": "25.0.3",
|
||||||
"typescript": "4.9.4",
|
"typescript": "4.9.4",
|
||||||
"vite": "5.4.19",
|
"vite": "5.4.19",
|
||||||
"vite-plugin-pwa": "0.20.5",
|
"vite-plugin-pwa": "0.20.5",
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
/* eslint-disable jsx-a11y/media-has-caption */
|
/* eslint-disable jsx-a11y/media-has-caption */
|
||||||
import React, { ReactNode, useCallback, useEffect, useRef, useState } from 'react';
|
import React, { ReactNode, useCallback, useEffect, useRef, useState } from 'react';
|
||||||
import { useAtomValue, useSetAtom } from 'jotai';
|
import { useAtomValue, useSetAtom } from 'jotai';
|
||||||
|
import { MatrixRTCSession } from 'matrix-js-sdk/lib/matrixrtc/MatrixRTCSession';
|
||||||
import FocusTrap from 'focus-trap-react';
|
import FocusTrap from 'focus-trap-react';
|
||||||
import {
|
import {
|
||||||
Avatar,
|
Avatar,
|
||||||
|
|
@ -92,14 +93,12 @@ function IncomingCall({ dm, info, onIgnore, onAnswer, onReject }: IncomingCallPr
|
||||||
const session = useCallSession(room);
|
const session = useCallSession(room);
|
||||||
useCallMembersChange(
|
useCallMembersChange(
|
||||||
session,
|
session,
|
||||||
useCallback(
|
useCallback(() => {
|
||||||
(members) => {
|
const members = MatrixRTCSession.sessionMembershipsForRoom(room, session.sessionDescription);
|
||||||
if (members.length === 0) {
|
if (members.length === 0) {
|
||||||
onIgnore();
|
onIgnore();
|
||||||
}
|
}
|
||||||
},
|
}, [room, session, onIgnore])
|
||||||
[onIgnore]
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const playSound = useCallback(() => {
|
const playSound = useCallback(() => {
|
||||||
|
|
@ -265,8 +264,7 @@ function IncomingCallListener({ callEmbed, joined }: IncomingCallListenerProps)
|
||||||
const refEventId = relation?.event_id;
|
const refEventId = relation?.event_id;
|
||||||
|
|
||||||
const mention =
|
const mention =
|
||||||
content['m.mentions']?.room ||
|
content['m.mentions'].room || content['m.mentions'].user_ids?.includes(mx.getSafeUserId());
|
||||||
content['m.mentions']?.user_ids?.includes(mx.getSafeUserId());
|
|
||||||
if (!sender || !refEventId || !mention || Date.now() >= senderTs + lifetime) {
|
if (!sender || !refEventId || !mention || Date.now() >= senderTs + lifetime) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -22,7 +22,7 @@ export function CallStatus({ callEmbed }: CallStatusProps) {
|
||||||
const { room } = callEmbed;
|
const { room } = callEmbed;
|
||||||
|
|
||||||
const callSession = useCallSession(room);
|
const callSession = useCallSession(room);
|
||||||
const callMembers = useCallMembers(callSession);
|
const callMembers = useCallMembers(room, callSession);
|
||||||
const screenSize = useScreenSize();
|
const screenSize = useScreenSize();
|
||||||
const callJoined = useCallJoined(callEmbed);
|
const callJoined = useCallJoined(callEmbed);
|
||||||
const speakers = useCallSpeakers(callEmbed);
|
const speakers = useCallSpeakers(callEmbed);
|
||||||
|
|
|
||||||
|
|
@ -82,7 +82,7 @@ export function LiveChip({ count, room, members }: LiveChipProps) {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<MenuItem
|
<MenuItem
|
||||||
key={callMember.memberId}
|
key={callMember.membershipID}
|
||||||
size="400"
|
size="400"
|
||||||
variant="Surface"
|
variant="Surface"
|
||||||
radii="300"
|
radii="300"
|
||||||
|
|
|
||||||
|
|
@ -29,7 +29,7 @@ export function MemberGlance({ room, members, speakers, max = 6 }: MemberGlanceP
|
||||||
return (
|
return (
|
||||||
<Box alignItems="Center">
|
<Box alignItems="Center">
|
||||||
{visibleMembers.map((callMember) => {
|
{visibleMembers.map((callMember) => {
|
||||||
const { userId } = callMember;
|
const userId = callMember.sender;
|
||||||
if (!userId) return null;
|
if (!userId) return null;
|
||||||
const name = getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId;
|
const name = getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId;
|
||||||
const avatarMxc = getMemberAvatarMxc(room, userId);
|
const avatarMxc = getMemberAvatarMxc(room, userId);
|
||||||
|
|
@ -39,7 +39,7 @@ export function MemberGlance({ room, members, speakers, max = 6 }: MemberGlanceP
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<StackedAvatar
|
<StackedAvatar
|
||||||
key={callMember.memberId}
|
key={callMember.membershipID}
|
||||||
className={speakers.has(callMember.sender) ? css.SpeakerAvatarOutline : undefined}
|
className={speakers.has(callMember.sender) ? css.SpeakerAvatarOutline : undefined}
|
||||||
title={name}
|
title={name}
|
||||||
as="button"
|
as="button"
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { CallMembership } from 'matrix-js-sdk/lib/matrixrtc/CallMembership';
|
import { CallMembership, SessionMembershipData } from 'matrix-js-sdk/lib/matrixrtc/CallMembership';
|
||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { Avatar, Box, Icon, Icons, Text } from 'folds';
|
import { Avatar, Box, Icon, Icons, Text } from 'folds';
|
||||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
|
import { useMatrixClient } from '../../hooks/useMatrixClient';
|
||||||
|
|
@ -12,6 +12,12 @@ import { UserAvatar } from '../../components/user-avatar';
|
||||||
import { getMouseEventCords } from '../../utils/dom';
|
import { getMouseEventCords } from '../../utils/dom';
|
||||||
import * as css from './styles.css';
|
import * as css from './styles.css';
|
||||||
|
|
||||||
|
interface MemberWithMembershipData {
|
||||||
|
membershipData?: SessionMembershipData & {
|
||||||
|
'm.call.intent': 'video' | 'audio';
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
type CallMemberCardProps = {
|
type CallMemberCardProps = {
|
||||||
member: CallMembership;
|
member: CallMembership;
|
||||||
};
|
};
|
||||||
|
|
@ -22,7 +28,7 @@ export function CallMemberCard({ member }: CallMemberCardProps) {
|
||||||
|
|
||||||
const openUserProfile = useOpenUserRoomProfile();
|
const openUserProfile = useOpenUserRoomProfile();
|
||||||
|
|
||||||
const { userId } = member;
|
const userId = member.sender;
|
||||||
if (!userId) return null;
|
if (!userId) return null;
|
||||||
|
|
||||||
const name = getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId;
|
const name = getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId;
|
||||||
|
|
@ -31,12 +37,13 @@ export function CallMemberCard({ member }: CallMemberCardProps) {
|
||||||
? mxcUrlToHttp(mx, avatarMxc, useAuthentication, 96, 96) ?? undefined
|
? mxcUrlToHttp(mx, avatarMxc, useAuthentication, 96, 96) ?? undefined
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
const audioOnly = member.callIntent === 'audio';
|
const audioOnly =
|
||||||
|
(member as unknown as MemberWithMembershipData).membershipData?.['m.call.intent'] === 'audio';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SequenceCard
|
<SequenceCard
|
||||||
as="button"
|
as="button"
|
||||||
key={member.memberId}
|
key={member.membershipID}
|
||||||
className={css.CallMemberCard}
|
className={css.CallMemberCard}
|
||||||
variant="SurfaceVariant"
|
variant="SurfaceVariant"
|
||||||
radii="500"
|
radii="500"
|
||||||
|
|
@ -85,7 +92,7 @@ export function CallMemberRenderer({
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{truncatedMembers.map((member) => (
|
{truncatedMembers.map((member) => (
|
||||||
<CallMemberCard key={member.memberId} member={member} />
|
<CallMemberCard key={member.membershipID} member={member} />
|
||||||
))}
|
))}
|
||||||
{members.length > max && (
|
{members.length > max && (
|
||||||
<SequenceCard
|
<SequenceCard
|
||||||
|
|
|
||||||
|
|
@ -90,7 +90,7 @@ function CallPrescreen() {
|
||||||
);
|
);
|
||||||
|
|
||||||
const callSession = useCallSession(room);
|
const callSession = useCallSession(room);
|
||||||
const callMembers = useCallMembers(callSession);
|
const callMembers = useCallMembers(room, callSession);
|
||||||
const hasParticipant = callMembers.length > 0;
|
const hasParticipant = callMembers.length > 0;
|
||||||
|
|
||||||
const callEmbed = useCallEmbed();
|
const callEmbed = useCallEmbed();
|
||||||
|
|
|
||||||
|
|
@ -282,7 +282,7 @@ export function RoomNavItem({
|
||||||
|
|
||||||
const optionsVisible = hover || !!menuAnchor;
|
const optionsVisible = hover || !!menuAnchor;
|
||||||
const callSession = useCallSession(room);
|
const callSession = useCallSession(room);
|
||||||
const callMembers = useCallMembers(callSession);
|
const callMembers = useCallMembers(room, callSession);
|
||||||
const startCall = useCallStart(direct);
|
const startCall = useCallStart(direct);
|
||||||
const callEmbed = useCallEmbed();
|
const callEmbed = useCallEmbed();
|
||||||
const callPref = useAtomValue(useCallPreferencesAtom());
|
const callPref = useAtomValue(useCallPreferencesAtom());
|
||||||
|
|
|
||||||
|
|
@ -27,7 +27,7 @@ export function Room() {
|
||||||
const mx = useMatrixClient();
|
const mx = useMatrixClient();
|
||||||
|
|
||||||
const callSession = useCallSession(room);
|
const callSession = useCallSession(room);
|
||||||
const callMembers = useCallMembers(callSession);
|
const callMembers = useCallMembers(room, callSession);
|
||||||
const callEmbed = useCallEmbed();
|
const callEmbed = useCallEmbed();
|
||||||
|
|
||||||
const [isDrawer] = useSetting(settingsAtom, 'isPeopleDrawer');
|
const [isDrawer] = useSetting(settingsAtom, 'isPeopleDrawer');
|
||||||
|
|
|
||||||
|
|
@ -27,6 +27,7 @@ import { HTMLReactParserOptions } from 'html-react-parser';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { ReactEditor } from 'slate-react';
|
import { ReactEditor } from 'slate-react';
|
||||||
import { Editor } from 'slate';
|
import { Editor } from 'slate';
|
||||||
|
import { SessionMembershipData } from 'matrix-js-sdk/lib/matrixrtc/CallMembership';
|
||||||
import to from 'await-to-js';
|
import to from 'await-to-js';
|
||||||
import { useAtomValue, useSetAtom } from 'jotai';
|
import { useAtomValue, useSetAtom } from 'jotai';
|
||||||
import {
|
import {
|
||||||
|
|
@ -1474,7 +1475,7 @@ export function RoomTimeline({ room, eventId, roomInputRef, editor }: RoomTimeli
|
||||||
const senderId = mEvent.getSender() ?? '';
|
const senderId = mEvent.getSender() ?? '';
|
||||||
const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
|
const senderName = getMemberDisplayName(room, senderId) || getMxIdLocalPart(senderId);
|
||||||
|
|
||||||
const content = mEvent.getContent();
|
const content = mEvent.getContent<SessionMembershipData>();
|
||||||
const prevContent = mEvent.getPrevContent();
|
const prevContent = mEvent.getPrevContent();
|
||||||
|
|
||||||
const callJoined = content.application;
|
const callJoined = content.application;
|
||||||
|
|
|
||||||
|
|
@ -46,7 +46,7 @@ export function About({ requestClose }: AboutProps) {
|
||||||
<Box direction="Column" gap="100">
|
<Box direction="Column" gap="100">
|
||||||
<Box gap="100" alignItems="End">
|
<Box gap="100" alignItems="End">
|
||||||
<Text size="H3">Cinny</Text>
|
<Text size="H3">Cinny</Text>
|
||||||
<Text size="T200">v4.12.2</Text>
|
<Text size="T200">v4.12.1</Text>
|
||||||
</Box>
|
</Box>
|
||||||
<Text>Yet another matrix client.</Text>
|
<Text>Yet another matrix client.</Text>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,6 @@ import { Room } from 'matrix-js-sdk';
|
||||||
import {
|
import {
|
||||||
MatrixRTCSession,
|
MatrixRTCSession,
|
||||||
MatrixRTCSessionEvent,
|
MatrixRTCSessionEvent,
|
||||||
MatrixRTCSessionEventHandlerMap,
|
|
||||||
} from 'matrix-js-sdk/lib/matrixrtc/MatrixRTCSession';
|
} from 'matrix-js-sdk/lib/matrixrtc/MatrixRTCSession';
|
||||||
import { CallMembership } from 'matrix-js-sdk/lib/matrixrtc/CallMembership';
|
import { CallMembership } from 'matrix-js-sdk/lib/matrixrtc/CallMembership';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
@ -34,27 +33,32 @@ export const useCallSession = (room: Room): MatrixRTCSession => {
|
||||||
return session;
|
return session;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const useCallMembersChange = (
|
export const useCallMembers = (room: Room, session: MatrixRTCSession): CallMembership[] => {
|
||||||
session: MatrixRTCSession,
|
const [memberships, setMemberships] = useState<CallMembership[]>(
|
||||||
callback: (members: CallMembership[]) => void
|
MatrixRTCSession.sessionMembershipsForRoom(room, session.sessionDescription)
|
||||||
): void => {
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const handleMembershipsChange: MatrixRTCSessionEventHandlerMap[MatrixRTCSessionEvent.MembershipsChanged] =
|
const updateMemberships = () => {
|
||||||
(oldestMembership, newMemberships) => {
|
setMemberships(MatrixRTCSession.sessionMembershipsForRoom(room, session.sessionDescription));
|
||||||
callback(newMemberships);
|
|
||||||
};
|
|
||||||
|
|
||||||
session.on(MatrixRTCSessionEvent.MembershipsChanged, handleMembershipsChange);
|
|
||||||
return () => {
|
|
||||||
session.removeListener(MatrixRTCSessionEvent.MembershipsChanged, handleMembershipsChange);
|
|
||||||
};
|
};
|
||||||
}, [session, callback]);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const useCallMembers = (session: MatrixRTCSession): CallMembership[] => {
|
updateMemberships();
|
||||||
const [memberships, setMemberships] = useState<CallMembership[]>(session.memberships);
|
|
||||||
|
|
||||||
useCallMembersChange(session, setMemberships);
|
session.on(MatrixRTCSessionEvent.MembershipsChanged, updateMemberships);
|
||||||
|
return () => {
|
||||||
|
session.removeListener(MatrixRTCSessionEvent.MembershipsChanged, updateMemberships);
|
||||||
|
};
|
||||||
|
}, [session, room]);
|
||||||
|
|
||||||
return memberships;
|
return memberships;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const useCallMembersChange = (session: MatrixRTCSession, callback: () => void): void => {
|
||||||
|
useEffect(() => {
|
||||||
|
session.on(MatrixRTCSessionEvent.MembershipsChanged, callback);
|
||||||
|
return () => {
|
||||||
|
session.removeListener(MatrixRTCSessionEvent.MembershipsChanged, callback);
|
||||||
|
};
|
||||||
|
}, [session, callback]);
|
||||||
|
};
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
import { createContext, RefObject, useCallback, useContext, useEffect, useState } from 'react';
|
import { createContext, RefObject, useCallback, useContext, useEffect, useState } from 'react';
|
||||||
|
import { MatrixRTCSession } from 'matrix-js-sdk/lib/matrixrtc/MatrixRTCSession';
|
||||||
import { MatrixClient, Room } from 'matrix-js-sdk';
|
import { MatrixClient, Room } from 'matrix-js-sdk';
|
||||||
import { useSetAtom } from 'jotai';
|
import { useSetAtom } from 'jotai';
|
||||||
import {
|
import {
|
||||||
|
|
@ -44,7 +45,8 @@ export const createCallEmbed = (
|
||||||
pref?: CallPreferences
|
pref?: CallPreferences
|
||||||
): CallEmbed => {
|
): CallEmbed => {
|
||||||
const rtcSession = mx.matrixRTC.getRoomSession(room);
|
const rtcSession = mx.matrixRTC.getRoomSession(room);
|
||||||
const ongoing = rtcSession.memberships.length > 0;
|
const ongoing =
|
||||||
|
MatrixRTCSession.sessionMembershipsForRoom(room, rtcSession.sessionDescription).length > 0;
|
||||||
|
|
||||||
const intent = CallEmbed.getIntent(dm, ongoing, pref?.video);
|
const intent = CallEmbed.getIntent(dm, ongoing, pref?.video);
|
||||||
const widget = CallEmbed.getWidget(mx, room, intent, themeKind);
|
const widget = CallEmbed.getWidget(mx, room, intent, themeKind);
|
||||||
|
|
|
||||||
|
|
@ -8,7 +8,7 @@ import { useCallJoined } from './useCallEmbed';
|
||||||
export const useCallSpeakers = (callEmbed: CallEmbed): Set<string> => {
|
export const useCallSpeakers = (callEmbed: CallEmbed): Set<string> => {
|
||||||
const [speakers, setSpeakers] = useState(new Set<string>());
|
const [speakers, setSpeakers] = useState(new Set<string>());
|
||||||
const callSession = useCallSession(callEmbed.room);
|
const callSession = useCallSession(callEmbed.room);
|
||||||
const callMembers = useCallMembers(callSession);
|
const callMembers = useCallMembers(callEmbed.room, callSession);
|
||||||
const joined = useCallJoined(callEmbed);
|
const joined = useCallJoined(callEmbed);
|
||||||
|
|
||||||
const videoContainers = useMemo(() => {
|
const videoContainers = useMemo(() => {
|
||||||
|
|
|
||||||
|
|
@ -15,7 +15,7 @@ export function AuthFooter() {
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer"
|
rel="noreferrer"
|
||||||
>
|
>
|
||||||
v4.12.2
|
v4.12.1
|
||||||
</Text>
|
</Text>
|
||||||
<Text as="a" size="T300" href="https://twitter.com/cinnyapp" target="_blank" rel="noreferrer">
|
<Text as="a" size="T300" href="https://twitter.com/cinnyapp" target="_blank" rel="noreferrer">
|
||||||
Twitter
|
Twitter
|
||||||
|
|
|
||||||
|
|
@ -24,7 +24,7 @@ export function WelcomePage() {
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer noopener"
|
rel="noreferrer noopener"
|
||||||
>
|
>
|
||||||
v4.12.2
|
v4.12.1
|
||||||
</a>
|
</a>
|
||||||
</span>
|
</span>
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -8,6 +8,7 @@ import {
|
||||||
type IWidgetApiErrorResponseDataDetails,
|
type IWidgetApiErrorResponseDataDetails,
|
||||||
type ISearchUserDirectoryResult,
|
type ISearchUserDirectoryResult,
|
||||||
type IGetMediaConfigResult,
|
type IGetMediaConfigResult,
|
||||||
|
type UpdateDelayedEventAction,
|
||||||
OpenIDRequestState,
|
OpenIDRequestState,
|
||||||
SimpleObservable,
|
SimpleObservable,
|
||||||
IOpenIDUpdate,
|
IOpenIDUpdate,
|
||||||
|
|
@ -52,11 +53,14 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
stateKey: string | null = null,
|
stateKey: string | null = null,
|
||||||
targetRoomId: string | null = null
|
targetRoomId: string | null = null
|
||||||
): Promise<ISendEventDetails> {
|
): Promise<ISendEventDetails> {
|
||||||
|
const client = this.mx;
|
||||||
const roomId = targetRoomId || this.inRoomId;
|
const roomId = targetRoomId || this.inRoomId;
|
||||||
|
|
||||||
|
if (!client || !roomId) throw new Error('Not in a room or not attached to a client');
|
||||||
|
|
||||||
let r: { event_id: string } | null;
|
let r: { event_id: string } | null;
|
||||||
if (typeof stateKey === 'string') {
|
if (typeof stateKey === 'string') {
|
||||||
r = await this.mx.sendStateEvent(
|
r = await client.sendStateEvent(
|
||||||
roomId,
|
roomId,
|
||||||
eventType as keyof StateEvents,
|
eventType as keyof StateEvents,
|
||||||
content as StateEvents[keyof StateEvents],
|
content as StateEvents[keyof StateEvents],
|
||||||
|
|
@ -64,9 +68,9 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
);
|
);
|
||||||
} else if (eventType === EventType.RoomRedaction) {
|
} else if (eventType === EventType.RoomRedaction) {
|
||||||
// special case: extract the `redacts` property and call redact
|
// special case: extract the `redacts` property and call redact
|
||||||
r = await this.mx.redactEvent(roomId, content.redacts);
|
r = await client.redactEvent(roomId, content.redacts);
|
||||||
} else {
|
} else {
|
||||||
r = await this.mx.sendEvent(
|
r = await client.sendEvent(
|
||||||
roomId,
|
roomId,
|
||||||
eventType as keyof TimelineEvents,
|
eventType as keyof TimelineEvents,
|
||||||
content as TimelineEvents[keyof TimelineEvents]
|
content as TimelineEvents[keyof TimelineEvents]
|
||||||
|
|
@ -84,8 +88,11 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
stateKey: string | null = null,
|
stateKey: string | null = null,
|
||||||
targetRoomId: string | null = null
|
targetRoomId: string | null = null
|
||||||
): Promise<ISendDelayedEventDetails> {
|
): Promise<ISendDelayedEventDetails> {
|
||||||
|
const client = this.mx;
|
||||||
const roomId = targetRoomId || this.inRoomId;
|
const roomId = targetRoomId || this.inRoomId;
|
||||||
|
|
||||||
|
if (!client || !roomId) throw new Error('Not in a room or not attached to a client');
|
||||||
|
|
||||||
let delayOpts;
|
let delayOpts;
|
||||||
if (delay !== null) {
|
if (delay !== null) {
|
||||||
delayOpts = {
|
delayOpts = {
|
||||||
|
|
@ -103,7 +110,7 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
let r: SendDelayedEventResponse | null;
|
let r: SendDelayedEventResponse | null;
|
||||||
if (stateKey !== null) {
|
if (stateKey !== null) {
|
||||||
// state event
|
// state event
|
||||||
r = await this.mx._unstable_sendDelayedStateEvent(
|
r = await client._unstable_sendDelayedStateEvent(
|
||||||
roomId,
|
roomId,
|
||||||
delayOpts,
|
delayOpts,
|
||||||
eventType as keyof StateEvents,
|
eventType as keyof StateEvents,
|
||||||
|
|
@ -112,7 +119,7 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
// message event
|
// message event
|
||||||
r = await this.mx._unstable_sendDelayedEvent(
|
r = await client._unstable_sendDelayedEvent(
|
||||||
roomId,
|
roomId,
|
||||||
delayOpts,
|
delayOpts,
|
||||||
null,
|
null,
|
||||||
|
|
@ -127,16 +134,15 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public async cancelScheduledDelayedEvent(delayId: string): Promise<void> {
|
public async updateDelayedEvent(
|
||||||
await this.mx._unstable_cancelScheduledDelayedEvent(delayId);
|
delayId: string,
|
||||||
}
|
action: UpdateDelayedEventAction
|
||||||
|
): Promise<void> {
|
||||||
|
const client = this.mx;
|
||||||
|
|
||||||
public async restartScheduledDelayedEvent(delayId: string): Promise<void> {
|
if (!client) throw new Error('Not in a room or not attached to a client');
|
||||||
await this.mx._unstable_restartScheduledDelayedEvent(delayId);
|
|
||||||
}
|
|
||||||
|
|
||||||
public async sendScheduledDelayedEvent(delayId: string): Promise<void> {
|
await client._unstable_updateDelayedEvent(delayId, action);
|
||||||
await this.mx._unstable_sendScheduledDelayedEvent(delayId);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async sendToDevice(
|
public async sendToDevice(
|
||||||
|
|
@ -144,8 +150,10 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
encrypted: boolean,
|
encrypted: boolean,
|
||||||
contentMap: { [userId: string]: { [deviceId: string]: object } }
|
contentMap: { [userId: string]: { [deviceId: string]: object } }
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
|
const client = this.mx;
|
||||||
|
|
||||||
if (encrypted) {
|
if (encrypted) {
|
||||||
const crypto = this.mx.getCrypto();
|
const crypto = client.getCrypto();
|
||||||
if (!crypto) throw new Error('E2EE not enabled');
|
if (!crypto) throw new Error('E2EE not enabled');
|
||||||
|
|
||||||
// attempt to re-batch these up into a single request
|
// attempt to re-batch these up into a single request
|
||||||
|
|
@ -171,11 +179,11 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
JSON.parse(stringifiedContent)
|
JSON.parse(stringifiedContent)
|
||||||
);
|
);
|
||||||
|
|
||||||
await this.mx.queueToDevice(batch);
|
await client.queueToDevice(batch);
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
await this.mx.queueToDevice({
|
await client.queueToDevice({
|
||||||
eventType,
|
eventType,
|
||||||
batch: Object.entries(contentMap).flatMap(([userId, userContentMap]) =>
|
batch: Object.entries(contentMap).flatMap(([userId, userContentMap]) =>
|
||||||
Object.entries(userContentMap).map(([deviceId, content]) => ({
|
Object.entries(userContentMap).map(([deviceId, content]) => ({
|
||||||
|
|
@ -255,6 +263,7 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
limit?: number,
|
limit?: number,
|
||||||
direction?: 'f' | 'b'
|
direction?: 'f' | 'b'
|
||||||
): Promise<IReadEventRelationsResult> {
|
): Promise<IReadEventRelationsResult> {
|
||||||
|
const client = this.mx;
|
||||||
const dir = direction as Direction;
|
const dir = direction as Direction;
|
||||||
const targetRoomId = roomId ?? this.inRoomId ?? undefined;
|
const targetRoomId = roomId ?? this.inRoomId ?? undefined;
|
||||||
|
|
||||||
|
|
@ -262,7 +271,7 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
throw new Error('Error while reading the current room');
|
throw new Error('Error while reading the current room');
|
||||||
}
|
}
|
||||||
|
|
||||||
const { events, nextBatch, prevBatch } = await this.mx.relations(
|
const { events, nextBatch, prevBatch } = await client.relations(
|
||||||
targetRoomId,
|
targetRoomId,
|
||||||
eventId,
|
eventId,
|
||||||
relationType ?? null,
|
relationType ?? null,
|
||||||
|
|
@ -281,7 +290,9 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
searchTerm: string,
|
searchTerm: string,
|
||||||
limit?: number
|
limit?: number
|
||||||
): Promise<ISearchUserDirectoryResult> {
|
): Promise<ISearchUserDirectoryResult> {
|
||||||
const { limited, results } = await this.mx.searchUserDirectory({ term: searchTerm, limit });
|
const client = this.mx;
|
||||||
|
|
||||||
|
const { limited, results } = await client.searchUserDirectory({ term: searchTerm, limit });
|
||||||
|
|
||||||
return {
|
return {
|
||||||
limited,
|
limited,
|
||||||
|
|
@ -294,11 +305,15 @@ export class CallWidgetDriver extends WidgetDriver {
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getMediaConfig(): Promise<IGetMediaConfigResult> {
|
public async getMediaConfig(): Promise<IGetMediaConfigResult> {
|
||||||
return this.mx.getMediaConfig();
|
const client = this.mx;
|
||||||
|
|
||||||
|
return client.getMediaConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async uploadFile(file: XMLHttpRequestBodyInit): Promise<{ contentUri: string }> {
|
public async uploadFile(file: XMLHttpRequestBodyInit): Promise<{ contentUri: string }> {
|
||||||
const uploadResult = await this.mx.uploadContent(file);
|
const client = this.mx;
|
||||||
|
|
||||||
|
const uploadResult = await client.uploadContent(file);
|
||||||
|
|
||||||
return { contentUri: uploadResult.content_uri };
|
return { contentUri: uploadResult.content_uri };
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue