80 lines
1.8 KiB
TypeScript
80 lines
1.8 KiB
TypeScript
import { useState } from 'react'
|
|
import html2canvas from 'html2canvas'
|
|
|
|
/**
|
|
* @module Main_Hook
|
|
* Hook return
|
|
* @typedef {Array} HookReturn
|
|
* @property {string} HookReturn[0] - image string
|
|
* @property {string} HookReturn[1] - take screen shot string
|
|
* @property {object} HookReturn[2] - errors
|
|
*/
|
|
|
|
/**
|
|
* hook for creating screenshot from html node
|
|
* @returns {HookReturn}
|
|
*/
|
|
const useScreenshot: () => [
|
|
HTMLCanvasElement | null,
|
|
(node: HTMLElement) => Promise<void | HTMLCanvasElement>,
|
|
{ error: null }
|
|
] = () => {
|
|
const [image, setImage] = useState<HTMLCanvasElement | null>(null)
|
|
const [error, setError] = useState(null)
|
|
/**
|
|
* convert html node to image
|
|
* @param {HTMLElement} node
|
|
*/
|
|
const takeScreenshot = (node: HTMLElement) => {
|
|
if (!node) {
|
|
throw new Error('You should provide correct html node.')
|
|
}
|
|
return html2canvas(node)
|
|
.then((canvas) => {
|
|
const croppedCanvas = document.createElement('canvas')
|
|
const croppedCanvasContext = croppedCanvas.getContext('2d')
|
|
// init data
|
|
const cropPositionTop = 0
|
|
const cropPositionLeft = 0
|
|
const cropWidth = canvas.width
|
|
const cropHeight = canvas.height
|
|
|
|
croppedCanvas.width = cropWidth
|
|
croppedCanvas.height = cropHeight
|
|
|
|
croppedCanvasContext?.drawImage(
|
|
canvas,
|
|
cropPositionLeft,
|
|
cropPositionTop
|
|
)
|
|
|
|
setImage(croppedCanvas)
|
|
return croppedCanvas
|
|
})
|
|
.catch(setError)
|
|
}
|
|
|
|
return [
|
|
image,
|
|
takeScreenshot,
|
|
{
|
|
error,
|
|
},
|
|
]
|
|
}
|
|
|
|
/**
|
|
* creates name of file
|
|
* @param {string} extension
|
|
* @param {string[]} parts of file name
|
|
*/
|
|
const createFileName = (extension = '', ...names) => {
|
|
if (!extension) {
|
|
return ''
|
|
}
|
|
|
|
return `${names.join('')}.${extension}`
|
|
}
|
|
|
|
export { useScreenshot, createFileName }
|