1 实现效果
2 组件代码
1. 模板代码 template
<template>
<template v-if="!noColor">
<ColorPicker v-model="colorData.color" size="small" :colors="colorData.colors"/>
<Button size="small" ghost type="primary" @click.native="addColorTag">配色</Button>
</template>
<Button v-if="hasHyperLink" class="mr-5" size="small" ghost type="primary" @click.native="addHyperLink">超链接</Button>
<Button v-if="!noPreview" class="mr-5" size="small" ghost type="primary" @click.native="previewClick">预览</Button>
<slot></slot>
<Input v-if="!inputSelector" ref="editorRef" type="textarea" :value="content" placeholder="请输入内容" :autosize="{ minRows:3, maxRows: 8 }" @input="inputChange"></Input>
</template>
2. 处理应用程序逻辑 ColorDataCls类
2.1 邮件预设颜色常量 MAIL
const MAIL = {
defaultColor: '#9d1802',
createPresetColors: function(type) {
switch (type) {
case 'notice':
return ['#9d1802', '#00800f', '#964129'];
default:
return ['#9d1802', '#00800f', '#e57c00', '#3ef672', '#964129'];
}
}
}
2.2 构造者函数+基本功能
class ColorDataCls {
constructor(props) {
this.color = MAIL.defaultColor;
this.colors = MAIL.createPresetColors();
props && Object.assign(this, props);
}
static addColorTag(inputEle, inputContent, color) {}
static addHyperLink(inputEle, inputContent) {}
static previewContentTransform(inputEle) {}
}
3. 父组件可传递属性 props
export default {
let idNum = 0;
createId() {
return `CWID${idNum++}`;
},
props: {
content: String,
noColor: Boolean,
hasHyperLink: Boolean,
noPreview: Boolean,
inputSelector: String,
colorData: {
type: Object,
default() {
return new ColorDataCls();
}
}
}
}
4. 事件处理程序
4.1 配色
addColorTag() {
let element =
this.inputSelector ?
document.querySelector(this.inputSelector) : this.$refs.editorRef.$refs.textarea;
this.$emit(
'update:content',
ColorDataCls.addColorTag(element, this.content, this.colorData.color)
);
this.$emit('addColor', this.colorData.color);
}
4.2 预览
previewClick() {
this.$emit(
'preview',
ColorDataCls.previewContentTransform(this.content)
);
}
4.3 监听输入框事件
inputChange(value) {
this.$emit('update:content', value);
}
5. 处理应用程序逻辑具体实现
5.1 为选中文字添加颜色标签
static addColorTag(inputEle, inputContent, color) {
let rangeData = {
text: '',
start: 0,
end: 0
};
if(inputEle.setSelectionRange) {
rangeData.start = inputEle.selectionStart;
rangeData.end = inputEle.selectionEnd;
rangeData.text =
rangeData.start != rangeData.end ?
inputContent.slice(rangeData.start, rangeData.end) : '';
return `${inputContent.slice(0, rangeData.start)}<color=${color}>${rangeData.text}</color>${inputContent.slice(rangeData.end)}`;
}
}
5.2 标签转换
static previewContentTransform(inputEle) {
return inputContent
.replace(/\n/g, '<br/>')
.replace(/\s/g, ' ')
.replace(/(size=|color=)/g, 'font $1')
.replace(/(size>|color>)/g, 'font>')
.replace(/<a\ /g, "<a style='font-size:22px; color:#379a61; text-decoration:underline' target='_blank' ")
}
5.3 有限层标签嵌套
static changeColorRegs(content) {
const reg = /\[color=(.*?)\]((\[color=(.*?)\](.*?)\[\/color\]|.)*?)\[\/color\]/g
return content.replace(reg, (match, p1, p2) => {
let str = `<span style="color: ${p1}">`
if(reg.test(p2)) {
return str + this.changeColorRegs(p2) + '</span>'
} else {
return str + p2 + '</span>'
}
})
}
5.4 找到闭合标签的前一个同类型起始标签实现配对
static changeColorRegs(content, contentFlage) {
const regBegin = /\[color=(.*?)\]/
const regFindLast = /(.*)\[color=(.*?)\](.*)/g
const regContent = /(\[color=.*?\][\s\S]*?)(?=\[\/color\])\[\/color\]/g
return content.replace(regContent, (match, p1) => {
if(regBegin.test(p1)) {
return p1.replace(regFindLast, (inner_match, inner_p1, inner_p2, inner_p3) => {
if(regBegin.test(inner_p1)) {
contentFlage.colorFlage = true
}
return `${inner_p1}<span style="color: ${inner_p2}">${inner_p3}</span>`
})
}
return match
})
}
static previewContentTransform(inputContent) {
let content = inputContent
.replace(/</g, '<')
.replace(/>/g, '>')
.replace(/\n/g, '<br/>')
.replace(/\s/g, ' ')
let contentFlage = {
colorFlage: false,
sizeFlage: false,
bFlage: false,
}
do {
contentFlage.colorFlage = false
content = this.changeColorRegs(content, contentFlage)
} while(contentFlage.colorFlage)
return content
}
3 应用
3.1 配色+预览
<color-wrapper :content.sync="pepper.data.content" @preview="mailPreviewOpen">
<span class="text-danger">{money}替换档位金额数</span>
</color-wrapper>
<mail-preview :data="mailPreviewData"></mail-preview>
mailPreviewOpen(content) {
let molData = this.pepper.data;
this.mailPreviewData.title = molData.title;
this.mailPreviewData.content = content;
this.mailPreviewData.isDisplay = true;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· .NET10 - 预览版1新功能体验(一)