Vue 可视化无痕埋点的探索

代码地址:
https://github.com/geminate/vue-visual-track
由于埋点需求灵活复杂,本项目并未实现具体埋点功能,仅为可视化无痕埋点的探索

一. 简介

之前公司经常有一些埋点需求,比如监控某个按钮的点击时间与次数、输入框的聚焦次数等,当时采用的方案都是修改原项目代码添加对应的事件。再通过请求发到统计端。

但是这样处理存在的问题就是如果需要对埋点进行调整,必须要修改项目内部代码并发版,而且埋点技术性太强,只能由开发处理。

为了解决这个问题,参照网上已有的相关方案,我在 可视化无痕埋点的方案上做了一下探索。

最终形成的是一个 基于 Electron 的埋点程序,由于埋点需求的灵活性很强,因此只实现了最基本的功能。

二. 可视化埋点操作流程

1. 在可视化界面上打开项目地址,并圈选需要埋点的 DOM 元素

5.gif

2. 对选中的 DOM 元素添加自定义埋点事件

1564651373(1).jpg

3. 在完成了多个 DOM 元素的事件添加后,根据保存的事件列表生成一个 JS 文件

1564651487(1).jpg

4. 在原 VUE 项目中引入生成的 JS 文件,完成埋点

三. 项目编写过程中遇到的问题

1. 如何实现 可视化选择 DOM 元素

最初打算建立一个 web 项目,然后通过 webview 嵌入需要埋点的项目的地址,但由于 webview 权限较低,最终选择了桌面程序 Electron。

Electron 的 webview 组件有着比较高的权限,可以向 webview 内部打开的页面嵌入自己的JS代码,因此我们可以为打开的页面里的所有元素添加指向事件,在指向时改变其背景色,也就简单实现了 DOM 的可视化选择。这个与 chrome 控制台的 dom 元素框选十分类似。

利用 webview 的 executeJavaScript 方法可以向 webview 打开的页面中嵌入自己的 JS 字符串,这里使用了 raw-loader 将文件作为字符串引入

import insertJs from './insert.jsraw'

onDomReady () {
    this.$refs.webview.executeJavaScript(insertJs)
}
const {ipcRenderer} = require('electron')
let selectDom = null

// eslint-disable-next-line no-unused-vars
var selectPlugin = {

  startSelect: function () {
    document.body.style.cursor = 'crosshair'
    this.addEvent(document.body, 'mouseover', this.mouseOverHandler)
    this.addEvent(document.body, 'mouseout', this.mouseOutHandler)
  },

  stopSelect: function () {
    document.body.style.cursor = 'auto'
    this.removeEvent(document.body, 'mouseover', this.mouseOverHandler)
    this.removeEvent(document.body, 'mouseout', this.mouseOutHandler)
  },

  mouseOverHandler: function (e) {
    e.stopPropagation()
    e.target.style.backgroundColor = 'rgb(160,191,232)'
    selectDom = e.target
  },

  mouseOutHandler: function (e) {
    e.stopPropagation()
    e.target.style.backgroundColor = ''
  }
  
  ...
}

2. 如何获取选中元素的唯一选择器

在能够在目标页面上进行可视化选择之后,我们需要获取到 能够唯一标识这个元素的选择器,以便将来生成的 JS 文件中能够根据这个选择器找到我们圈选的元素。


  getUniqueSelect: function (node) {
    let path
    while (node) {
      let name = node.localName
      if (name) {
        name = name.toLowerCase()
        const parent = node.parentElement
        let reChildrenNode = []
        const childNodes = (parent && parent.childNodes) || []
        for (let i = 0; i < childNodes.length; i++) {
          if (childNodes[i].nodeName.toLowerCase() === name && !/\s/.test(childNodes.nodeValue)) {
            reChildrenNode.push(childNodes[i])
          }
        }
        if (reChildrenNode.length > 1) {
          const index = reChildrenNode.indexOf(node) + 1
          name += ':nth-of-type(' + index + ')'
        }
        path = name + (path ? '>' + path : '')
        node = parent
      }
    }
    return path
  }
  

上面的方法会返回类似 html>body>div>div>ul:nth-of-type(1)>li:nth-of-type(3)>a 这样的选择器字符串,选择器可以作为圈选元素的 唯一DOM标识。

3. 如何实现 VUE 的无痕埋点事件注入

想要实现 Vue 的无痕埋点,在对各个元素添加事件的时候 只能采取 事先注册好的命令式埋点。

在页面一打开的时候就将埋点事件注册到 document 元素上,之后用 e.target 和 事件的 选择器字符串进行匹配即可。

由于 选择器字符串的唯一性 只在同一个页面中有效,因此我们想要在 vue 项目中完全确定一个事件的所属元素的话,必须要有 页面地址和唯一选择器两样才可以。

新版本的 vue-router 使用 pushState 进行页面跳转,因此我们想要 无痕的监控 Vue 页面跳转需要对该方法进行修改,加入我们自己的回调函数。

const pushState = window.history.pushState
window.history.pushState = function (obj, name, hash) {
  if (typeof window.onPushstate === 'function') {
    window.onPushstate(hash)
  }
  return pushState.apply(window.history, arguments)
}

const popState = window.history.popState
window.history.popState = function (obj, name, hash) {
  if (typeof window.onpopstate === 'function') {
    window.onpopstate(hash)
  }
  return popState.apply(window.history, arguments)
}

const replaceState = window.history.replaceState
window.history.replaceState = function (obj, name, hash) {
  if (typeof window.onPushstate === 'function') {
    window.onPushstate(hash)
  }
  return replaceState.apply(window.history, arguments)
}

window.onPushstate = function () {
  // do something
}

window.onpopstate = function () {
   // do something
}

在页面的每次跳转中 监控页面的相关事件,并与 之前创建的 事件列表进行比对即可

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,219评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,363评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,933评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,020评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,400评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,640评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,896评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,597评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,327评论 1 244
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,581评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,072评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,399评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,054评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,849评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,672评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,585评论 2 270