ray-template/src/hooks/web/useDomToImage.ts
2024-03-09 23:39:59 +08:00

153 lines
3.5 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-12-14
*
* @workspace ray-template
*
* @remark 今天也是元气满满撸代码的一天
*/
import domToImage from 'dom-to-image'
import { unrefElement } from '@/utils'
import type { Options as ReDomToImageOptions } from 'dom-to-image'
import type { BasicTarget, TargetType } from '@/types'
export type ImageType = keyof typeof domToImageMethods
export type DomToImageResult = string | Blob | Uint8ClampedArray | undefined
export interface UseDomToImageOptions<T extends TargetType = Element>
extends ReDomToImageOptions {
/**
*
*
* 指定图片类型,允许传递 imageType 参数,用于指定图片类型
*
* @default jpeg
*/
imageType?: ImageType
/**
*
*
* 在 dom 转换为图片之前执行
*
* @param element current dom
*
* @default undefined
*/
beforeCreate?: (element: T | null | undefined) => void
/**
*
* @param element current dom
* @param result dom to image result
*
* 在 dom 转换为图片之后执行
*
* @default undefined
*/
created?: (result: DomToImageResult, element: T) => void
/**
*
* @param error dom to image error
*
* 在 dom 转换为图片失败时执行
*
* @default undefined
*/
createdError?: (error?: Error) => void
/**
*
* @param element current dom
*
* 无论 dom 转换为图片成功或失败,都会执行
*
* @default undefined
*/
finally?: () => void
}
const domToImageMethods = {
svg: domToImage.toSvg,
png: domToImage.toPng,
jpeg: domToImage.toJpeg,
blob: domToImage.toBlob,
pixelData: domToImage.toPixelData,
}
/**
*
* @param target ref dom
* @param options dom-to-image options
*
* @see https://github.com/tsayen/dom-to-image
*
* @description
* 使用 dom-to-image 将 dom 转换为图片,基于 dom-to-image v2.6.0。
* 拓展了 imageType 参数,用于指定图片类型。
*
* create 方法支持在执行时传递 imageType 参数,用于指定图片类型。并且优先级大于 options.imageType
* 当然,你也可以不传递 imageType 参数,此时会使用 options.imageType
* 如果都未传递,则默认使用 jpeg。
*
* @example
* const refDom = ref<HTMLElement>()
* const { create, stop } = useDomToImage(refDom, {
* beforeCreate: (element) => { ... },
* created: (element, result) => { ... },
* createdError: (error) => { ... },
* finally: () => { ... },
* })
*/
export const useDomToImage = <T extends HTMLElement>(
target: BasicTarget<T>,
options?: UseDomToImageOptions,
) => {
const {
beforeCreate,
created,
createdError,
finally: _finally,
imageType: _imageType,
} = options ?? {}
const run = (
imageType?: UseDomToImageOptions['imageType'],
): Promise<DomToImageResult> => {
return new Promise((resolve, reject) => {
const element = unrefElement(target)
beforeCreate?.(element)
if (!element) {
createdError?.()
return reject('useDomToImage: element is undefined.')
}
domToImageMethods[imageType ?? _imageType ?? 'jpeg']?.(element, options)
.then((res) => {
created?.(res, element)
return resolve(res)
})
.catch((error: Error) => {
createdError?.(error)
return reject(error)
})
.finally(() => {
_finally?.()
})
})
}
return {
create: run,
}
}
export type UseDomToImageReturnType = ReturnType<typeof useDomToImage>