-
-
Notifications
You must be signed in to change notification settings - Fork 781
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: Innei <[email protected]>
- Loading branch information
Showing
5 changed files
with
205 additions
and
10 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,197 @@ | ||
import { useNodeViewContext } from '@prosemirror-adapter/react' | ||
import { useCallback, useRef, useState } from 'react' | ||
import type { MilkdownPlugin } from '@milkdown/ctx' | ||
import type { FC } from 'react' | ||
import type { PluginCtx } from './types' | ||
|
||
import { imageSchema } from '@milkdown/preset-commonmark' | ||
import { $view } from '@milkdown/utils' | ||
|
||
import { useIsMobile } from '~/atoms' | ||
import { MotionButtonBase, StyledButton } from '~/components/ui/button' | ||
import { FloatPopover } from '~/components/ui/float-popover' | ||
import { Form, FormInput } from '~/components/ui/form' | ||
import { FixedZoomedImage } from '~/components/ui/image' | ||
import { useCurrentModal, useModalStack } from '~/components/ui/modal' | ||
import { toast } from '~/lib/toast' | ||
|
||
const base64ToFile = (base64: string) => { | ||
const arr = base64.split(',') | ||
const mime = arr[0].match(/:(.*?);/)![1] | ||
const bstr = atob(arr[1]) | ||
let n = bstr.length | ||
const u8arr = new Uint8Array(n) | ||
while (n--) { | ||
u8arr[n] = bstr.charCodeAt(n) | ||
} | ||
return new File([u8arr], 'image.png', { type: mime }) | ||
} | ||
const Image = () => { | ||
const { node, setAttrs } = useNodeViewContext() | ||
|
||
const { src, alt, title } = node.attrs | ||
|
||
const isMobile = useIsMobile() | ||
const isOnline = src.startsWith('http') | ||
const handleUploadImage = () => { | ||
if (src.startsWith('http')) return | ||
const formData = new FormData() | ||
formData.append('file', base64ToFile(src)) | ||
|
||
fetch('/api/s3', { | ||
method: 'POST', | ||
body: formData, | ||
}) | ||
.then((response) => response.json()) | ||
.then((data) => { | ||
if (data.url) { | ||
toast.success('上传成功') | ||
setAttrs({ src: data.url }) | ||
} else { | ||
toast.error('上传失败') | ||
} | ||
}) | ||
.catch((error: any) => { | ||
console.error(error) | ||
toast.error(`上传失败${error?.message}`) | ||
}) | ||
} | ||
|
||
const modal = useModalStack() | ||
return ( | ||
<div className="my-4 flex center"> | ||
<div className="group relative inline-block"> | ||
<FixedZoomedImage src={src} containerWidth={isMobile ? 450 : 600} /> | ||
|
||
<div className="absolute bottom-1 left-1 z-[10] opacity-0 duration-200 group-hover:opacity-100"> | ||
<StyledButton | ||
onClick={() => { | ||
modal.present({ | ||
title: '编辑图片', | ||
content: () => ( | ||
<ImageEdit | ||
onSubmit={(state) => { | ||
setAttrs(state) | ||
}} | ||
src={src} | ||
alt={alt} | ||
title={title} | ||
/> | ||
), | ||
}) | ||
}} | ||
variant="secondary" | ||
className="box-content flex h-6 w-6 rounded-full p-2 center" | ||
> | ||
<i className="icon-[mingcute--edit-line] text-[16px]" /> | ||
</StyledButton> | ||
</div> | ||
{!isOnline && ( | ||
<FloatPopover | ||
placement="right" | ||
triggerElement={ | ||
<MotionButtonBase | ||
onClick={handleUploadImage} | ||
className="absolute bottom-1 right-1 box-content flex rounded-full bg-base-100 text-red-500 center" | ||
> | ||
<i className="icon-[mingcute--warning-line] text-[24px]" /> | ||
</MotionButtonBase> | ||
} | ||
> | ||
此图片是一个内嵌图片,还没有上传到服务器,点击上传到服务器(需要配置 | ||
S3) | ||
</FloatPopover> | ||
)} | ||
</div> | ||
</div> | ||
) | ||
} | ||
|
||
export const ImagePlugin: (pluginCtx: PluginCtx) => MilkdownPlugin[] = ({ | ||
nodeViewFactory, | ||
}) => [ | ||
$view(imageSchema.node, () => | ||
nodeViewFactory({ | ||
component: Image, | ||
}), | ||
), | ||
] | ||
|
||
interface ImageEditData { | ||
src: string | ||
alt?: string | ||
title?: string | ||
} | ||
interface ImageEditProps extends ImageEditData { | ||
onSubmit: (data: ImageEditData) => void | ||
} | ||
const ImageEdit: FC<ImageEditProps> = ({ onSubmit, src, alt, title }) => { | ||
const currentModel = useCurrentModal() | ||
const inputs = useRef([ | ||
{ | ||
name: 'src', | ||
placeholder: 'https://', | ||
rules: [ | ||
{ | ||
validator: (value: string) => value.startsWith('http'), | ||
message: '请输入正确的图片链接 https://', | ||
}, | ||
], | ||
}, | ||
{ | ||
name: 'alt', | ||
placeholder: '图片信息', | ||
rules: [ | ||
{ | ||
validator: (value: string) => value.length <= 200, | ||
message: '图片信息不能超过200个字符', | ||
}, | ||
], | ||
}, | ||
{ | ||
name: 'title', | ||
placeholder: '图片标题', | ||
rules: [ | ||
{ | ||
validator: (value: string) => value.length <= 50, | ||
message: '图片标题不能超过50个字符', | ||
}, | ||
], | ||
}, | ||
]).current | ||
|
||
const handleSubmit = () => { | ||
onSubmit(state) | ||
currentModel.dismiss() | ||
} | ||
|
||
const [state, setState] = useState({ | ||
src, | ||
alt: alt || '', | ||
title: title || '', | ||
} as ImageEditData) | ||
|
||
const setValue = useCallback((key: keyof typeof state, value: string) => { | ||
setState((prevState) => ({ ...prevState, [key]: value })) | ||
}, []) | ||
const handleChange = useCallback((e: React.ChangeEvent<HTMLInputElement>) => { | ||
setValue(e.target.name as keyof typeof state, e.target.value) | ||
}, []) | ||
|
||
return ( | ||
<Form className="w-[300px] space-y-4 text-center" onSubmit={handleSubmit}> | ||
{inputs.map((input) => ( | ||
<FormInput | ||
key={input.name} | ||
value={(state as any)[input.name]} | ||
onChange={handleChange} | ||
{...input} | ||
/> | ||
))} | ||
|
||
<StyledButton variant="primary" type="submit"> | ||
好 | ||
</StyledButton> | ||
</Form> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters