You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
104 lines
2.9 KiB
104 lines
2.9 KiB
import { getCurrentInstance, onBeforeUnmount, ref, Ref, shallowRef, unref } from 'vue' |
|
import { useRafThrottle } from '@/utils/domUtils' |
|
import { addResizeListener, removeResizeListener } from '@/utils/event' |
|
import { isDef } from '@/utils/is' |
|
|
|
const domSymbol = Symbol('watermark-dom') |
|
const sourceMap = new WeakMap<HTMLElement, {}>() |
|
|
|
export function useWatermark(appendEl: Ref<HTMLElement | null> = ref(document.body) as Ref<HTMLElement>) { |
|
const appendElRaw = unref(appendEl) |
|
if (appendElRaw && sourceMap.has(appendElRaw)) { |
|
return sourceMap.get(appendElRaw) |
|
} |
|
const func = useRafThrottle(function () { |
|
const el = unref(appendEl) |
|
if (!el) return |
|
const { clientHeight: height, clientWidth: width } = el |
|
updateWatermark({ height, width }) |
|
}) |
|
const id = domSymbol.toString() |
|
const watermarkEl = shallowRef<HTMLElement>() |
|
|
|
const clear = () => { |
|
const domId = unref(watermarkEl) |
|
watermarkEl.value = undefined |
|
const el = unref(appendEl) |
|
if (!el) return |
|
domId && el.removeChild(domId) |
|
removeResizeListener(el, func) |
|
} |
|
|
|
function createBase64(str: string) { |
|
const can = document.createElement('canvas') |
|
const width = 300 |
|
const height = 240 |
|
Object.assign(can, { width, height }) |
|
|
|
const cans = can.getContext('2d') |
|
if (cans) { |
|
cans.rotate((-20 * Math.PI) / 120) |
|
cans.font = '15px Vedana' |
|
cans.fillStyle = 'rgba(0, 0, 0, 0.15)' |
|
cans.textAlign = 'left' |
|
cans.textBaseline = 'middle' |
|
cans.fillText(str, width / 20, height) |
|
} |
|
return can.toDataURL('image/png') |
|
} |
|
|
|
function updateWatermark( |
|
options: { |
|
width?: number |
|
height?: number |
|
str?: string |
|
} = {} |
|
) { |
|
const el = unref(watermarkEl) |
|
if (!el) return |
|
if (isDef(options.width)) { |
|
el.style.width = `${options.width}px` |
|
} |
|
if (isDef(options.height)) { |
|
el.style.height = `${options.height}px` |
|
} |
|
if (isDef(options.str)) { |
|
el.style.background = `url(${createBase64(options.str)}) left top repeat` |
|
} |
|
} |
|
|
|
const createWatermark = (str: string) => { |
|
if (unref(watermarkEl)) { |
|
updateWatermark({ str }) |
|
return id |
|
} |
|
const div = document.createElement('div') |
|
watermarkEl.value = div |
|
div.id = id |
|
div.style.pointerEvents = 'none' |
|
div.style.top = '0px' |
|
div.style.left = '0px' |
|
div.style.position = 'absolute' |
|
div.style.zIndex = '100000' |
|
const el = unref(appendEl) |
|
if (!el) return id |
|
const { clientHeight: height, clientWidth: width } = el |
|
updateWatermark({ str, width, height }) |
|
el.appendChild(div) |
|
sourceMap.set(el, { setWatermark, clear }) |
|
return id |
|
} |
|
|
|
function setWatermark(str: string) { |
|
createWatermark(str) |
|
addResizeListener(document.documentElement, func) |
|
const instance = getCurrentInstance() |
|
if (instance) { |
|
onBeforeUnmount(() => { |
|
clear() |
|
}) |
|
} |
|
} |
|
|
|
return { setWatermark, clear } |
|
}
|
|
|