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.
107 lines
2.8 KiB
107 lines
2.8 KiB
/** |
|
* @program: kicc-ui |
|
* @description: 水印工具 |
|
* @author: entfrm开发团队-王翔 |
|
* @create: 2022/4/10 |
|
*/ |
|
|
|
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'); |
|
|
|
export function useWatermark( |
|
appendEl: Ref<HTMLElement | null> = ref(document.body) as Ref<HTMLElement> |
|
) { |
|
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); |
|
return id; |
|
}; |
|
|
|
function setWatermark(str: string) { |
|
createWatermark(str); |
|
addResizeListener(document.documentElement, func); |
|
const instance = getCurrentInstance(); |
|
if (instance) { |
|
onBeforeUnmount(() => { |
|
clear(); |
|
}); |
|
} |
|
} |
|
|
|
return { setWatermark, clear }; |
|
}
|
|
|