Vue3埋点指令封装

😎前言:对于Vue项目来说,如果遇到一个埋点的需求,我们最好的解决方案就是封装一个指令,然后再有埋点需求的地方使用,接下来就来封装一个埋点指令吧。👇👇

代码实现

track.ts文件

javascript 复制代码
import HttpAxios from '@/utils/axios'
​
//定义埋点请求
export function track(params) {
  return HttpAxios.post(
    'http://xxxxxx.xxxxx',
    params,
    {
      isCloseLoading: true
    }
  )
}
​
export default {
  install(Vue, options) {
    options = options || {}
    /**
     * 格式化绑定到dom上的数据
     * @param {*} binding
     */
    function formatBinding(binding) {
      let trackData = ''
      let eventMode = 'click'
      if (typeof binding.value === 'object') {
        if ('event' in binding.value) {
          eventMode = binding.value.event
        }
        if ('data' in binding.value) {
          trackData = binding.value.data
        } else {
          trackData = binding.value
        }
      } else {
        trackData = binding.value
      }
      return {
        eventMode,
        trackData
      }
    }
    // 初始化
    if ('init' in options && options.init instanceof Function) {
      try {
        options.init()
      } catch (error) {
        if (options.debug) {
          console.log(error)
        }
      }
    }
    Vue.directive('track', {
      mounted(el, binding) {
        const format = formatBinding(binding)
        el.trackData = format.trackData
        const params = {
          systemName: options.systemName,
          ...el.trackData //指令绑定的数据
        }
        el.addEventListener(format.eventMode, e => {
          try {
            if ('callback' in options && options.callback instanceof Function) {
              options.callback(params)
            } else {
              // 若未定义回调函数,则默认调用track方法
              track(params)
            }
            if (options.debug) {
              console.log(el.trackData)
            }
          } catch (error) {
            if (options.debug) {
              console.log(error)
            }
          }
        })
      },
      update(el, binding) {
        const format = formatBinding(binding)
        el.trackData = format.trackData
      }
    })
  }
}

main.ts文件

javascript 复制代码
// 引入埋点
import VTrack from '@monorepo/shared/utils/track'
​
// 创建vue实例
const app = createApp(App)
​
// 1.挂载埋点,没有回调函数的方式
app.use(VTrack, { systemName: '基础平台', debug: false })
​
// 2.挂载埋点,回调函数的方式
app.use(VTrack, {
  callback(data, e) {
    //可以自定义埋点请求
    console.log(data, e);
  },
  systemName: '基础平台',
  debug: false,
});

使用:

xml 复制代码
<template>
  <button v-track="{ menuName: '按钮' }">DEBUG</button>
</template>

关于指令项目规范化

src目录下,创建directives文件夹存放该项目所需的指令,如图所示:

index.ts文件统一注册指令:

typescript 复制代码
import type { App } from 'vue'
import { hasRole } from './permission/hasRole'
import { hasPermi } from './permission/hasPermi'
​
/**
 * 导出指令:v-xxx
 * @methods hasRole 用户权限,用法: v-hasRole
 * @methods hasPermi 按钮权限,用法: v-hasPermi
 */
export const setupAuth = (app: App<Element>) => {
  hasRole(app)
  hasPermi(app)
}

main.ts文件

javascript 复制代码
// 指令
import * as directives from '@/directives'
​
// 创建vue实例
const app = createApp(App)
​
// 注册指令
for (const key in directives) {
  directives[key](app)
}

大功告成🎉🎉🎉

相关推荐
mCell2 小时前
GSAP ScrollTrigger 详解
前端·javascript·动效
gnip2 小时前
Node.js 子进程:child_process
前端·javascript
excel5 小时前
为什么在 Three.js 中平面能产生“起伏效果”?
前端
excel6 小时前
Node.js 断言与测试框架示例对比
前端
天蓝色的鱼鱼8 小时前
前端开发者的组件设计之痛:为什么我的组件总是难以维护?
前端·react.js
codingandsleeping8 小时前
使用orval自动拉取swagger文档并生成ts接口
前端·javascript
石金龙9 小时前
[译] Composition in CSS
前端·css
白水清风9 小时前
微前端学习记录(qiankun、wujie、micro-app)
前端·javascript·前端工程化
Ticnix9 小时前
函数封装实现Echarts多表渲染/叠加渲染
前端·echarts
用户22152044278009 小时前
new、原型和原型链浅析
前端·javascript