亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

vue動(dòng)態(tài)渲染svg、添加點(diǎn)擊事件的實(shí)現(xiàn)

 更新時(shí)間:2020年03月13日 10:02:22   作者:小謳  
這篇文章主要介紹了vue動(dòng)態(tài)渲染svg、添加點(diǎn)擊事件的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

業(yè)務(wù)需求(vue項(xiàng)目中)

1.頁(yè)面展示svg內(nèi)容
2.監(jiān)聽(tīng)svg內(nèi)部的點(diǎn)擊事件
3.動(dòng)態(tài)改變svg內(nèi)部元素的屬性和值

html標(biāo)簽

經(jīng)多次實(shí)驗(yàn),用embed、img等標(biāo)簽改變src屬性的方式,均無(wú)法實(shí)現(xiàn)上述全部功能(尤其是svg內(nèi)部點(diǎn)擊事件),最終采用Vue.extend()方法完整實(shí)現(xiàn),代碼也較為簡(jiǎn)潔,html結(jié)構(gòu)如下:

<template>
 <div>
  <div id="svgTemplate"></div>
 </div>
</template>

直接將svg文件的內(nèi)容復(fù)制粘貼到.vue文件里,是可以在標(biāo)簽內(nèi)直接添加@click事件完成需求的,方式簡(jiǎn)單但會(huì)造成文件過(guò)長(zhǎng),本文不多陳述

實(shí)現(xiàn)思路

1.創(chuàng)建xhr對(duì)象

const xhr = new XMLHttpRequest();
this.svgUrl = ...; // svg的絕對(duì)地址,在瀏覽器中打開能看到的那個(gè)
xhr.open("GET", this.svgUrl, true);
xhr.send();

2.監(jiān)聽(tīng)xhr對(duì)象(獲取svg的dom -> 添加事件 -> 修改dom -> 轉(zhuǎn)成虛擬dom并掛載)

xhr.addEventListener("load", () => {
  // ① 獲取svg的dom
  const resXML = xhr.responseXML;
  this.svgDom = resXML.documentElement.cloneNode(true);   // console.log(this.svgDom);
  
  // ② 添加click事件
  let btn = this.svgDom.getElementById("...");
  btn.setAttribute("v-on:click", "this.handleClick()");
  // ↑↑↑ 此處注意:原生事件handleClick此時(shí)在window層,解決辦法見(jiàn)后文

  // ③ 修改 dom
  this.svgDom.getElementById("...").childNodes[0].nodeValue = ...
  this.svgDom.getElementById("...").setAttribute("style",
     `....; fill:${this.photoResult.resultColor}; ...`);
  // ↑↑↑ 用js操作dom的語(yǔ)法,動(dòng)態(tài)設(shè)置svg部件的屬性和值
  
  // ④ 將svgDom對(duì)象轉(zhuǎn)換成vue的虛擬dom,創(chuàng)建實(shí)例并掛載到元素上
  var oSerializer = new XMLSerializer();
  var sXML = oSerializer.serializeToString(this.svgDom);
  var Profile = Vue.extend({
    template: "<div id='svgTemplate'>" + sXML + "</div>"
  });
  new Profile().$mount("#svgTemplate");
});

3.將methods里要執(zhí)行的事件綁定到window下面,供外部(剛添加的 handleClick 事件)調(diào)用

async mounted() {
  window["handleClick"] = () => {
    this.takePhoto();
  };
},
methods:{
  takePhoto(){ ... }
}

到這里就基本完成需求:動(dòng)態(tài)渲染了svg、用js操作dom的語(yǔ)法修改svg部件的屬性和值、給svg部件動(dòng)態(tài)添加了事件 handleClick,最后將 takePhoto() 事件綁定給了 window 對(duì)象的 handleClick,可以放心大膽的在 takePhoto() 里寫你要執(zhí)行的內(nèi)容了!

特殊注意

