chore(deps): sync upstream changes (2026-05-25) #4

Merged
imnyang merged 4 commits from upstream-sync-202605250002 into dev 2026-05-25 14:50:14 +09:00
19 changed files with 4546 additions and 9961 deletions
Showing only changes of commit 259a97ff72 - Show all commits

Merge remote-tracking branch 'upstream/dev' into upstream-sync-202605250002

Akiyama Mizuki 2026-05-25 00:02:01 +00:00

View file

@ -1,39 +1,23 @@
name: Production deploy name: Production deploy
on: on:
workflow_dispatch: release:
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'
@ -42,7 +26,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 ${{ steps.vars.outputs.tag }}' deploy-message: 'Prod deploy ${{ github.ref_name }}'
enable-commit-comment: false enable-commit-comment: false
github-token: ${{ secrets.GITHUB_TOKEN }} github-token: ${{ secrets.GITHUB_TOKEN }}
production-deploy: true production-deploy: true
@ -52,6 +36,9 @@ 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
@ -67,16 +54,12 @@ 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
@ -84,8 +67,6 @@ 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
@ -108,9 +89,6 @@ 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:
@ -118,4 +96,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 }}

5432
package-lock.json generated

File diff suppressed because it is too large Load diff

View file

@ -1,6 +1,6 @@
{ {
"name": "cinny", "name": "cinny",
"version": "4.12.1", "version": "4.12.2",
"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",
"semantic-release": "semantic-release" "bump": "node scripts/update-version.js"
}, },
"lint-staged": { "lint-staged": {
"*.{ts,tsx,js,jsx}": "eslint", "*.{ts,tsx,js,jsx}": "eslint",
@ -29,35 +29,6 @@
"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",
@ -96,7 +67,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": "38.2.0", "matrix-js-sdk": "41.5.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",
@ -123,8 +94,6 @@
"@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",
@ -150,7 +119,6 @@
"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",

View file

@ -1,7 +1,6 @@
/* 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,
@ -93,12 +92,14 @@ function IncomingCall({ dm, info, onIgnore, onAnswer, onReject }: IncomingCallPr
const session = useCallSession(room); const session = useCallSession(room);
useCallMembersChange( useCallMembersChange(
session, session,
useCallback(() => { useCallback(
const members = MatrixRTCSession.sessionMembershipsForRoom(room, session.sessionDescription); (members) => {
if (members.length === 0) { if (members.length === 0) {
onIgnore(); onIgnore();
} }
}, [room, session, onIgnore]) },
[onIgnore]
)
); );
const playSound = useCallback(() => { const playSound = useCallback(() => {
@ -264,7 +265,8 @@ 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'].user_ids?.includes(mx.getSafeUserId()); content['m.mentions']?.room ||
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;
} }

View file

@ -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(room, callSession); const callMembers = useCallMembers(callSession);
const screenSize = useScreenSize(); const screenSize = useScreenSize();
const callJoined = useCallJoined(callEmbed); const callJoined = useCallJoined(callEmbed);
const speakers = useCallSpeakers(callEmbed); const speakers = useCallSpeakers(callEmbed);

View file

@ -82,7 +82,7 @@ export function LiveChip({ count, room, members }: LiveChipProps) {
return ( return (
<MenuItem <MenuItem
key={callMember.membershipID} key={callMember.memberId}
size="400" size="400"
variant="Surface" variant="Surface"
radii="300" radii="300"

View file

@ -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.sender; const { userId } = callMember;
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.membershipID} key={callMember.memberId}
className={speakers.has(callMember.sender) ? css.SpeakerAvatarOutline : undefined} className={speakers.has(callMember.sender) ? css.SpeakerAvatarOutline : undefined}
title={name} title={name}
as="button" as="button"

View file

@ -1,4 +1,4 @@
import { CallMembership, SessionMembershipData } from 'matrix-js-sdk/lib/matrixrtc/CallMembership'; import { CallMembership } 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,12 +12,6 @@ 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;
}; };
@ -28,7 +22,7 @@ export function CallMemberCard({ member }: CallMemberCardProps) {
const openUserProfile = useOpenUserRoomProfile(); const openUserProfile = useOpenUserRoomProfile();
const userId = member.sender; const { userId } = member;
if (!userId) return null; if (!userId) return null;
const name = getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId; const name = getMemberDisplayName(room, userId) ?? getMxIdLocalPart(userId) ?? userId;
@ -37,13 +31,12 @@ export function CallMemberCard({ member }: CallMemberCardProps) {
? mxcUrlToHttp(mx, avatarMxc, useAuthentication, 96, 96) ?? undefined ? mxcUrlToHttp(mx, avatarMxc, useAuthentication, 96, 96) ?? undefined
: undefined; : undefined;
const audioOnly = const audioOnly = member.callIntent === 'audio';
(member as unknown as MemberWithMembershipData).membershipData?.['m.call.intent'] === 'audio';
return ( return (
<SequenceCard <SequenceCard
as="button" as="button"
key={member.membershipID} key={member.memberId}
className={css.CallMemberCard} className={css.CallMemberCard}
variant="SurfaceVariant" variant="SurfaceVariant"
radii="500" radii="500"
@ -92,7 +85,7 @@ export function CallMemberRenderer({
return ( return (
<> <>
{truncatedMembers.map((member) => ( {truncatedMembers.map((member) => (
<CallMemberCard key={member.membershipID} member={member} /> <CallMemberCard key={member.memberId} member={member} />
))} ))}
{members.length > max && ( {members.length > max && (
<SequenceCard <SequenceCard

View file

@ -90,7 +90,7 @@ function CallPrescreen() {
); );
const callSession = useCallSession(room); const callSession = useCallSession(room);
const callMembers = useCallMembers(room, callSession); const callMembers = useCallMembers(callSession);
const hasParticipant = callMembers.length > 0; const hasParticipant = callMembers.length > 0;
const callEmbed = useCallEmbed(); const callEmbed = useCallEmbed();

View file

@ -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(room, callSession); const callMembers = useCallMembers(callSession);
const startCall = useCallStart(direct); const startCall = useCallStart(direct);
const callEmbed = useCallEmbed(); const callEmbed = useCallEmbed();
const callPref = useAtomValue(useCallPreferencesAtom()); const callPref = useAtomValue(useCallPreferencesAtom());

View file

@ -27,7 +27,7 @@ export function Room() {
const mx = useMatrixClient(); const mx = useMatrixClient();
const callSession = useCallSession(room); const callSession = useCallSession(room);
const callMembers = useCallMembers(room, callSession); const callMembers = useCallMembers(callSession);
const callEmbed = useCallEmbed(); const callEmbed = useCallEmbed();
const [isDrawer] = useSetting(settingsAtom, 'isPeopleDrawer'); const [isDrawer] = useSetting(settingsAtom, 'isPeopleDrawer');

View file

@ -27,7 +27,6 @@ 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 {
@ -1475,7 +1474,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<SessionMembershipData>(); const content = mEvent.getContent();
const prevContent = mEvent.getPrevContent(); const prevContent = mEvent.getPrevContent();
const callJoined = content.application; const callJoined = content.application;

View file

@ -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.1</Text> <Text size="T200">v4.12.2</Text>
</Box> </Box>
<Text>Yet another matrix client.</Text> <Text>Yet another matrix client.</Text>
</Box> </Box>

View file

@ -2,6 +2,7 @@ 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';
@ -33,32 +34,27 @@ export const useCallSession = (room: Room): MatrixRTCSession => {
return session; return session;
}; };
export const useCallMembers = (room: Room, session: MatrixRTCSession): CallMembership[] => { export const useCallMembersChange = (
const [memberships, setMemberships] = useState<CallMembership[]>( session: MatrixRTCSession,
MatrixRTCSession.sessionMembershipsForRoom(room, session.sessionDescription) callback: (members: CallMembership[]) => void
); ): void => {
useEffect(() => { useEffect(() => {
const updateMemberships = () => { const handleMembershipsChange: MatrixRTCSessionEventHandlerMap[MatrixRTCSessionEvent.MembershipsChanged] =
setMemberships(MatrixRTCSession.sessionMembershipsForRoom(room, session.sessionDescription)); (oldestMembership, newMemberships) => {
}; callback(newMemberships);
};
updateMemberships(); session.on(MatrixRTCSessionEvent.MembershipsChanged, handleMembershipsChange);
session.on(MatrixRTCSessionEvent.MembershipsChanged, updateMemberships);
return () => { return () => {
session.removeListener(MatrixRTCSessionEvent.MembershipsChanged, updateMemberships); session.removeListener(MatrixRTCSessionEvent.MembershipsChanged, handleMembershipsChange);
};
}, [session, room]);
return memberships;
};
export const useCallMembersChange = (session: MatrixRTCSession, callback: () => void): void => {
useEffect(() => {
session.on(MatrixRTCSessionEvent.MembershipsChanged, callback);
return () => {
session.removeListener(MatrixRTCSessionEvent.MembershipsChanged, callback);
}; };
}, [session, callback]); }, [session, callback]);
}; };
export const useCallMembers = (session: MatrixRTCSession): CallMembership[] => {
const [memberships, setMemberships] = useState<CallMembership[]>(session.memberships);
useCallMembersChange(session, setMemberships);
return memberships;
};

View file

@ -1,5 +1,4 @@
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 {
@ -45,8 +44,7 @@ export const createCallEmbed = (
pref?: CallPreferences pref?: CallPreferences
): CallEmbed => { ): CallEmbed => {
const rtcSession = mx.matrixRTC.getRoomSession(room); const rtcSession = mx.matrixRTC.getRoomSession(room);
const ongoing = const ongoing = rtcSession.memberships.length > 0;
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);

View file

@ -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(callEmbed.room, callSession); const callMembers = useCallMembers(callSession);
const joined = useCallJoined(callEmbed); const joined = useCallJoined(callEmbed);
const videoContainers = useMemo(() => { const videoContainers = useMemo(() => {

View file

@ -15,7 +15,7 @@ export function AuthFooter() {
target="_blank" target="_blank"
rel="noreferrer" rel="noreferrer"
> >
v4.12.1 v4.12.2
</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

View file

@ -24,7 +24,7 @@ export function WelcomePage() {
target="_blank" target="_blank"
rel="noreferrer noopener" rel="noreferrer noopener"
> >
v4.12.1 v4.12.2
</a> </a>
</span> </span>
} }

View file

@ -8,7 +8,6 @@ import {
type IWidgetApiErrorResponseDataDetails, type IWidgetApiErrorResponseDataDetails,
type ISearchUserDirectoryResult, type ISearchUserDirectoryResult,
type IGetMediaConfigResult, type IGetMediaConfigResult,
type UpdateDelayedEventAction,
OpenIDRequestState, OpenIDRequestState,
SimpleObservable, SimpleObservable,
IOpenIDUpdate, IOpenIDUpdate,
@ -53,14 +52,11 @@ 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 client.sendStateEvent( r = await this.mx.sendStateEvent(
roomId, roomId,
eventType as keyof StateEvents, eventType as keyof StateEvents,
content as StateEvents[keyof StateEvents], content as StateEvents[keyof StateEvents],
@ -68,9 +64,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 client.redactEvent(roomId, content.redacts); r = await this.mx.redactEvent(roomId, content.redacts);
} else { } else {
r = await client.sendEvent( r = await this.mx.sendEvent(
roomId, roomId,
eventType as keyof TimelineEvents, eventType as keyof TimelineEvents,
content as TimelineEvents[keyof TimelineEvents] content as TimelineEvents[keyof TimelineEvents]
@ -88,11 +84,8 @@ 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 = {
@ -110,7 +103,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 client._unstable_sendDelayedStateEvent( r = await this.mx._unstable_sendDelayedStateEvent(
roomId, roomId,
delayOpts, delayOpts,
eventType as keyof StateEvents, eventType as keyof StateEvents,
@ -119,7 +112,7 @@ export class CallWidgetDriver extends WidgetDriver {
); );
} else { } else {
// message event // message event
r = await client._unstable_sendDelayedEvent( r = await this.mx._unstable_sendDelayedEvent(
roomId, roomId,
delayOpts, delayOpts,
null, null,
@ -134,15 +127,16 @@ export class CallWidgetDriver extends WidgetDriver {
}; };
} }
public async updateDelayedEvent( public async cancelScheduledDelayedEvent(delayId: string): Promise<void> {
delayId: string, await this.mx._unstable_cancelScheduledDelayedEvent(delayId);
action: UpdateDelayedEventAction }
): Promise<void> {
const client = this.mx;
if (!client) throw new Error('Not in a room or not attached to a client'); public async restartScheduledDelayedEvent(delayId: string): Promise<void> {
await this.mx._unstable_restartScheduledDelayedEvent(delayId);
}
await client._unstable_updateDelayedEvent(delayId, action); public async sendScheduledDelayedEvent(delayId: string): Promise<void> {
await this.mx._unstable_sendScheduledDelayedEvent(delayId);
} }
public async sendToDevice( public async sendToDevice(
@ -150,10 +144,8 @@ 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 = client.getCrypto(); const crypto = this.mx.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
@ -179,11 +171,11 @@ export class CallWidgetDriver extends WidgetDriver {
JSON.parse(stringifiedContent) JSON.parse(stringifiedContent)
); );
await client.queueToDevice(batch); await this.mx.queueToDevice(batch);
}) })
); );
} else { } else {
await client.queueToDevice({ await this.mx.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]) => ({
@ -263,7 +255,6 @@ 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;
@ -271,7 +262,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 client.relations( const { events, nextBatch, prevBatch } = await this.mx.relations(
targetRoomId, targetRoomId,
eventId, eventId,
relationType ?? null, relationType ?? null,
@ -290,9 +281,7 @@ export class CallWidgetDriver extends WidgetDriver {
searchTerm: string, searchTerm: string,
limit?: number limit?: number
): Promise<ISearchUserDirectoryResult> { ): Promise<ISearchUserDirectoryResult> {
const client = this.mx; const { limited, results } = await this.mx.searchUserDirectory({ term: searchTerm, limit });
const { limited, results } = await client.searchUserDirectory({ term: searchTerm, limit });
return { return {
limited, limited,
@ -305,15 +294,11 @@ export class CallWidgetDriver extends WidgetDriver {
} }
public async getMediaConfig(): Promise<IGetMediaConfigResult> { public async getMediaConfig(): Promise<IGetMediaConfigResult> {
const client = this.mx; return this.mx.getMediaConfig();
return client.getMediaConfig();
} }
public async uploadFile(file: XMLHttpRequestBodyInit): Promise<{ contentUri: string }> { public async uploadFile(file: XMLHttpRequestBodyInit): Promise<{ contentUri: string }> {
const client = this.mx; const uploadResult = await this.mx.uploadContent(file);
const uploadResult = await client.uploadContent(file);
return { contentUri: uploadResult.content_uri }; return { contentUri: uploadResult.content_uri };
} }