【小程序】uniapp自定义图标组件可动态更换svg颜色

组件描述

通过图标名称加载对应svg,size参数调整图标大小,color参数调整图标颜色

解决思路:

  1. 存svg获svg,对象方式
  2. 正则替换svg的fill值,不改变源文件,通过base64直接加载
  3. 缓存svg源文件,避免重复读取文件

一、手动上传svg文件

将项目需要的svg文件保存到static目录下,通过svg.js(如下图)将icon名称和路径映射,方便后续通过名称加载svg文件

提示:可以考虑在main.js文件里设置全局对象,不使用icon组件在不同页面也可以访问svg文件

typescript 复制代码
import svg from './utils/svg.js'
Vue.prototype.$svg = svg;

二、使用Image组件加载图片资源

1.设置image

html 复制代码
	<image class="icon" :style="{ width: `${size || 18}px`, height: `${size || 18}px` }" :src="svgData"
		mode="scaleToFill" />

2.组件参数

typescript 复制代码
	props: {
		title: {
			type: String,
			default: '',
			required: true,
		},
		size: {
			type: String,
			default: '18',
		},
		color: {
			type: String,
			default: '#999',
		},
	},

3.读取缓存对象

typescript 复制代码
	watch: {
		color(newVal) {
			let svgData = this.svgCache[this.$svg[this.title]];
			if (svgData && newVal) {
				let newFile = this.changeColor(svgData, newVal);
				this.svgData = that.svgToBase64(newFile)
			}
		}
	},

三、读取svg文件,正则替换fill的color值

动态更换svg颜色,原理就是修改svg的fill属性

1.读取svg文件,写入缓存对象

typescript 复制代码
		getSvgFile(src) {
		//读取svg文件
			let that = this;
			const fs = wx.getFileSystemManager();
			fs.readFile({
				filePath: src,
				encoding: 'UTF-8',
				position: 0,
				success(res) {
					let data = res.data;
					if (data) {
						if (data instanceof ArrayBuffer) {
							// 有些设备数据格式是ArrayBuffer需要转换成svg字符串
							that.svgCache[src] = that.arrayBufferToStr(data);
						} else {
							that.svgCache[src] = data;
						}
						let newFile = that.changeColor(that.svgCache[src], that.color)
						that.svgData = that.svgToBase64(newFile);
					}
				},
				fail(res) {
					console.error(res)
				}
			})
		},
		changeColor(data, color) {
			// console.log(data);
			let newSvg;
			if (/fill=".*?"/.test(data)) {
				newSvg = data?.replace(/fill=".*?"/g, `fill="${color}"`);  // SVG有默认色
			} else {
				newSvg = data?.replace(/<svg /g, `<svg fill="${color}" `); // 无默认色
			}
			return newSvg
		},

2.使用Uint8Array进行文件数据转换

替换color值的svg字符串是容易转换成base64,在不改动svg源文件情况下,image可以直接加载处理后的base64

typescript 复制代码
		arrayBufferToStr(buffer) {
			let str = '';
			const uint8Array = new Uint8Array(buffer);
			for (let i = 0; i < uint8Array.length; i++) {
				str += String.fromCharCode(uint8Array[i]);
			}
			return str;
		},
		svgToBase64(svg) {
			const utf8 = new Uint8Array(svg.length);
			for (let i = 0; i < svg.length; i++) {
				utf8[i] = svg.charCodeAt(i);
			}
			// 使用Buffer进行Base64编码
			const base64 = wx.arrayBufferToBase64(utf8.buffer);
			// 返回带有data URI前缀的Base64字符串
			return 'data:image/svg+xml;base64,' + base64;
		},

引用

html 复制代码
<icon :title="item.icon" size="20" :color="currentTab === index ? '#000000' : '#999999'"></icon>

参考: 动态设置svg颜色

相关推荐
光影少年20 分钟前
vue2与vue3的全局通信插件,如何实现自定义的插件
前端·javascript·vue.js
Rattenking24 分钟前
React 源码学习01 ---- React.Children.map 的实现与应用
javascript·学习·react.js
丁总学Java1 小时前
微信小程序,点击bindtap事件后,没有跳转到详情页,有可能是app.json中没有正确配置页面路径
微信小程序·小程序·json
熊的猫1 小时前
JS 中的类型 & 类型判断 & 类型转换
前端·javascript·vue.js·chrome·react.js·前端框架·node.js
瑶琴AI前端2 小时前
uniapp组件实现省市区三级联动选择
java·前端·uni-app
说私域2 小时前
基于开源 AI 智能名片、S2B2C 商城小程序的用户获取成本优化分析
人工智能·小程序
mosen8682 小时前
Uniapp去除顶部导航栏-小程序、H5、APP适用
vue.js·微信小程序·小程序·uni-app·uniapp
qq22951165023 小时前
微信小程序的汽车维修预约管理系统
微信小程序·小程序·汽车
别拿曾经看以后~3 小时前
【el-form】记一例好用的el-input输入框回车调接口和el-button按钮防重点击
javascript·vue.js·elementui
川石课堂软件测试3 小时前
性能测试|docker容器下搭建JMeter+Grafana+Influxdb监控可视化平台
运维·javascript·深度学习·jmeter·docker·容器·grafana