給svg的dom部件添加事件時(shí):
1.經(jīng)多次嘗試,只有 setAttribute + v-on:click 寫法有效
2.setAttribute 不支持 @click(非原生事件),會(huì)報(bào)語(yǔ)法錯(cuò)誤
3.addEventListener 和 onclick 均會(huì)被 vue 攔截
將svgDom對(duì)象轉(zhuǎn)換成vue的虛擬dom時(shí):
1.如果報(bào)錯(cuò)如下

則將 import Vue from "vue" 改為 import Vue from "vue/dist/vue.esm.js"
其原因及其他解決辦法本文不做探討可自行百度。
2.vue.extend() 方法是 vue 的一個(gè)構(gòu)造器,用來(lái)動(dòng)態(tài)創(chuàng)建 vue 實(shí)例,template 組件模板只能有一個(gè)根元素
3.$mount 手動(dòng)掛載到 id 為 svgTemplate的 元素上,掛載后將替換原本的dom(替換原本的 <div id="svgTemplate"></div>)。由于每次更新 svg 都要重新掛載,沒(méi)有找到 dom 元素是無(wú)法掛載的,因此 template 里面最外層的 div 也要加上 id 的屬性:

var Profile = Vue.extend({
   template: "<div id='svgTemplate'>" + sXML + "</div>" 
   // ↑↑↑ 最外層的 id 不能省略,否則首次渲染后找不到 #svgTemplate
});
new Profile().$mount("#svgTemplate"); 
// ↑↑↑ 原本的 #svgTemplate 將被替換成 Profile 的 template

完整代碼

<template>
 <div>
  <div id="svgTemplate"></div>
 </div>
</template>
<script>
import Vue from "vue/dist/vue.esm.js";

// window.handleClick = () => {
  // 原本的 handleClick 事件是 window 的
// };

export default {
 name: "svg-drawing",
 data() {
  return {
   /* 全局 */
   svgUrl: "", // svg的url
   svgDom: null, // 獲取到的svg元素
   /* svg的變量 */
   photoResult: {
    resultVal: 0, // 測(cè)試結(jié)果 - 值
    resultMsg: "未檢測(cè)", // 測(cè)試結(jié)果 - 字段
    resultColor: "#dcdee2" // 測(cè)試結(jié)果 - 字段背景色
   }
  };
 },
 async mounted() {
  // 將takePhoto方法綁定到window下面,提供給外部調(diào)用
  window["handleClick"] = () => {
   this.takePhoto();
  };
 },
 created() {
  this.getSvg();
 },
 methods: {
  // 初始化svg
  getSvg() {
   /* 創(chuàng)建xhr對(duì)象 */
   const xhr = new XMLHttpRequest();
   this.svgUrl = this.baseUrl + "/svgs/" + "test.svg";
   xhr.open("GET", this.svgUrl, true);
   xhr.send();

   /* 監(jiān)聽(tīng)xhr對(duì)象 */
   xhr.addEventListener("load", () => {
    /* 1. 獲取 dom */
    const resXML = xhr.responseXML;
    this.svgDom = resXML.documentElement.cloneNode(true);

    /* 2.SVG對(duì)象添加click事件 */
    let btnTakePhotoDom = this.svgDom.getElementById("...");
    btnTakePhotoDom.setAttribute("v-on:click", "this.handleClick()");

    /* 3. 修改 dom */
    this.svgDom.getElementById("...").childNodes[0].nodeValue = ...;
    this.svgDom.getElementById("...").setAttribute("style",
     `....; fill:${this.photoResult.resultColor}; ...`);

    /* 4.將svgDom對(duì)象轉(zhuǎn)換成vue的虛擬dom */
    var oSerializer = new XMLSerializer();
    var sXML = oSerializer.serializeToString(this.svgDom);
    var Profile = Vue.extend({
     template: "<div id='svgTemplate'>" + sXML + "</div>"
    });
    // 創(chuàng)建實(shí)例,并掛載到元素上
    new Profile().$mount("#svgTemplate");
   });
  },
  // 事件
  takePhoto() { ... },
 },
 beforeDestroy() {
  this.svgDom = null;
 },
 watch: {
  photoResult: {
   handler(newVal, oldVal) {
    this.getSvg();
   },
   deep: true
  }
 }
};
</script>

