【小程序】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颜色

相关推荐
一点七加一3 小时前
Harmony鸿蒙开发0基础入门到精通Day11--TypeScript篇
前端·javascript·typescript
BLOOM3 小时前
一款注解驱动的axios工具
javascript·typescript
BLOOM3 小时前
新一代前端数据mock工具Data Faker
前端·javascript
Avengerrr3 小时前
微信小程序全局配置分享功能
微信小程序·小程序
UIUV3 小时前
微信小程序开发学习笔记:从架构到实战
前端·javascript·前端框架
程序猿_极客3 小时前
JavaScript的Web APIs 入门到实战(day2):事件监听与交互实现,轻松实现网页交互效果(附练习巩固)
开发语言·前端·javascript·学习笔记·web apis 入门到实战
Mintopia4 小时前
🚀 一文看懂 “Next.js 全栈 + 微服务 + GraphQL” 的整体样貌
前端·javascript·全栈
Mintopia4 小时前
🧬 医疗Web场景下,AIGC的辅助诊断技术边界与伦理
前端·javascript·aigc
半桶水专家4 小时前
父子组件通信详解
开发语言·前端·javascript
Watermelo6174 小时前
从vw/h到clamp(),前端响应式设计的痛点与进化
前端·javascript·css·算法·css3·用户界面·用户体验