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)
}

大功告成🎉🎉🎉

相关推荐
颜酱4 小时前
图结构完全解析:从基础概念到遍历实现
javascript·后端·算法
失忆爆表症5 小时前
05_UI 组件库集成指南:Shadcn/ui + Tailwind CSS v4
前端·css·ui
小迷糊的学习记录5 小时前
Vuex 与 pinia
前端·javascript·vue.js
发现一只大呆瓜5 小时前
前端性能优化:图片懒加载的三种手写方案
前端·javascript·面试
不爱吃糖的程序媛5 小时前
Flutter 与 OpenHarmony 通信:Flutter Channel 使用指南
前端·javascript·flutter
利刃大大5 小时前
【Vue】Element-Plus快速入门 && Form && Card && Table && Tree && Dialog && Menu
前端·javascript·vue.js·element-plus
NEXT065 小时前
AI 应用工程化实战:使用 LangChain.js 编排 DeepSeek 复杂工作流
前端·javascript·langchain
念风零壹5 小时前
AI 时代的前端技术:从系统编程到 JavaScript/TypeScript
前端·ai
光影少年6 小时前
react的hooks防抖和节流是怎样做的
前端·javascript·react.js
小毛驴8506 小时前
Vue 路由示例
前端·javascript·vue.js