到此這篇關(guān)于vue動(dòng)態(tài)渲染svg、添加點(diǎn)擊事件的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)vue動(dòng)態(tài)渲染svg、添加點(diǎn)擊事件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Vue3使用路由VueRouter4的簡(jiǎn)單示例

    Vue3使用路由VueRouter4的簡(jiǎn)單示例

    在vue.js項(xiàng)目中使用vue-router,可以使用路由進(jìn)行界面或路徑跳轉(zhuǎn),下面這篇文章主要給大家介紹了關(guān)于Vue3使用路由VueRouter4的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2021-07-07
  • 如何利用vue展示.docx文件、excel文件和csv文件內(nèi)容

    如何利用vue展示.docx文件、excel文件和csv文件內(nèi)容

    最近遇到了一些新的需求,需要前端實(shí)現(xiàn)文件預(yù)覽功能,下面這篇文章主要給大家介紹了關(guān)于如何利用vue展示.docx文件、excel文件和csv文件內(nèi)容的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-04-04
  • van-uploader保存文件到后端回顯后端接口返回的數(shù)據(jù)

    van-uploader保存文件到后端回顯后端接口返回的數(shù)據(jù)

    前端開發(fā)想省時(shí)間就是要找框架呀,下面這篇文章主要給大家介紹了關(guān)于van-uploader保存文件到后端回顯后端接口返回的數(shù)據(jù),文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • vue 解決文本框被鍵盤遮住的問(wèn)題

    vue 解決文本框被鍵盤遮住的問(wèn)題

    今天小編就為大家分享一篇vue 解決文本框被鍵盤遮住的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-11-11
  • vue中防抖和節(jié)流的使用方法

    vue中防抖和節(jié)流的使用方法

    防抖和節(jié)流是我們?cè)陂_發(fā)過(guò)程中常用優(yōu)化性能的方式,可以減少不必要的計(jì)算,不浪費(fèi)資源,只在適合的時(shí)候再進(jìn)行觸發(fā)計(jì)算,這篇文章主要給大家介紹了關(guān)于vue中防抖和節(jié)流使用的相關(guān)資料,需要的朋友可以參考下
    2021-08-08
  • Vue.js上傳圖片到阿里云OSS存儲(chǔ)的方法示例

    Vue.js上傳圖片到阿里云OSS存儲(chǔ)的方法示例

    這篇文章主要介紹了Vue.js上傳圖片到阿里云OSS存儲(chǔ)的方法示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-12-12
  • 解決vue項(xiàng)目運(yùn)行npm run serve報(bào)錯(cuò)的問(wèn)題

    解決vue項(xiàng)目運(yùn)行npm run serve報(bào)錯(cuò)的問(wèn)題

    這篇文章主要介紹了解決vue項(xiàng)目運(yùn)行npm run serve報(bào)錯(cuò)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-10-10
  • 在vue中使用vuex,修改state的值示例

    在vue中使用vuex,修改state的值示例

    今天小編就為大家分享一篇在vue中使用vuex,修改state的值示例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-11-11
  • Vue進(jìn)階之CodeMirror的應(yīng)用小結(jié)

    Vue進(jìn)階之CodeMirror的應(yīng)用小結(jié)

    CodeMirror支持在線編輯代碼,風(fēng)格包括js, java, php, c++等等100多種語(yǔ)言,下面這篇文章主要來(lái)和大家講講CodeMirror的應(yīng)用,感興趣的可以了解一下
    2023-06-06
  • 詳解Vue-cli 創(chuàng)建的項(xiàng)目如何跨域請(qǐng)求

    詳解Vue-cli 創(chuàng)建的項(xiàng)目如何跨域請(qǐng)求

    本篇文章主要介紹了詳解Vue-cli 創(chuàng)建的項(xiàng)目如何跨域請(qǐng)求 ,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-05-05

最新評(píng)論