Vue.js中動態(tài)更改svg的相關(guān)屬性詳解
引言
公司項目中有一個關(guān)于圖標(biāo)庫管理的需求,大致需要在頁面能夠動態(tài)去更改對應(yīng)svg圖標(biāo)的大小、顏色等(這里的更改顏色限制線性圖標(biāo))。在網(wǎng)上查找了相關(guān)資料,做了技術(shù)的預(yù)研及demo的編寫,在此記錄一下。
怎樣將一個遠(yuǎn)程的svg圖標(biāo)資源"下載"到本地
首頁我們可以利用XMLHttpRequest對象來請求對應(yīng)的svg圖標(biāo)的遠(yuǎn)程資源鏈接地址,并監(jiān)聽實現(xiàn)XMLHttpRequest對象的load事件,將返回的資源進(jìn)行dom對象的轉(zhuǎn)換、string轉(zhuǎn)換為xml。
代碼如下:
const xhr = new XMLHttpRequest();
xhr.open('GET', 'https://www.xx.com/img/xxx.svg', true);
xhr.send();
/* 監(jiān)聽xhr對象 */
xhr.onreadystatechange = function () {
if (xhr.readyState == 4 && xhr.status == 200) {
console.log(xhr.responseXML, 'xhr.responseXML---------')
}
};
xhr.addEventListener('load', () => {
const resXML = stringToXml(xhr.response);
this.svgDom = resXML.documentElement.cloneNode(true);
});這里的工具函數(shù)stringToXml的完整代碼如下:
//將字符串轉(zhuǎn)化成dom對象;string轉(zhuǎn)換為xml
function stringToXml (xmlString) {
let xmlDoc;
if (typeof xmlString == "string") {
//FF
if (document.implementation.createDocument) {
const parser = new DOMParser();
xmlDoc = parser.parseFromString(xmlString, "text/xml");
} else if (window.ActiveXObject) {
// eslint-disable-next-line no-undef
xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
xmlDoc.async = false;
xmlDoc.loadXML(xmlString);
}
}
else {
xmlDoc = xmlString;
}
return xmlDoc;
}這樣就可以獲取到遠(yuǎn)程svg資源對應(yīng)的dom結(jié)構(gòu)了。
怎樣更改svgdom結(jié)構(gòu)里面的相關(guān)屬性
產(chǎn)品的要求需要能夠動態(tài)更改對應(yīng)svg圖標(biāo)的寬、高、顏色值等。要實現(xiàn)這樣的功能有以下幾個小點:
- 將svgDom對象轉(zhuǎn)換成vue的虛擬dom,代碼如下:
const oSerializer = new XMLSerializer(); - 根據(jù)序列化的對象提供的
serializeToString方法將svgDom對象進(jìn)行字符串化;
通過svgDom對象提供的寬、高屬性值,結(jié)合正則來遍歷svgDom字符串化后的字符串,進(jìn)行寬高值的替換。代碼如下:
let sXML = oSerializer.serializeToString(this.svgDom);
sXML = sXML.replace(`width="${this.svgDom.width.baseVal.value}"`, 'width="40"').replace(`height="${this.svgDom.height.baseVal.value}"`, 'height="40"')- 根據(jù)
sXML來截取svg結(jié)構(gòu)表示的字符串里對應(yīng)的顏色值,并結(jié)合is-color這個插件判斷是否是一個真正的顏色,是的話,根據(jù)想要替換的顏色值進(jìn)行全局替換就行。代碼如下:
let curColor = sXML.split('#')[1].substr(0, 6)
if (!isColor(`#${curColor}`)) {
curColor = sXML.split('#')[1].substr(0, 3)
}
sXML = sXML.replace(new RegExp(`#${curColor}`, "gm"), '#90EE90')- 通過
Vue實例提供的extend方法創(chuàng)建實例并掛載到某個元素上,代碼如下:
const Profile = Vue.extend({
template: "<div id='svgTemplate'>" + sXML + '</div>'
});
// 創(chuàng)建實例,并掛載到元素上
new Profile().$mount('#svgTemplate');處理前的效果圖:

處理后的效果圖(將svg寬高由原來的20變?yōu)?0,將顏色值改為"#90EE90"):

最終完整的代碼如下:
testSvg () {
const xhr = new XMLHttpRequest();
xhr.open('GET', 'https://www.xx.com/img/xxx.svg', true);
xhr.send();
/* 監(jiān)聽xhr對象 */
xhr.onreadystatechange = function () {
if (xhr.readyState == 4 && xhr.status == 200) {
console.log(xhr.responseXML, 'xhr.responseXML---------')
}
};
xhr.addEventListener('load', () => {
const resXML = stringToXml(xhr.response);
this.svgDom = resXML.documentElement.cloneNode(true);
/* 將svgDom對象轉(zhuǎn)換成vue的虛擬dom */
const oSerializer = new XMLSerializer();
let sXML = oSerializer.serializeToString(this.svgDom);
let curColor = sXML.split('#')[1].substr(0, 6)
if (!isColor(`#${curColor}`)) {
curColor = sXML.split('#')[1].substr(0, 3)
}
sXML = sXML.replace(`width="${this.svgDom.width.baseVal.value}"`, 'width="40"').replace(`height="${this.svgDom.height.baseVal.value}"`, 'height="40"').replace(new RegExp(`#${curColor}`, "gm"), '#90EE90')
const Profile = Vue.extend({
template: "<div id='svgTemplate'>" + sXML + '</div>'
});
// 創(chuàng)建實例,并掛載到元素上
new Profile().$mount('#svgTemplate');
});
},以上就是Vue.js中動態(tài)更改svg的相關(guān)屬性詳解的詳細(xì)內(nèi)容,更多關(guān)于Vue.js動態(tài)更改svg屬性的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
vue+axios實現(xiàn)文件下載及vue中使用axios的實例
這篇文章主要介紹了vue+axios實現(xiàn)文件下載及vue中使用axios的實例,需要的朋友可以參考下2018-09-09
父子組件生命周期及子組件獲取數(shù)據(jù)傳值問題剖析
這篇文章主要介紹了父子組件生命周期及子組件獲取數(shù)據(jù)問題剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10
elementui使用el-upload組件如何實現(xiàn)自定義上傳
這篇文章主要介紹了elementui使用el-upload組件如何實現(xiàn)自定義上傳,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08
Vue中的transition封裝組件的實現(xiàn)方法
這篇文章主要介紹了Vue中的transition封裝組件的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
Vue.js中class與style的增強(qiáng)綁定實現(xiàn)方法
由于Class和Style綁定使用頻繁,字符串拼接麻煩且易錯,因此,Vue.js 做了專門的增強(qiáng),表達(dá)式結(jié)果的類型除了字符串之外,還可以是對象或數(shù)組,本文給大家講解Vue.js中class與style的增強(qiáng)綁定知識,感興趣的朋友一起看看吧2023-04-04
Uniapp 實現(xiàn)頂部標(biāo)簽頁切換功能(詳細(xì)步驟)
本文介紹了如何在UniApp中實現(xiàn)頂部標(biāo)簽頁切換功能,u-tab-bar組件提供了便捷的標(biāo)簽切換功能和豐富的樣式選項,而swiper組件則更加靈活,支持自定義切換方式,根據(jù)自己的需求選擇合適的方式實現(xiàn)頂部標(biāo)簽頁切換,感興趣的朋友一起看看吧2025-02-02
vue中radio根據(jù)動態(tài)值綁定checked無效的解決
這篇文章主要介紹了vue中radio根據(jù)動態(tài)值綁定checked無效的解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03

