mirror of https://github.com/buster-so/buster.git
169 lines
7.2 KiB
TypeScript
169 lines
7.2 KiB
TypeScript
'use client';
|
|
|
|
import * as React from 'react';
|
|
import LiteYouTubeEmbed from 'react-lite-youtube-embed';
|
|
|
|
import type { TMediaEmbedElement } from 'platejs';
|
|
import type { PlateElementProps } from 'platejs/react';
|
|
|
|
import { parseTwitterUrl, parseVideoUrl } from '@platejs/media';
|
|
import { MediaEmbedPlugin, useMediaState } from '@platejs/media/react';
|
|
import { ResizableProvider, useResizableValue } from '@platejs/resizable';
|
|
import { PlateElement, useFocused, useReadOnly, useSelected, withHOC } from 'platejs/react';
|
|
|
|
import { cn } from '@/lib/utils';
|
|
|
|
import { Caption, CaptionTextarea } from './CaptionNode';
|
|
import { MediaToolbar } from './MediaToolbar';
|
|
import { mediaResizeHandleVariants, Resizable, ResizeHandle } from './ResizeHandle';
|
|
import { Code3 } from '../../icons';
|
|
import { PopoverAnchor, PopoverBase, PopoverContent } from '../../popover';
|
|
import { useEffect, useState } from 'react';
|
|
import { Title } from '../../typography';
|
|
import { Input } from '../../inputs';
|
|
import { Button } from '../../buttons';
|
|
|
|
export const MediaEmbedElement = withHOC(
|
|
ResizableProvider,
|
|
function MediaEmbedElement(props: PlateElementProps<TMediaEmbedElement>) {
|
|
const {
|
|
align = 'center',
|
|
embed,
|
|
focused,
|
|
isTweet,
|
|
isVideo,
|
|
isYoutube,
|
|
readOnly,
|
|
selected
|
|
} = useMediaState({
|
|
urlParsers: [parseTwitterUrl, parseVideoUrl]
|
|
});
|
|
const width = useResizableValue('width');
|
|
const provider = embed?.provider;
|
|
const hasElement = !!embed?.url;
|
|
|
|
if (!hasElement) {
|
|
return <MediaEmbedPlaceholder {...props} />;
|
|
}
|
|
|
|
return (
|
|
<MediaToolbar plugin={MediaEmbedPlugin}>
|
|
<PlateElement className="media-embed py-2.5" {...props}>
|
|
<figure className="group relative m-0 w-full cursor-default" contentEditable={false}>
|
|
<Resizable
|
|
align={align}
|
|
options={{
|
|
align,
|
|
maxWidth: isTweet ? 550 : '100%',
|
|
minWidth: isTweet ? 300 : 100
|
|
}}>
|
|
<ResizeHandle
|
|
className={mediaResizeHandleVariants({ direction: 'left' })}
|
|
options={{ direction: 'left' }}
|
|
/>
|
|
|
|
{isVideo ? (
|
|
isYoutube ? (
|
|
<LiteYouTubeEmbed
|
|
id={embed!.id!}
|
|
title="youtube"
|
|
wrapperClass={cn(
|
|
'rounded-sm',
|
|
focused && selected && 'ring-2 ring-ring ring-offset-2',
|
|
'relative block cursor-pointer bg-black bg-cover bg-center [contain:content]',
|
|
'[&.lyt-activated]:before:absolute [&.lyt-activated]:before:top-0 [&.lyt-activated]:before:h-[60px] [&.lyt-activated]:before:w-full [&.lyt-activated]:before:bg-top [&.lyt-activated]:before:bg-repeat-x [&.lyt-activated]:before:pb-[50px] [&.lyt-activated]:before:[transition:all_0.2s_cubic-bezier(0,_0,_0.2,_1)]',
|
|
'[&.lyt-activated]:before:bg-[url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAADGCAYAAAAT+OqFAAAAdklEQVQoz42QQQ7AIAgEF/T/D+kbq/RWAlnQyyazA4aoAB4FsBSA/bFjuF1EOL7VbrIrBuusmrt4ZZORfb6ehbWdnRHEIiITaEUKa5EJqUakRSaEYBJSCY2dEstQY7AuxahwXFrvZmWl2rh4JZ07z9dLtesfNj5q0FU3A5ObbwAAAABJRU5ErkJggg==)]',
|
|
'after:block after:pb-[var(--aspect-ratio)] after:content-[""]',
|
|
'[&_>_iframe]:absolute [&_>_iframe]:top-0 [&_>_iframe]:left-0 [&_>_iframe]:size-full',
|
|
'[&_>_.lty-playbtn]:z-1 [&_>_.lty-playbtn]:h-[46px] [&_>_.lty-playbtn]:w-[70px] [&_>_.lty-playbtn]:rounded-[14%] [&_>_.lty-playbtn]:bg-[#212121] [&_>_.lty-playbtn]:opacity-80 [&_>_.lty-playbtn]:[transition:all_0.2s_cubic-bezier(0,_0,_0.2,_1)]',
|
|
'[&:hover_>_.lty-playbtn]:bg-[red] [&:hover_>_.lty-playbtn]:opacity-100',
|
|
'[&_>_.lty-playbtn]:before:border-y-[11px] [&_>_.lty-playbtn]:before:border-r-0 [&_>_.lty-playbtn]:before:border-l-[19px] [&_>_.lty-playbtn]:before:border-[transparent_transparent_transparent_#fff] [&_>_.lty-playbtn]:before:content-[""]',
|
|
'[&_>_.lty-playbtn]:absolute [&_>_.lty-playbtn]:top-1/2 [&_>_.lty-playbtn]:left-1/2 [&_>_.lty-playbtn]:[transform:translate3d(-50%,-50%,0)]',
|
|
'[&_>_.lty-playbtn]:before:absolute [&_>_.lty-playbtn]:before:top-1/2 [&_>_.lty-playbtn]:before:left-1/2 [&_>_.lty-playbtn]:before:[transform:translate3d(-50%,-50%,0)]',
|
|
'[&.lyt-activated]:cursor-[unset]',
|
|
'[&.lyt-activated]:before:pointer-events-none [&.lyt-activated]:before:opacity-0',
|
|
'[&.lyt-activated_>_.lty-playbtn]:pointer-events-none [&.lyt-activated_>_.lty-playbtn]:opacity-0!'
|
|
)}
|
|
/>
|
|
) : (
|
|
<div
|
|
className={cn(
|
|
provider === 'vimeo' && 'pb-[75%]',
|
|
provider === 'youku' && 'pb-[56.25%]',
|
|
provider === 'dailymotion' && 'pb-[56.0417%]',
|
|
provider === 'coub' && 'pb-[51.25%]'
|
|
)}>
|
|
<iframe
|
|
className={cn(
|
|
'absolute top-0 left-0 size-full rounded-sm',
|
|
isVideo && 'border-0',
|
|
focused && selected && 'ring-ring ring-2 ring-offset-2'
|
|
)}
|
|
title="embed"
|
|
src={embed!.url}
|
|
allowFullScreen
|
|
/>
|
|
</div>
|
|
)
|
|
) : null}
|
|
|
|
<ResizeHandle
|
|
className={mediaResizeHandleVariants({ direction: 'right' })}
|
|
options={{ direction: 'right' }}
|
|
/>
|
|
</Resizable>
|
|
|
|
<Caption style={{ width }} align={align}>
|
|
<CaptionTextarea placeholder="Write a caption..." />
|
|
</Caption>
|
|
</figure>
|
|
|
|
{props.children}
|
|
</PlateElement>
|
|
</MediaToolbar>
|
|
);
|
|
}
|
|
);
|
|
|
|
export const MediaEmbedPlaceholder = (props: PlateElementProps<TMediaEmbedElement>) => {
|
|
const readOnly = useReadOnly();
|
|
const selected = useSelected();
|
|
const focused = useFocused();
|
|
|
|
const isFocused = focused && selected && !readOnly;
|
|
|
|
return (
|
|
<PlateElement className="media-embed py-2.5" {...props}>
|
|
<PopoverBase open={isFocused}>
|
|
<PopoverAnchor>
|
|
<div
|
|
className={cn(
|
|
'bg-muted hover:bg-primary/10 flex cursor-pointer items-center rounded-sm p-3 pr-9 select-none'
|
|
)}
|
|
contentEditable={false}>
|
|
<div className="text-muted-foreground/80 relative mr-3 flex [&_svg]:size-6">
|
|
<Code3 />
|
|
</div>
|
|
|
|
<div className="text-muted-foreground text-sm whitespace-nowrap">Add a media embed</div>
|
|
</div>
|
|
{props.children}
|
|
</PopoverAnchor>
|
|
|
|
<PopoverContent
|
|
className="w-[250px] p-0"
|
|
onOpenAutoFocus={(e) => {
|
|
console.log('onOpenAutoFocus', e);
|
|
e.preventDefault();
|
|
}}>
|
|
<Title as="h4">Add a media embed</Title>
|
|
<div className="bg-gray-light h-0.5 w-full" />
|
|
|
|
<Input placeholder="Enter a URL" />
|
|
<Button block>Add media</Button>
|
|
</PopoverContent>
|
|
</PopoverBase>
|
|
</PlateElement>
|
|
);
|
|
};
|