关于vConsole 源码的理解分享(vConsole一个移动端调试控制台工具)(1)

2023-12-20 01:58

本文主要是介绍关于vConsole 源码的理解分享(vConsole一个移动端调试控制台工具)(1),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一个轻量、可拓展、针对手机网页的前端开发者调试面板。

特性

  • 查看 console 日志
  • 查看网络请求
  • 查看页面 element 结构
  • 查看 Cookies 和 localStorage
  • 手动执行 JS 命令行
  • 自定义插件

这是github的readme介绍,对于调试移动端以及上线后出现的一些问题确实是一个很好的工具!! 为腾讯开源点个赞。
使用方法请参考:腾讯开源vConsle移动端调试控制台工具 (请原谅我把它叫的那么长哈哈哈)

源码分析

总的来说,vConsole的源码还是很清晰的,采用了es6的语法,以面向对象的形式即类的显示来组织代码,按照控制台的ui来分割,这一点值得借鉴和学习。

源码代码结构

VConsole的类结构图

简单的画了一个草图,首先要明确是 VConsole为核心类,负责插件的生成,事件绑定,运行等,而vConsolePlugin 类似一个抽象类,具体实现渲染绑定等由子类去具体实现。所以VConsole的扩展是通过添加VConsolePlugin的实现来的。(如果有歧义请指正批评)

模板引擎

vConsole采用的是一个叫做 Mito.js 的 一个简单的模板引擎(估计是腾讯自己内部写的一个模板引擎),这个文件在lib文件夹下,代码很简短就一个rener方法,然后配合模板html使用

//使用指定数据将模板文本编译成 element 对象或者 HTML 字符串。
//(required) tpl: 模板字符串。
//(required) data: 一组 key-value 形式的数据源。
//(optional) toString: 布尔值,用于设定返回值为 element 对象还是 HTML 字符串,默认为 `false`。
//返回值 Element 对象或者 HTML 字符串
function render(tpl, data, toString)

模板语法涉及到了常用的if else ,for和switch 常用的流程语句,对付一些简单的需求这个已经足够了,详细语法使用参考官方文档的helper_function.md

核心模块
核心模块主要是core.js的这个js文件,也就是VConsole类的定义 。

首先来看下这个类的Construction方法,我把注释写在代码里 ,这样看起来 应该很清晰了

