最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • 基于vue开发chrome插件,实现获取界面数据和保存到数据库功能

    正文概述 掘金(启晨1991)   2020-12-01   397

    基于vue开发chrome插件,实现获取界面数据和保存到数据库功能

    前言

    最近在评估项目时,要开启评估平台,查看平台和保存平台,感觉非常繁琐,开发了一款可以获取评估平台数据,查看项目排期和直接保存数据到数据库的chrome插件,由于我需要使用之前vue封装的一个日历插件,这里就用vue来开发这个插件。

    开发前准备

    要开发一个chrome插件,我们首先需要了解chrome插件的基本结构和对应的功能。
    每个扩展的文件类型和目录数量有所不同,但都必须有 manifest。 一些基本但有用的扩展程序可能仅由 manifest 及其工具栏图标组成。

    manifest.json

      {
        "name": "My Extension",  // "扩展名"
        "version": "2.1", // 当前创建扩展版本号
        "description": "Gets information from Google.", //"扩展描述"
        "icons": {  // 扩展工具界面使用图标
          "128": "icon_16.png",
          "128": "icon_32.png",
          "128": "icon_48.png",
          "128": "icon_128.png"
        },
        "background": {  // 扩展常常用一个单独的长时间运行的脚本来管理一些任务或者状态
          "persistent": false,
          "scripts": ["background_script.js"]  // 后台常驻脚本,自动运行,直到关闭浏览器。可根据需求自行设置
        },
        "permissions": ["https://*.google.com/", "activeTab"],  //开启拓展权限
        "browser_action": { 
          "default_icon": "icon_16.png",   // 器右上角显示
          "default_popup": "popup.html"  /** 鼠标移入,显示简短扩展文本描述 **/
        },
         "content_scripts": [{   // ontent scripts是在Web页面内运行的javascript脚本。通过使用标准的DOM,它们可以获取浏览器所访问页面的详细信息,并可以修改这些信息。
        "js": ["script/contentscript.js"], /** 需要注入的脚本 **/
        "matches": [   /**  匹配网址(支持正则),成功即注入(其余属性自行查询) **/
            "http://*/*",
            "https://*/*"
          ]
        }]
       }
    

    vue开发chrome插件

    我们需要使用vue来开发插件,几经搜索,查到一款样板,很方便我们进行vue开发插件,便引入该样板来进行开发。

    引入vue-web-extension样板来实现vue开发

      npm install -g @vue/cli
      npm install -g @vue/cli-init
      vue init kocal/vue-web-extension new-tab-page
    

    然后切换到项目目录安装依赖项

      cd new-tab-page
      npm install
    

    我们可以运行

      npm run watch:dev
    

    在项目根目录中会得到一个dist 文件夹,我们直接安装解压的扩展程序,选择这个dist,就可以进行开发并监视更改。

    样板文件的基本格式

    ├── dist
    │ └── <the built extension>
    ├── node_modules
    │ └── <one or two files and folders>
    ├── package.json
    ├── package-lock.json
    ├── scripts
    │ ├── build-zip.js
    │ └── remove-evals.js
    ├── src
    │ ├── background.js
    │ ├── icons
    │ │ ├── icon_128.png
    │ │ ├── icon_48.png
    │ │ └── icon.xcf
    │ ├── manifest.json
    │ └── popup
    │ ├── App.vue
    │ ├── popup.html
    │ └── popup.js
    └── webpack.config.js
    

    在 package.json 文件中还声明了各种脚本。我们将用 npm run watch:dev 来开发扩展,然后使用 npm run build-zip 生成一个ZIP文件以上传到 Chrome Web Store。

    创建插件界面

    我们直接修改popup.html

    popup.html

    <!DOCTYPE html>
    <html lang="zh">
    <head>
      <meta charset="UTF-8">
      <title>Title</title>
    </head>
    <body>
      <link href="popup.css" rel="stylesheet">
      <div id="app">
      </div>
      <script src="popup.js"></script>
    </body>
    </html>
    

    这里我们引入popup.css和popup.js 在popup.css放入我们需要用的样式 在popup.js中,来引入我们的vue文件

    popup.js

      import Vue from 'vue'
     import { Tabs,TabPane, Dialog, Button,Form,FormItem,Input,DatePicker,Message,Alert,Tooltip,MessageBox } from 'element-ui';
     import 'element-ui/lib/theme-chalk/index.css';
     import App from './App'
     Vue.use(Tabs);
     Vue.use(TabPane);
     Vue.use(Dialog);
     Vue.use(Button);
     Vue.use(Form);
     Vue.use(FormItem);
     Vue.use(Input);
     Vue.use(DatePicker);
     Vue.use(Tooltip);
     Vue.use(Alert);
     Vue.prototype.$message = Message;
     Vue.prototype.$confirm = MessageBox.confirm;
     new Vue({
       el: '#app',
       render: h => h(App)
     })
    

    这里,我们主要按需引入element-ui中的控件,和app.vue组件

    app.vue

      <template>
     <div id="app" style="height: 580px;overflow-y: hidden;width:680px;">
       <div>
         模板
       </div>
       <customPlan :projectData="projectData" :loginPerson="loginPerson"></customPlan>
     </div>
    </template>
    
    <script>
    import customPlan from '../components/customPlan'
    let { Pinyin } = require('../script/pinyin')
    let pinyin = new Pinyin()
    export default {
     components: { customPlan },
     data() {
       return {
         loginPerson: '',
         projectData: {
           departmentName: '',
           developer: '',
           endDate: '',
           evaluator: '',
           isDeprecated: false,
           isIncludeSaturday: false,
           isNewComponent: false,
           issureAdress: '',
           msg: '',
           name: '',
           startDate: '',
           workDay: '',
           year: 2020
         }
       }
     },
     created() {
       this.getUrl()
     },
     methods: {  
       getCaption(obj) {
         var index = obj.lastIndexOf(',')
         obj = obj.substring(index + 1, obj.length)
         return obj
       },
       /**   
        * @desc 获取当前页面的url
        */
       getUrl() {
         chrome.tabs.getSelected(null, tab => {
           console.log(tab,"tab")
           this.projectData.issureAdress = tab.url
           chrome.tabs.sendMessage(tab.id, { greet: 'hello' }, response => {
             if (response && response.developer && response.processName) {
               let developer = pinyin
                 .getFullChars(this.getCaption(response.developer))
                 .toLowerCase()
               this.projectData.evaluator = developer
               this.projectData.name = response.processName
             } else if (response && response.developer && !response.processName) {
               var index = response.developer.lastIndexOf('@')
               response.developer = response.developer.substring(
                 index + 1,
                 response.developer.length
               )
               this.loginPerson = response.loginPerson
               this.projectData.evaluator = response.developer
               this.projectData.name =response.peocessName
             }
           })
         })
       }
     }
    }
    </script>
    

    在manifest.json中引入

       "browser_action": {
         "default_title": "测试",
         "default_popup": "popup/popup.html"
       },
    

    这里我们主要引入了我们的日历控件customPlan,大家可以按需引入自己需要的组件。到这里,我们的插件界面基本搭建完成了。

    获取当前界面数据,并在插件中进行监听

    需要获取当前界面数据,就需要在Web页面内运行的javascript脚本。通过使用标准的DOM,它们可以获取浏览器所访问页面的详细信息,并可以修改这些信息。就需要content_scripts里面引入我们需要的contentscript.js文件,在这个js文件中,可以获取浏览器所访问页面的详细信息

      "content_scripts": [{
        "js": ["script/contentscript.js"],
        "matches": [
          "http://*/*",
          "https://*/*"
        ]
      }]
    

    contentscript.js文件配置如下

    document.addEventListener('click', function (e) {
        let isCurrect = e.path.length > 3&&e.path[4].innerText&&e.path[4].innerText.indexOf('提交需求') != -1 && e.target.innerText === '确 定' && document.getElementsByClassName('layout-nav') && document.getElementsByClassName('layout-nav')[0].children
        if (isCurrect) {
            if (document.getElementsByClassName('user-table') && document.getElementsByClassName('user-table')[0] && document.getElementsByClassName('user-table')[0].getElementsByClassName('el-table__row').length > 0) {
                var port = chrome.runtime.connect({ name: "custommanage" });//通道名称
                let loginPerson = document.getElementsByClassName('layout-nav') && document.getElementsByClassName('layout-nav')[0].children ? document.getElementsByClassName('layout-nav')[0].children[0].innerText : ''
                let partMentName = document.getElementsByClassName('layout-nav') && document.getElementsByClassName('layout-nav')[0].children ? document.getElementsByClassName('layout-nav')[0].children[3].innerText : ''
                let processName = document.getElementsByClassName('el-input__inner') && document.getElementsByClassName('layout-nav')[0].children ? document.getElementsByClassName('el-input__inner')[0].title : ''
                let tableElement = document.getElementsByClassName('user-table') ? document.getElementsByClassName('user-table')[0].getElementsByClassName('el-table__row') : []
                let choseSelect = []
                for (let value of tableElement) {
                    if (value.innerText.indexOf(partMentName) !== -1) {
                        choseSelect = value
                    }
                }
                let developPerson = ''
                let startTime = ''
                let endTime = ''
                if (choseSelect && choseSelect.getElementsByTagName('td')) {
                    developPerson = choseSelect.getElementsByTagName('td')[1].innerText
                    startTime = choseSelect.getElementsByTagName('td')[3].getElementsByTagName('input')[0].title
                    endTime = choseSelect.getElementsByTagName('td')[4].getElementsByTagName('input')[0].title
                }
                let item = {
                    "loginPerson": loginPerson,
                    "processName": processName,
                    "developPerson": developPerson,
                    "startTime": startTime,
                    "endTime": endTime
                }
                port.postMessage(item);//发送消息   
            } else {
                alert('未查到该项目预排人员与预排时间,请点开插件或打开定制管理系统手动添加项目!')
            }
        }
    });
    

    这里获取元素就是js基本知识了。主要使用chrome插件的api

    chrome.runtime.connect

    • 保持长期连接的模式,在content scripts与Chrome扩展程序页面之间建立通道(可以为通道命名),可以处理多个消息。在通道的两端分别拥有一个chrome.runtime.Port对象,用以收发消息。这里主要在我们点击需要的按钮时,就会向chrome插件发送消息。

    在content scripts主动建立通道如下:

     var port = chrome.runtime.connect({name: "custommanage"});//通道名称
     port.postMessage({joke: "Knock knock"});//发送消息
     port.onMessage.addListener(function(msg) {//监听消息
         port.postMessage({answer: "custommanage"});
     });
    

    获取到界面信息后,在content scripts发生请求消息给Google Chrome扩展程序,我们在插件中就需要获取获取的界面信息了

    chrome扩展获取信息

    我们在background.js中建立通道,获取web界面传回的信息

    chrome.tabs.query(
      { active: true, currentWindow: true },
      function (tabs) {
        var port = chrome.tabs.connect(//建立通道
          tabs[0].id,
          { name: "custommanage" }//通道名称
        );
      });
    chrome.runtime.onConnect.addListener((port) => {
      console.assert(port.name == "custommanage");
      port.onMessage.addListener((res) => {   
          addActon(res)
      });
    });
    

    addAction函数即是保存我们获取的数据到数据库。

     /**
        * @desc 添加获取数据到数据库
        */
    function addProject (params) {   
          let paramsObj = Object.assign({},  params)
          let optsUpdata = {
            method: 'POST', //请求方法
            body: JSON.stringify(paramsObj), //请求体
            headers: {
              Accept: 'application/json',
              'Content-Type': 'application/json'
            }
          }
          fetch('http://****/api/EditConfirmWork', optsUpdata)
            .then(response => {
              return response.json()
            })
            .then(data => {
              if (data.code === 0) {
                alert('更新成功!')
              }
            })
            .catch(error => {
              alert(error)
            })
    }
    

    这里我们采用fetch函数来连接数据库,和修改数据库,后端接口也需要做一些跨域相关处理,才能正常连接,我这里用的Node开发的后端,大致代码如下

    //跨域
    app.all('*', function (req, res, next) {
      res.header("Access-Control-Allow-Origin", "*"); 
      res.header('Access-Control-Allow-Methods', 'PUT, GET, POST, DELETE, OPTIONS');
      res.header('Access-Control-Allow-Headers', 'Origin, X-Requested-With, Content-Type, Accept');
      res.header('Access-Control-Allow-Credentials', true)
      next();
    });
    

    到此,获取界面数据,并自动保存到数据库功能已完成,background.js我们在manifest.json引用下。

    "background": {
        "scripts": ["script/background.js"]
      },
    

    我们需要将编辑好的插件通过webpack打包,还需要在webpack.config.js配置一下,然后运行npm run watch:dev 就可以得到我们需要的dist,安装到扩展程序就可使用了。

    webpack.config.js配置如下

    const webpack = require('webpack');
    const ejs = require('ejs');
    const MiniCssExtractPlugin = require('mini-css-extract-plugin');
    const WebpackShellPlugin = require('webpack-shell-plugin');
    const CopyWebpackPlugin = require('copy-webpack-plugin');
    const ChromeExtensionReloader = require('webpack-chrome-extension-reloader');
    const { VueLoaderPlugin } = require('vue-loader');
    const { version } = require('./package.json');
    
    const config = {
      mode: process.env.NODE_ENV,
      context: __dirname + '/src',
      entry: {
        'popup/popup': './popup/popup.js',
        'script/contentscript': './script/contentscript.js',
        'script/background': './script/background.js'
      },
      output: {
        path: __dirname + '/dist',
        filename: '[name].js',
      },
      resolve: {
        extensions: ['.js', '.vue'],
      },  
      module: {
        rules: [
          {
            test: /\.vue$/,
            loaders: 'vue-loader',
          },
          {
            test: /\.js$/,
            loader: 'babel-loader',
            exclude: /node_modules/,
          },
          {
            test: /\.css$/,
            use: [MiniCssExtractPlugin.loader, 'css-loader'],
          },
          {
            test: /\.scss$/,
            use: [MiniCssExtractPlugin.loader, 'css-loader', 'sass-loader'],
          },
          {
            test: /\.sass$/,
            use: [MiniCssExtractPlugin.loader, 'css-loader', 'sass-loader?indentedSyntax'],
          },
          {
            test: /\.(png|jpg|gif|svg|ico)$/,
            loader: 'file-loader',
            options: {
              name: '[name].[ext]?emitFile=false',
            },
          },
          {
            test: /\.(eot|svg|ttf|woff|woff2)(\?\S*)?$/,
            loader: 'url-loader',
            options: {
              esModule: false,
              limin: 10000,
              name: "font/[name].[hash:8].[ext]"
            }
          }
        ],
      },
      plugins: [    
        new VueLoaderPlugin(),
        new MiniCssExtractPlugin({
          filename: '[name].css',
        }),
        new CopyWebpackPlugin([
          { from: 'icons', to: 'icons', ignore: ['icon.xcf'] },
          { from: 'popup/popup.html', to: 'popup/popup.html', transform: transformHtml },
          {
            from: 'manifest.json',
            to: 'manifest.json',
            transform: (content) => {
              const jsonContent = JSON.parse(content);
              jsonContent.version = version;
    
              if (config.mode === 'development') {
                jsonContent['content_security_policy'] = "script-src 'self' 'unsafe-eval'; object-src 'self'";
              }
    
              return JSON.stringify(jsonContent, null, 2);
            },
          },
        ])
      ],
    };
    
    if (config.mode === 'production') {
      config.plugins = (config.plugins || []).concat([
        new webpack.DefinePlugin({
          'process.env': {
            NODE_ENV: '"production"',
          },
        }),
      ]);
    }
    
    if (process.env.HMR === 'true') {
      config.plugins = (config.plugins || []).concat([
        new ChromeExtensionReloader(),
      ]);
    }
    
    function transformHtml(content) {
      return ejs.render(content.toString(), {
        ...process.env,
      });
    }
    
    module.exports = config;
    
    

    我们数据改变后,如果想点开插件就查看对应界面,这里就按需引入我们需要的组件,来实现不同的界面展示。

    最后附上manifest.json完整的配置

      {
      "name": "插件",
      "description": "描述",
      "version": 2.0,
      "manifest_version": 2,
      "icons": {
        "48": "icons/icon_426.png",
        "128": "icons/icon_426.png"
      },
      "browser_action": {
        "default_title": "插件",
        "default_popup": "popup/popup.html"
      },
      "permissions": [
        "tabs",
        "<all_urls>"
      ],
      "background": {
        "scripts": ["script/background.js"]
      },
      "content_scripts": [{
        "js": ["script/contentscript.js"],
        "matches": [
          "http://*/*",
          "https://*/*"
        ]
      }]
    }
    

    参考

    www.cnblogs.com/champagne/p… www.jianshu.com/p/b3e544162… github.com/facert/chro…


    起源地下载网 » 基于vue开发chrome插件,实现获取界面数据和保存到数据库功能

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元