在 vue 中我们可以使用 Vue.directive()方法注册全局指令。也可以只用 directives 选项注册局部指令。
| const debounce = { |
| inserted: function (el, binding) { |
| let timer; |
| el.addEventListener("keyup", () => { |
| if (timer) { |
| clearTimeout(timer); |
| } |
| timer = setTimeout(() => { |
| binding.value(); |
| }, 1000); |
| }); |
| }, |
| }; |
| |
| export default debounce; |
| const copy = { |
| bind(el, { value }) { |
| el.$value = value; |
| el.handler = () => { |
| if (!el.$value) { |
| |
| console.log("无复制内容"); |
| return; |
| } |
| |
| const textarea = document.createElement("textarea"); |
| |
| textarea.readOnly = "readonly"; |
| textarea.style.position = "absolute"; |
| textarea.style.left = "-9999px"; |
| |
| textarea.value = el.$value; |
| |
| document.body.appendChild(textarea); |
| |
| textarea.select(); |
| const result = document.execCommand("Copy"); |
| if (result) { |
| console.log("复制成功"); |
| } |
| document.body.removeChild(textarea); |
| }; |
| |
| el.addEventListener("click", el.handler); |
| }, |
| |
| componentUpdated(el, { value }) { |
| el.$value = value; |
| }, |
| |
| unbind(el) { |
| el.removeEventListener("click", el.handler); |
| }, |
| }; |
| |
| export default copy; |
| const longpress = { |
| bind: function (el, binding, vNode) { |
| if (typeof binding.value !== "function") { |
| throw "callback must be a function"; |
| } |
| |
| let pressTimer = null; |
| |
| let start = (e) => { |
| if (e.type === "click" && e.button !== 0) { |
| return; |
| } |
| if (pressTimer === null) { |
| pressTimer = setTimeout(() => { |
| handler(); |
| }, 2000); |
| } |
| }; |
| |
| let cancel = (e) => { |
| if (pressTimer !== null) { |
| clearTimeout(pressTimer); |
| pressTimer = null; |
| } |
| }; |
| |
| const handler = (e) => { |
| binding.value(e); |
| }; |
| |
| el.addEventListener("mousedown", start); |
| el.addEventListener("touchstart", start); |
| |
| el.addEventListener("click", cancel); |
| el.addEventListener("mouseout", cancel); |
| el.addEventListener("touchend", cancel); |
| el.addEventListener("touchcancel", cancel); |
| }, |
| |
| componentUpdated(el, { value }) { |
| el.$value = value; |
| }, |
| |
| unbind(el) { |
| el.removeEventListener("click", el.handler); |
| }, |
| }; |
| |
| export default longpress; |
| let findEle = (parent, type) => { |
| return parent.tagName.toLowerCase() === type |
| ? parent |
| : parent.querySelector(type); |
| }; |
| |
| const trigger = (el, type) => { |
| const e = document.createEvent("HTMLEvents"); |
| e.initEvent(type, true, true); |
| el.dispatchEvent(e); |
| }; |
| |
| const emoji = { |
| bind: function (el, binding, vnode) { |
| // 正则规则可根据需求自定义 |
| var regRule = /[^u4E00-u9FA5|d|a-zA-Z|rns,.?!,。?!…—&$=()-+/*{}[]]|s/g; |
| let $inp = findEle(el, "input"); |
| el.$inp = $inp; |
| $inp.handle = function () { |
| let val = $inp.value; |
| $inp.value = val.replace(regRule, ""); |
| |
| trigger($inp, "input"); |
| }; |
| $inp.addEventListener("keyup", $inp.handle); |
| }, |
| unbind: function (el) { |
| el.$inp.removeEventListener("keyup", el.$inp.handle); |
| }, |
| }; |
| |
| export default emoji; |
| const LazyLoad = { |
| |
| install(Vue, options) { |
| const defaultSrc = options.default; |
| Vue.directive("lazy", { |
| bind(el, binding) { |
| LazyLoad.init(el, binding.value, defaultSrc); |
| }, |
| inserted(el) { |
| if (IntersectionObserver) { |
| LazyLoad.observe(el); |
| } else { |
| LazyLoad.listenerScroll(el); |
| } |
| }, |
| }); |
| }, |
| |
| init(el, val, def) { |
| el.setAttribute("data-src", val); |
| el.setAttribute("src", def); |
| }, |
| |
| observe(el) { |
| var io = new IntersectionObserver((entries) => { |
| const realSrc = el.dataset.src; |
| if (entries[0].isIntersecting) { |
| if (realSrc) { |
| el.src = realSrc; |
| el.removeAttribute("data-src"); |
| } |
| } |
| }); |
| io.observe(el); |
| }, |
| |
| listenerScroll(el) { |
| const handler = LazyLoad.throttle(LazyLoad.load, 300); |
| LazyLoad.load(el); |
| window.addEventListener("scroll", () => { |
| handler(el); |
| }); |
| }, |
| |
| load(el) { |
| const windowHeight = document.documentElement.clientHeight; |
| const elTop = el.getBoundingClientRect().top; |
| const elBtm = el.getBoundingClientRect().bottom; |
| const realSrc = el.dataset.src; |
| if (elTop - windowHeight < 0 && elBtm > 0) { |
| if (realSrc) { |
| el.src = realSrc; |
| el.removeAttribute("data-src"); |
| } |
| } |
| }, |
| |
| throttle(fn, delay) { |
| let timer; |
| let prevTime; |
| return function (...args) { |
| const currTime = Date.now(); |
| const context = this; |
| if (!prevTime) prevTime = currTime; |
| clearTimeout(timer); |
| |
| if (currTime - prevTime > delay) { |
| prevTime = currTime; |
| fn.apply(context, args); |
| clearTimeout(timer); |
| return; |
| } |
| |
| timer = setTimeout(function () { |
| prevTime = Date.now(); |
| timer = null; |
| fn.apply(context, args); |
| }, delay); |
| }; |
| }, |
| }; |
| |
| export default LazyLoad; |
| function checkArray(key) { |
| let arr = ["1", "2", "3", "4"]; |
| let index = arr.indexOf(key); |
| if (index > -1) { |
| return true; |
| } else { |
| return false; |
| } |
| } |
| |
| const permission = { |
| inserted: function (el, binding) { |
| let permission = binding.value; |
| if (permission) { |
| let hasPermission = checkArray(permission); |
| if (!hasPermission) { |
| |
| el.parentNode && el.parentNode.removeChild(el); |
| } |
| } |
| }, |
| }; |
| |
| export default permission; |
| function addWaterMarker(str, parentNode, font, textColor) { |
| |
| var can = document.createElement("canvas"); |
| parentNode.appendChild(can); |
| can.width = 200; |
| can.height = 150; |
| can.style.display = "none"; |
| var cans = can.getContext("2d"); |
| cans.rotate((-20 * Math.PI) / 180); |
| cans.font = font || "16px Microsoft JhengHei"; |
| cans.fillStyle = textColor || "rgba(180, 180, 180, 0.3)"; |
| cans.textAlign = "left"; |
| cans.textBaseline = "Middle"; |
| cans.fillText(str, can.width / 10, can.height / 2); |
| parentNode.style.backgroundImage = "url(" + can.toDataURL("image/png") + ")"; |
| } |
| |
| const waterMarker = { |
| bind: function (el, binding) { |
| addWaterMarker( |
| binding.value.text, |
| el, |
| binding.value.font, |
| binding.value.textColor |
| ); |
| }, |
| }; |
| |
| export default waterMarker; |
| const draggable = { |
| inserted: function (el) { |
| el.style.cursor = "move"; |
| el.onmousedown = function (e) { |
| let disx = e.pageX - el.offsetLeft; |
| let disy = e.pageY - el.offsetTop; |
| document.onmousemove = function (e) { |
| let x = e.pageX - disx; |
| let y = e.pageY - disy; |
| let maxX = |
| document.body.clientWidth - |
| parseInt(window.getComputedStyle(el).width); |
| let maxY = |
| document.body.clientHeight - |
| parseInt(window.getComputedStyle(el).height); |
| if (x < 0) { |
| x = 0; |
| } else if (x > maxX) { |
| x = maxX; |
| } |
| |
| if (y < 0) { |
| y = 0; |
| } else if (y > maxY) { |
| y = maxY; |
| } |
| |
| el.style.left = x + "px"; |
| el.style.top = y + "px"; |
| }; |
| document.onmouseup = function () { |
| document.onmousemove = document.onmouseup = null; |
| }; |
| }; |
| }, |
| }; |
| export default draggable; |
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通