constructor(opt) {//判断是否已经存在VConsole实例了,通过判断document文档结构是否存在了VCONSOLE_ID即(#__vconsole) 这个id。if (!!$.one(VCONSOLE_ID)) {console.debug('vConsole is already exists.');return;}//将this用that保存起来,这里的作用不言而喻,就是为了防止接下来的操作(方法)调用会改变this的指向let that = this;//版本号this.version = pkg.version;//基础的dom 可以理解为root  即id为VCONSOLE_ID的文档元素this.$dom = null;//再次判断是否已经初始化的标识,和一开始的方法一样的作用this.isInited = false;//配置选项 以下是对加载哪几个默认插件的配置this.option = {defaultPlugins: ['system', 'network', 'element', 'storage']};//当前活动的tab页框this.activedTab = '';//tab页框数组this.tabList = [];//插件列表this.pluginList = {};//vConsole的悬浮按钮的配置(位置)this.switchPos = {x: 10, // righty: 10, // bottomstartX: 0,startY: 0,endX: 0,endY: 0};// 暴露一些公用的方法出去,比如判断是否对象,数组等  详见在../lib/tool.js的文件this.tool = tool;//暴露$的方法,query.js文件夹this.$ = $;// 合并用户配置和默认配置。for in 用法了解下if (tool.isObject(opt)) {for (let key in opt) {this.option[key] = opt[key];}}//初始化添加插件 插件的初始化 添加到tabListthis._addBuiltInPlugins();// Vconsole的UI加载,let _onload = function() {if (that.isInited) {return;}//渲染VConsole的主面板,swicth按钮that._render();//面板上的交互 模拟 触摸滑动that._mockTap();//绑定事件 拖动swicth按钮,点击面板的tab等that._bindEvent();//面板渲染完成后的插件自动加载渲染that._autoRun();};//当document渲染加载完成  调用_onload方法if (document !== undefined) {if (document.readyState == 'complete') {_onload();} else {$.bind(window, 'load', _onload);}} else {// if document does not exist, wait for itlet _timer;let _pollingDocument = function() {if (!!document && document.readyState == 'complete') {_timer && clearTimeout(_timer);_onload();} else {_timer = setTimeout(_pollingDocument, 1);}};_timer = setTimeout(_pollingDocument, 1);}}

总的流程用流程图表示

Created with Raphaël 2.1.2 开始 初始化一些参数:tablist,Pluginlist,版本号等 加载初始化配置的插件 是否加载document完成 VConsole加载UI,绑定事件,运行插件等 结束 等待document加载完 yes no

大概VConsole类的流程和做的一些事情都阐述完了,接下来分析一些重要的方法

addPlugin();

添加一个新的插件,这个为基本组件如何加到VConsole的公用方法,实现加入到VConsole体系,并且运行起来。
/*** add a new plugin* @public* @param object VConsolePlugin object* @return boolean*/addPlugin(plugin) {// ignore this plugin if it has already been installedif (this.pluginList[plugin.id] !== undefined) {console.debug('Plugin ' + plugin.id + ' has already been added.');return false;}this.pluginList[plugin.id] = plugin;// init plugin only if vConsole is readyif (this.isInited) {this._initPlugin(plugin);// if it's the first plugin, show it by defaultif (this.tabList.length == 1) {this.showTab(this.tabList[0]);}}return true;}

而在这个方法中最重要的方法就是下面这个方法_initPlugin();初始化插件,包括加入panel中的点击事件等一些操作,其中的insertAdjacentElement()方法相当于jq中的insertAfter和insertBefore等方法

/*** init a plugin* @private*/_initPlugin(plugin) {let that = this;plugin.vConsole = this;// start initplugin.trigger('init');// render tab (if it is a tab plugin then it should has tab-related events)plugin.trigger('renderTab', function(tabboxHTML) {// add to tabListthat.tabList.push(plugin.id);// render tabbarlet $tabbar = $.render(tplTabbar, { id: plugin.id, name: plugin.name });$.one('.vc-tabbar', that.$dom).insertAdjacentElement('beforeend', $tabbar);// render tabboxlet $tabbox = $.render(tplTabbox, { id: plugin.id });if (!!tabboxHTML) {if (tool.isString(tabboxHTML)) {$tabbox.innerHTML += tabboxHTML;} else if (tool.isFunction(tabboxHTML.appendTo)) {tabboxHTML.appendTo($tabbox);} else if (tool.isElement(tabboxHTML)) {$tabbox.insertAdjacentElement('beforeend', tabboxHTML);}}$.one('.vc-content', that.$dom).insertAdjacentElement('beforeend', $tabbox);});// render top barplugin.trigger('addTopBar', function(btnList) {if (!btnList) {return;}let $topbar = $.one('.vc-topbar', that.$dom);for (let i = 0; i < btnList.length; i++) {let item = btnList[i];let $item = $.render(tplTopBarItem, {name: item.name || 'Undefined',className: item.className || '',pluginID: plugin.id});if (item.data) {for (let k in item.data) {$item.dataset[k] = item.data[k];}}if (tool.isFunction(item.onClick)) {$.bind($item, 'click', function(e) {let enable = item.onClick.call($item);if (enable === false) {// do nothing} else {$.removeClass($.all('.vc-topbar-' + plugin.id), 'vc-actived');$.addClass($item, 'vc-actived');}});}$topbar.insertAdjacentElement('beforeend', $item);}});// render tool barplugin.trigger('addTool', function(toolList) {if (!toolList) {return;}let $defaultBtn = $.one('.vc-tool-last', that.$dom);for (let i = 0; i < toolList.length; i++) {let item = toolList[i];let $item = $.render(tplToolItem, {name: item.name || 'Undefined',pluginID: plugin.id});if (item.global == true) {$.addClass($item, 'vc-global-tool');}if (tool.isFunction(item.onClick)) {$.bind($item, 'click', function(e) {item.onClick.call($item);});}$defaultBtn.parentNode.insertBefore($item, $defaultBtn);}});// end initplugin.isReady = true;plugin.trigger('ready');}

这个函数是初始化插件,例如日志模块,网络模块都是以插件的形式,集成进来的,既然是插件,,那就要有插槽,自然,插件提供插口。这个函数就是一个插槽,当插件集成进来的时候,就通过触发插槽里的插口来实现调用插件的方法,这个有五个个触发的方法,插件中实现了就会触发

  • init 初始化插件
  • renderTab 渲染tab 基础的如log system这个
  • addTopBar 渲染 bar栏 tab下的子栏目
  • addTool 添加工具方法 如clear hide
  • ready 插件准备完毕,开始输出 log 等
VConsolePlugin 这个抽象类做了什么
class VConsolePlugin {constructor(id, name = 'newPlugin') {//id 为log system 来标识插件this.id = id;this.name = name;//组件的准备状态  在VConsole的初始化完插件 设置为true,即组件渲染完成this.isReady = false;//事件列表this.eventList = {};}get id() {return this._id;}set id(value) {if (!value) {throw 'Plugin ID cannot be empty';}this._id = value.toLowerCase();}get name() {return this._name;}set name(value) {if (!value) {throw 'Plugin name cannot be empty';}this._name = value;}//获取VConsole实例get vConsole() {return this._vConsole || undefined;}set vConsole(value) {if (!value) {throw 'vConsole cannot be empty';}this._vConsole = value;}/*** register an event* @public* @param string* @param function*/on(eventName, callback) {this.eventList[eventName] = callback;return this;}/*** trigger an event* @public* @param string* @param mixed*/trigger(eventName, data) {if (typeof this.eventList[eventName] === 'function') {// registered by `.on()` methodthis.eventList[eventName].call(this, data);} else {// registered by `.onXxx()` methodlet method = 'on' + eventName.charAt(0).toUpperCase() + eventName.slice(1);if (typeof this[method] === 'function') {this[method].call(this, data);}}return this;}} // END class

其实也没做什么事,但是有一个很重要的trigger方法

/*** trigger an event* @public* @param string* @param mixed*/trigger(eventName, data) {if (typeof this.eventList[eventName] === 'function') {// registered by `.on()` methodthis.eventList[eventName].call(this, data);} else {// registered by `.onXxx()` methodlet method = 'on' + eventName.charAt(0).toUpperCase() + eventName.slice(1);if (typeof this[method] === 'function') {this[method].call(this, data);}}return this;}

这样一来,只要每个插件的方法 以 on 开头 ,就可以通过基类触发方法的执行了,具体的逻辑是先去找 on注册的,如果没有 就去实例中找,还有一点这个方法是返回对象本身的。

这篇关于关于vConsole 源码的理解分享(vConsole一个移动端调试控制台工具)(1)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/514446

相关文章

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

Python办公自动化实战之打造智能邮件发送工具

《Python办公自动化实战之打造智能邮件发送工具》在数字化办公场景中,邮件自动化是提升工作效率的关键技能,本文将演示如何使用Python的smtplib和email库构建一个支持图文混排,多附件,多... 目录前言一、基础配置:搭建邮件发送框架1.1 邮箱服务准备1.2 核心库导入1.3 基础发送函数二、

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

Python中你不知道的gzip高级用法分享

《Python中你不知道的gzip高级用法分享》在当今大数据时代,数据存储和传输成本已成为每个开发者必须考虑的问题,Python内置的gzip模块提供了一种简单高效的解决方案,下面小编就来和大家详细讲... 目录前言:为什么数据压缩如此重要1. gzip 模块基础介绍2. 基本压缩与解压缩操作2.1 压缩文

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

sqlite3 命令行工具使用指南

《sqlite3命令行工具使用指南》本文系统介绍sqlite3CLI的启动、数据库操作、元数据查询、数据导入导出及输出格式化命令,涵盖文件管理、备份恢复、性能统计等实用功能,并说明命令分类、SQL语... 目录一、启动与退出二、数据库与文件操作三、元数据查询四、数据操作与导入导出五、查询输出格式化六、实用功

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)