mirror of
https://github.com/cinnyapp/cinny.git
synced 2025-02-26 15:13:05 +01:00
Add support for spoilers on images (MSC4193) (#2212)
Some checks are pending
Deploy to Netlify (dev) / Deploy to Netlify (push) Waiting to run
Some checks are pending
Deploy to Netlify (dev) / Deploy to Netlify (push) Waiting to run
* Add support for MSC4193: Spoilers on Media * Clarify variable names and wording * Restore list atom * Improve spoilered image UX with autoload off * Use `aria-pressed` to indicate attachment spoiler state * Improve spoiler button tooltip wording, keep reveal button from conflicting with load errors
This commit is contained in:
parent
7c6ab366af
commit
dd4c1a94e6
9 changed files with 158 additions and 19 deletions
|
@ -22,6 +22,8 @@ import {
|
||||||
IThumbnailContent,
|
IThumbnailContent,
|
||||||
IVideoContent,
|
IVideoContent,
|
||||||
IVideoInfo,
|
IVideoInfo,
|
||||||
|
MATRIX_SPOILER_PROPERTY_NAME,
|
||||||
|
MATRIX_SPOILER_REASON_PROPERTY_NAME,
|
||||||
} from '../../../types/matrix/common';
|
} from '../../../types/matrix/common';
|
||||||
import { FALLBACK_MIMETYPE, getBlobSafeMimeType } from '../../utils/mimeTypes';
|
import { FALLBACK_MIMETYPE, getBlobSafeMimeType } from '../../utils/mimeTypes';
|
||||||
import { parseGeoUri, scaleYDimension } from '../../utils/common';
|
import { parseGeoUri, scaleYDimension } from '../../utils/common';
|
||||||
|
@ -177,6 +179,8 @@ type RenderImageContentProps = {
|
||||||
mimeType?: string;
|
mimeType?: string;
|
||||||
url: string;
|
url: string;
|
||||||
encInfo?: IEncryptedFile;
|
encInfo?: IEncryptedFile;
|
||||||
|
markedAsSpoiler?: boolean;
|
||||||
|
spoilerReason?: string;
|
||||||
};
|
};
|
||||||
type MImageProps = {
|
type MImageProps = {
|
||||||
content: IImageContent;
|
content: IImageContent;
|
||||||
|
@ -204,6 +208,8 @@ export function MImage({ content, renderImageContent, outlined }: MImageProps) {
|
||||||
mimeType: imgInfo?.mimetype,
|
mimeType: imgInfo?.mimetype,
|
||||||
url: mxcUrl,
|
url: mxcUrl,
|
||||||
encInfo: content.file,
|
encInfo: content.file,
|
||||||
|
markedAsSpoiler: content[MATRIX_SPOILER_PROPERTY_NAME],
|
||||||
|
spoilerReason: content[MATRIX_SPOILER_REASON_PROPERTY_NAME],
|
||||||
})}
|
})}
|
||||||
</AttachmentBox>
|
</AttachmentBox>
|
||||||
</Attachment>
|
</Attachment>
|
||||||
|
|
|
@ -3,6 +3,7 @@ import {
|
||||||
Badge,
|
Badge,
|
||||||
Box,
|
Box,
|
||||||
Button,
|
Button,
|
||||||
|
Chip,
|
||||||
Icon,
|
Icon,
|
||||||
Icons,
|
Icons,
|
||||||
Modal,
|
Modal,
|
||||||
|
@ -51,6 +52,8 @@ export type ImageContentProps = {
|
||||||
info?: IImageInfo;
|
info?: IImageInfo;
|
||||||
encInfo?: EncryptedAttachmentInfo;
|
encInfo?: EncryptedAttachmentInfo;
|
||||||
autoPlay?: boolean;
|
autoPlay?: boolean;
|
||||||
|
markedAsSpoiler?: boolean;
|
||||||
|
spoilerReason?: string;
|
||||||
renderViewer: (props: RenderViewerProps) => ReactNode;
|
renderViewer: (props: RenderViewerProps) => ReactNode;
|
||||||
renderImage: (props: RenderImageProps) => ReactNode;
|
renderImage: (props: RenderImageProps) => ReactNode;
|
||||||
};
|
};
|
||||||
|
@ -64,6 +67,8 @@ export const ImageContent = as<'div', ImageContentProps>(
|
||||||
info,
|
info,
|
||||||
encInfo,
|
encInfo,
|
||||||
autoPlay,
|
autoPlay,
|
||||||
|
markedAsSpoiler,
|
||||||
|
spoilerReason,
|
||||||
renderViewer,
|
renderViewer,
|
||||||
renderImage,
|
renderImage,
|
||||||
...props
|
...props
|
||||||
|
@ -77,6 +82,7 @@ export const ImageContent = as<'div', ImageContentProps>(
|
||||||
const [load, setLoad] = useState(false);
|
const [load, setLoad] = useState(false);
|
||||||
const [error, setError] = useState(false);
|
const [error, setError] = useState(false);
|
||||||
const [viewer, setViewer] = useState(false);
|
const [viewer, setViewer] = useState(false);
|
||||||
|
const [blurred, setBlurred] = useState(markedAsSpoiler ?? false);
|
||||||
|
|
||||||
const [srcState, loadSrc] = useAsyncCallback(
|
const [srcState, loadSrc] = useAsyncCallback(
|
||||||
useCallback(async () => {
|
useCallback(async () => {
|
||||||
|
@ -145,7 +151,7 @@ export const ImageContent = as<'div', ImageContentProps>(
|
||||||
punch={1}
|
punch={1}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{!autoPlay && srcState.status === AsyncStatus.Idle && (
|
{!autoPlay && !markedAsSpoiler && srcState.status === AsyncStatus.Idle && (
|
||||||
<Box className={css.AbsoluteContainer} alignItems="Center" justifyContent="Center">
|
<Box className={css.AbsoluteContainer} alignItems="Center" justifyContent="Center">
|
||||||
<Button
|
<Button
|
||||||
variant="Secondary"
|
variant="Secondary"
|
||||||
|
@ -160,7 +166,7 @@ export const ImageContent = as<'div', ImageContentProps>(
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
{srcState.status === AsyncStatus.Success && (
|
{srcState.status === AsyncStatus.Success && (
|
||||||
<Box className={css.AbsoluteContainer}>
|
<Box className={classNames(css.AbsoluteContainer, blurred && css.Blur)}>
|
||||||
{renderImage({
|
{renderImage({
|
||||||
alt: body,
|
alt: body,
|
||||||
title: body,
|
title: body,
|
||||||
|
@ -172,8 +178,42 @@ export const ImageContent = as<'div', ImageContentProps>(
|
||||||
})}
|
})}
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
|
{blurred && !error && srcState.status !== AsyncStatus.Error && (
|
||||||
|
<Box className={css.AbsoluteContainer} alignItems="Center" justifyContent="Center">
|
||||||
|
<TooltipProvider
|
||||||
|
tooltip={
|
||||||
|
typeof spoilerReason === 'string' && (
|
||||||
|
<Tooltip variant="Secondary">
|
||||||
|
<Text>{spoilerReason}</Text>
|
||||||
|
</Tooltip>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
position="Top"
|
||||||
|
align="Center"
|
||||||
|
>
|
||||||
|
{(triggerRef) => (
|
||||||
|
<Chip
|
||||||
|
ref={triggerRef}
|
||||||
|
variant="Secondary"
|
||||||
|
radii="Pill"
|
||||||
|
size="500"
|
||||||
|
outlined
|
||||||
|
onClick={() => {
|
||||||
|
setBlurred(false);
|
||||||
|
if (srcState.status === AsyncStatus.Idle) {
|
||||||
|
loadSrc();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Text size="B300">Spoiler</Text>
|
||||||
|
</Chip>
|
||||||
|
)}
|
||||||
|
</TooltipProvider>
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
{(srcState.status === AsyncStatus.Loading || srcState.status === AsyncStatus.Success) &&
|
{(srcState.status === AsyncStatus.Loading || srcState.status === AsyncStatus.Success) &&
|
||||||
!load && (
|
!load &&
|
||||||
|
!markedAsSpoiler && (
|
||||||
<Box className={css.AbsoluteContainer} alignItems="Center" justifyContent="Center">
|
<Box className={css.AbsoluteContainer} alignItems="Center" justifyContent="Center">
|
||||||
<Spinner variant="Secondary" />
|
<Spinner variant="Secondary" />
|
||||||
</Box>
|
</Box>
|
||||||
|
|
|
@ -30,3 +30,10 @@ export const AbsoluteFooter = style([
|
||||||
right: config.space.S100,
|
right: config.space.S100,
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
export const Blur = style([
|
||||||
|
DefaultReset,
|
||||||
|
{
|
||||||
|
filter: 'blur(44px)',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
|
@ -1,29 +1,42 @@
|
||||||
import React, { useEffect } from 'react';
|
import React, { useCallback, useEffect } from 'react';
|
||||||
import { Chip, Icon, IconButton, Icons, Text, color } from 'folds';
|
import { Chip, Icon, IconButton, Icons, Text, Tooltip, TooltipProvider, color } from 'folds';
|
||||||
import { UploadCard, UploadCardError, UploadCardProgress } from './UploadCard';
|
import { UploadCard, UploadCardError, UploadCardProgress } from './UploadCard';
|
||||||
import { TUploadAtom, UploadStatus, UploadSuccess, useBindUploadAtom } from '../../state/upload';
|
import { UploadStatus, UploadSuccess, useBindUploadAtom } from '../../state/upload';
|
||||||
import { useMatrixClient } from '../../hooks/useMatrixClient';
|
import { useMatrixClient } from '../../hooks/useMatrixClient';
|
||||||
import { TUploadContent } from '../../utils/matrix';
|
import { TUploadContent } from '../../utils/matrix';
|
||||||
import { getFileTypeIcon } from '../../utils/common';
|
import { getFileTypeIcon } from '../../utils/common';
|
||||||
|
import {
|
||||||
|
roomUploadAtomFamily,
|
||||||
|
TUploadItem,
|
||||||
|
TUploadMetadata,
|
||||||
|
} from '../../state/room/roomInputDrafts';
|
||||||
|
|
||||||
type UploadCardRendererProps = {
|
type UploadCardRendererProps = {
|
||||||
isEncrypted?: boolean;
|
isEncrypted?: boolean;
|
||||||
uploadAtom: TUploadAtom;
|
fileItem: TUploadItem;
|
||||||
|
setMetadata: (metadata: TUploadMetadata) => void;
|
||||||
onRemove: (file: TUploadContent) => void;
|
onRemove: (file: TUploadContent) => void;
|
||||||
onComplete?: (upload: UploadSuccess) => void;
|
onComplete?: (upload: UploadSuccess) => void;
|
||||||
};
|
};
|
||||||
export function UploadCardRenderer({
|
export function UploadCardRenderer({
|
||||||
isEncrypted,
|
isEncrypted,
|
||||||
uploadAtom,
|
fileItem,
|
||||||
|
setMetadata,
|
||||||
onRemove,
|
onRemove,
|
||||||
onComplete,
|
onComplete,
|
||||||
}: UploadCardRendererProps) {
|
}: UploadCardRendererProps) {
|
||||||
const mx = useMatrixClient();
|
const mx = useMatrixClient();
|
||||||
|
const uploadAtom = roomUploadAtomFamily(fileItem.file);
|
||||||
|
const { metadata } = fileItem;
|
||||||
const { upload, startUpload, cancelUpload } = useBindUploadAtom(mx, uploadAtom, isEncrypted);
|
const { upload, startUpload, cancelUpload } = useBindUploadAtom(mx, uploadAtom, isEncrypted);
|
||||||
const { file } = upload;
|
const { file } = upload;
|
||||||
|
|
||||||
if (upload.status === UploadStatus.Idle) startUpload();
|
if (upload.status === UploadStatus.Idle) startUpload();
|
||||||
|
|
||||||
|
const toggleSpoiler = useCallback(() => {
|
||||||
|
setMetadata({ ...metadata, markedAsSpoiler: !metadata.markedAsSpoiler });
|
||||||
|
}, [setMetadata, metadata]);
|
||||||
|
|
||||||
const removeUpload = () => {
|
const removeUpload = () => {
|
||||||
cancelUpload();
|
cancelUpload();
|
||||||
onRemove(file);
|
onRemove(file);
|
||||||
|
@ -53,6 +66,31 @@ export function UploadCardRenderer({
|
||||||
<Text size="B300">Retry</Text>
|
<Text size="B300">Retry</Text>
|
||||||
</Chip>
|
</Chip>
|
||||||
)}
|
)}
|
||||||
|
{file.type.startsWith('image') && (
|
||||||
|
<TooltipProvider
|
||||||
|
tooltip={
|
||||||
|
<Tooltip variant="SurfaceVariant">
|
||||||
|
<Text>Mark as Spoiler</Text>
|
||||||
|
</Tooltip>
|
||||||
|
}
|
||||||
|
position="Top"
|
||||||
|
align="Center"
|
||||||
|
>
|
||||||
|
{(triggerRef) => (
|
||||||
|
<IconButton
|
||||||
|
ref={triggerRef}
|
||||||
|
onClick={toggleSpoiler}
|
||||||
|
aria-label="Mark as Spoiler"
|
||||||
|
variant="SurfaceVariant"
|
||||||
|
radii="Pill"
|
||||||
|
size="300"
|
||||||
|
aria-pressed={metadata.markedAsSpoiler}
|
||||||
|
>
|
||||||
|
<Icon src={Icons.EyeBlind} size="200" />
|
||||||
|
</IconButton>
|
||||||
|
)}
|
||||||
|
</TooltipProvider>
|
||||||
|
)}
|
||||||
<IconButton
|
<IconButton
|
||||||
onClick={removeUpload}
|
onClick={removeUpload}
|
||||||
aria-label="Cancel Upload"
|
aria-label="Cancel Upload"
|
||||||
|
|
|
@ -167,10 +167,24 @@ export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
|
||||||
const encryptFiles = fulfilledPromiseSettledResult(
|
const encryptFiles = fulfilledPromiseSettledResult(
|
||||||
await Promise.allSettled(safeFiles.map((f) => encryptFile(f)))
|
await Promise.allSettled(safeFiles.map((f) => encryptFile(f)))
|
||||||
);
|
);
|
||||||
encryptFiles.forEach((ef) => fileItems.push(ef));
|
encryptFiles.forEach((ef) =>
|
||||||
|
fileItems.push({
|
||||||
|
...ef,
|
||||||
|
metadata: {
|
||||||
|
markedAsSpoiler: false,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
safeFiles.forEach((f) =>
|
safeFiles.forEach((f) =>
|
||||||
fileItems.push({ file: f, originalFile: f, encInfo: undefined })
|
fileItems.push({
|
||||||
|
file: f,
|
||||||
|
originalFile: f,
|
||||||
|
encInfo: undefined,
|
||||||
|
metadata: {
|
||||||
|
markedAsSpoiler: false,
|
||||||
|
},
|
||||||
|
})
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
setSelectedFiles({
|
setSelectedFiles({
|
||||||
|
@ -420,7 +434,14 @@ export const RoomInput = forwardRef<HTMLDivElement, RoomInputProps>(
|
||||||
// eslint-disable-next-line react/no-array-index-key
|
// eslint-disable-next-line react/no-array-index-key
|
||||||
key={index}
|
key={index}
|
||||||
isEncrypted={!!fileItem.encInfo}
|
isEncrypted={!!fileItem.encInfo}
|
||||||
uploadAtom={roomUploadAtomFamily(fileItem.file)}
|
fileItem={fileItem}
|
||||||
|
setMetadata={(metadata) =>
|
||||||
|
setSelectedFiles({
|
||||||
|
type: 'REPLACE',
|
||||||
|
item: fileItem,
|
||||||
|
replacement: { ...fileItem, metadata },
|
||||||
|
})
|
||||||
|
}
|
||||||
onRemove={handleRemoveUpload}
|
onRemove={handleRemoveUpload}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
|
|
|
@ -1,6 +1,10 @@
|
||||||
import { IContent, MatrixClient, MsgType } from 'matrix-js-sdk';
|
import { IContent, MatrixClient, MsgType } from 'matrix-js-sdk';
|
||||||
import to from 'await-to-js';
|
import to from 'await-to-js';
|
||||||
import { IThumbnailContent, MATRIX_BLUR_HASH_PROPERTY_NAME } from '../../../types/matrix/common';
|
import {
|
||||||
|
IThumbnailContent,
|
||||||
|
MATRIX_BLUR_HASH_PROPERTY_NAME,
|
||||||
|
MATRIX_SPOILER_PROPERTY_NAME,
|
||||||
|
} from '../../../types/matrix/common';
|
||||||
import {
|
import {
|
||||||
getImageFileUrl,
|
getImageFileUrl,
|
||||||
getThumbnail,
|
getThumbnail,
|
||||||
|
@ -42,9 +46,9 @@ const generateThumbnailContent = async (
|
||||||
export const getImageMsgContent = async (
|
export const getImageMsgContent = async (
|
||||||
mx: MatrixClient,
|
mx: MatrixClient,
|
||||||
item: TUploadItem,
|
item: TUploadItem,
|
||||||
mxc: string,
|
mxc: string
|
||||||
): Promise<IContent> => {
|
): Promise<IContent> => {
|
||||||
const { file, originalFile, encInfo } = item;
|
const { file, originalFile, encInfo, metadata } = item;
|
||||||
const [imgError, imgEl] = await to(loadImageElement(getImageFileUrl(originalFile)));
|
const [imgError, imgEl] = await to(loadImageElement(getImageFileUrl(originalFile)));
|
||||||
if (imgError) console.warn(imgError);
|
if (imgError) console.warn(imgError);
|
||||||
|
|
||||||
|
@ -52,6 +56,7 @@ export const getImageMsgContent = async (
|
||||||
msgtype: MsgType.Image,
|
msgtype: MsgType.Image,
|
||||||
filename: file.name,
|
filename: file.name,
|
||||||
body: file.name,
|
body: file.name,
|
||||||
|
[MATRIX_SPOILER_PROPERTY_NAME]: metadata.markedAsSpoiler,
|
||||||
};
|
};
|
||||||
if (imgEl) {
|
if (imgEl) {
|
||||||
const blurHash = encodeBlurHash(imgEl, 512, scaleYDimension(imgEl.width, 512, imgEl.height));
|
const blurHash = encodeBlurHash(imgEl, 512, scaleYDimension(imgEl.width, 512, imgEl.height));
|
||||||
|
@ -75,7 +80,7 @@ export const getImageMsgContent = async (
|
||||||
export const getVideoMsgContent = async (
|
export const getVideoMsgContent = async (
|
||||||
mx: MatrixClient,
|
mx: MatrixClient,
|
||||||
item: TUploadItem,
|
item: TUploadItem,
|
||||||
mxc: string,
|
mxc: string
|
||||||
): Promise<IContent> => {
|
): Promise<IContent> => {
|
||||||
const { file, originalFile, encInfo } = item;
|
const { file, originalFile, encInfo } = item;
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,11 @@ export type ListAction<T> =
|
||||||
type: 'PUT';
|
type: 'PUT';
|
||||||
item: T | T[];
|
item: T | T[];
|
||||||
}
|
}
|
||||||
|
| {
|
||||||
|
type: 'REPLACE';
|
||||||
|
item: T;
|
||||||
|
replacement: T;
|
||||||
|
}
|
||||||
| {
|
| {
|
||||||
type: 'DELETE';
|
type: 'DELETE';
|
||||||
item: T | T[];
|
item: T | T[];
|
||||||
|
@ -26,6 +31,10 @@ export const createListAtom = <T>() => {
|
||||||
}
|
}
|
||||||
if (action.type === 'PUT') {
|
if (action.type === 'PUT') {
|
||||||
set(baseListAtom, [...items, ...newItems]);
|
set(baseListAtom, [...items, ...newItems]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (action.type === 'REPLACE') {
|
||||||
|
set(baseListAtom, items.map((item) => item === action.item ? action.replacement : item));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
|
@ -3,22 +3,29 @@ import { atomFamily } from 'jotai/utils';
|
||||||
import { Descendant } from 'slate';
|
import { Descendant } from 'slate';
|
||||||
import { EncryptedAttachmentInfo } from 'browser-encrypt-attachment';
|
import { EncryptedAttachmentInfo } from 'browser-encrypt-attachment';
|
||||||
import { IEventRelation } from 'matrix-js-sdk';
|
import { IEventRelation } from 'matrix-js-sdk';
|
||||||
import { TListAtom, createListAtom } from '../list';
|
|
||||||
import { createUploadAtomFamily } from '../upload';
|
import { createUploadAtomFamily } from '../upload';
|
||||||
import { TUploadContent } from '../../utils/matrix';
|
import { TUploadContent } from '../../utils/matrix';
|
||||||
|
import { createListAtom } from '../list';
|
||||||
|
|
||||||
export const roomUploadAtomFamily = createUploadAtomFamily();
|
export type TUploadMetadata = {
|
||||||
|
markedAsSpoiler: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
export type TUploadItem = {
|
export type TUploadItem = {
|
||||||
file: TUploadContent;
|
file: TUploadContent;
|
||||||
originalFile: TUploadContent;
|
originalFile: TUploadContent;
|
||||||
|
metadata: TUploadMetadata;
|
||||||
encInfo: EncryptedAttachmentInfo | undefined;
|
encInfo: EncryptedAttachmentInfo | undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const roomIdToUploadItemsAtomFamily = atomFamily<string, TListAtom<TUploadItem>>(
|
export type TUploadListAtom = ReturnType<typeof createListAtom<TUploadItem>>;
|
||||||
|
|
||||||
|
export const roomIdToUploadItemsAtomFamily = atomFamily<string, TUploadListAtom>(
|
||||||
createListAtom
|
createListAtom
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const roomUploadAtomFamily = createUploadAtomFamily();
|
||||||
|
|
||||||
export type RoomIdToMsgAction =
|
export type RoomIdToMsgAction =
|
||||||
| {
|
| {
|
||||||
type: 'PUT';
|
type: 'PUT';
|
||||||
|
|
|
@ -2,6 +2,8 @@ import { EncryptedAttachmentInfo } from 'browser-encrypt-attachment';
|
||||||
import { MsgType } from 'matrix-js-sdk';
|
import { MsgType } from 'matrix-js-sdk';
|
||||||
|
|
||||||
export const MATRIX_BLUR_HASH_PROPERTY_NAME = 'xyz.amorgan.blurhash';
|
export const MATRIX_BLUR_HASH_PROPERTY_NAME = 'xyz.amorgan.blurhash';
|
||||||
|
export const MATRIX_SPOILER_PROPERTY_NAME = 'page.codeberg.everypizza.msc4193.spoiler';
|
||||||
|
export const MATRIX_SPOILER_REASON_PROPERTY_NAME = 'page.codeberg.everypizza.msc4193.spoiler.reason';
|
||||||
|
|
||||||
export type IImageInfo = {
|
export type IImageInfo = {
|
||||||
w?: number;
|
w?: number;
|
||||||
|
@ -47,6 +49,8 @@ export type IImageContent = {
|
||||||
url?: string;
|
url?: string;
|
||||||
info?: IImageInfo & IThumbnailContent;
|
info?: IImageInfo & IThumbnailContent;
|
||||||
file?: IEncryptedFile;
|
file?: IEncryptedFile;
|
||||||
|
[MATRIX_SPOILER_PROPERTY_NAME]?: boolean;
|
||||||
|
[MATRIX_SPOILER_REASON_PROPERTY_NAME]?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type IVideoContent = {
|
export type IVideoContent = {
|
||||||
|
@ -56,6 +60,8 @@ export type IVideoContent = {
|
||||||
url?: string;
|
url?: string;
|
||||||
info?: IVideoInfo & IThumbnailContent;
|
info?: IVideoInfo & IThumbnailContent;
|
||||||
file?: IEncryptedFile;
|
file?: IEncryptedFile;
|
||||||
|
[MATRIX_SPOILER_PROPERTY_NAME]?: boolean;
|
||||||
|
[MATRIX_SPOILER_REASON_PROPERTY_NAME]?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type IAudioContent = {
|
export type IAudioContent = {
|
||||||
|
|
Loading…
Reference in a new issue