feat: 小程序超级个体/个人资料/CKB获客;VIP列表展示过滤;管理端与API联调
- 超级个体:去掉首位特例;列表仅展示有头像且非微信默认昵称(vip.go) - 个人资料:居中头像、低调联系方式、点头像优先走存客宝 lead(ckbLeadToken) - 阅读页分享朋友圈复制与 toast 去重 - soul-api: miniprogram users 带 ckbLeadToken;其它 handler 与路由调整 - 脚本:content_upload、miniprogram 上传辅助等 Made-with: Cursor
This commit is contained in:
@@ -27,6 +27,29 @@ function decodeEntities(str) {
|
||||
.replace(/'/g, "'")
|
||||
}
|
||||
|
||||
/**
|
||||
* 单行展示用:昵称、#标签文案、章节外标题类字段 — 合并换行、<br>、连续空白(避免 TipTap/粘贴带入异常断行)
|
||||
*/
|
||||
function cleanSingleLineField(s) {
|
||||
if (!s && s !== 0) return ''
|
||||
let t = decodeEntities(String(s))
|
||||
.replace(/<br\s*\/?>/gi, ' ')
|
||||
.replace(/\r\n|\r|\n/g, ' ')
|
||||
.replace(/[\s\u00a0\u200b\u200c\u200d\ufeff\u3000]+/g, ' ')
|
||||
.trim()
|
||||
return t
|
||||
}
|
||||
|
||||
/** @提及昵称:去首尾空白、零宽、全角空格;合并内部换行/<br> */
|
||||
function cleanMentionNickname(n) {
|
||||
return cleanSingleLineField(n)
|
||||
}
|
||||
|
||||
/** 纯文本在 mention 节点前若已有「@」,去掉末尾 @,避免渲染成「找@@阿浪」 */
|
||||
function stripTrailingAtForMention(before) {
|
||||
return before.replace(/[@@][\s\u00a0\u200b]*$/u, '')
|
||||
}
|
||||
|
||||
/**
|
||||
* 将一个 HTML block 字符串解析为 segments 数组
|
||||
* 处理三种内联元素:mention / linkTag(span) / linkTag(a) / img
|
||||
@@ -39,20 +62,25 @@ function parseBlockToSegments(block) {
|
||||
let m
|
||||
|
||||
while ((m = tokenRe.exec(block)) !== null) {
|
||||
// 前置纯文本
|
||||
const before = decodeEntities(block.slice(lastEnd, m.index).replace(/<[^>]+>/g, ''))
|
||||
// 前置纯文本(mention 紧挨手写「找@」时去掉重复 @)
|
||||
let before = decodeEntities(block.slice(lastEnd, m.index).replace(/<[^>]+>/g, ''))
|
||||
const tag = m[0]
|
||||
if (/data-type="mention"/i.test(tag)) {
|
||||
before = stripTrailingAtForMention(before)
|
||||
}
|
||||
if (before.trim()) segs.push({ type: 'text', text: before })
|
||||
|
||||
const tag = m[0]
|
||||
|
||||
if (/data-type="mention"/i.test(tag)) {
|
||||
// @mention — TipTap mention span
|
||||
// @mention — TipTap mention span(span 内常见「@ 昵称」多空格,统一紧挨显示)
|
||||
const idMatch = tag.match(/data-id="([^"]*)"/)
|
||||
const labelMatch = tag.match(/data-label="([^"]*)"/)
|
||||
const innerText = tag.replace(/<[^>]+>/g, '')
|
||||
const userId = idMatch ? idMatch[1].trim() : ''
|
||||
const nickname = labelMatch ? labelMatch[1].trim() : innerText.replace(/^@/, '').trim()
|
||||
if (userId || nickname) segs.push({ type: 'mention', userId, nickname })
|
||||
let nickname = labelMatch ? labelMatch[1] : innerText.replace(/^[@@]\s*/, '')
|
||||
nickname = cleanMentionNickname((nickname || '').trim())
|
||||
if (userId || nickname) {
|
||||
segs.push({ type: 'mention', userId, nickname, mentionDisplay: '@' + nickname })
|
||||
}
|
||||
|
||||
} else if (/data-type="linkTag"/i.test(tag)) {
|
||||
// #linkTag — 自定义 span 格式(data-type="linkTag" data-url="..." data-tag-type="..." data-page-path="..." data-app-id="...")
|
||||
@@ -62,7 +90,7 @@ function parseBlockToSegments(block) {
|
||||
const tagIdMatch = tag.match(/data-tag-id="([^"]*)"/)
|
||||
const appIdMatch = tag.match(/data-app-id="([^"]*)"/)
|
||||
const mpKeyMatch = tag.match(/data-mp-key="([^"]*)"/)
|
||||
const innerText = tag.replace(/<[^>]+>/g, '').replace(/^#/, '').trim()
|
||||
const innerText = cleanSingleLineField(tag.replace(/<[^>]+>/g, '').replace(/^#/, ''))
|
||||
const url = urlMatch ? urlMatch[1] : ''
|
||||
const tagType = tagTypeMatch ? tagTypeMatch[1] : 'url'
|
||||
const pagePath = pagePathMatch ? pagePathMatch[1] : ''
|
||||
@@ -75,7 +103,7 @@ function parseBlockToSegments(block) {
|
||||
// #linkTag — 旧格式 <a href>(insertLinkTag 旧版产生,url 可能为空)
|
||||
// m[1] = href, m[2] = innerText(以 # 开头)
|
||||
const url = m[1] || ''
|
||||
const label = (m[2] || '').replace(/^#/, '').trim()
|
||||
const label = cleanSingleLineField((m[2] || '').replace(/^#/, ''))
|
||||
// 旧格式没有 tagType,在 onLinkTagTap 中会按 label 匹配缓存的 linkTags 配置降级处理
|
||||
segs.push({ type: 'linkTag', label: label || '#', url, tagType: '', pagePath: '', tagId: '' })
|
||||
|
||||
@@ -181,18 +209,31 @@ function stripMarkdownFormatting(text) {
|
||||
|
||||
/**
|
||||
* 对一行纯文本进行 @人名 / #标签 自动匹配,返回 segments 数组
|
||||
* config: { persons: [{personId, name, aliases}], linkTags: [{tagId, label, type, pagePath, mpKey, url, aliases}] }
|
||||
* config: { persons: [{ personId, token, name, label, aliases }], linkTags: [...] }
|
||||
* 点击加好友时须传 persons.token(与 CKB lead 的 targetUserId 一致),不能用 personId。
|
||||
*/
|
||||
function matchLineToSegments(line, config) {
|
||||
if (!config || (!config.persons?.length && !config.linkTags?.length)) {
|
||||
return [{ type: 'text', text: line }]
|
||||
}
|
||||
// 编辑器/系统在 @ 与人名之间插入的普通空格,合并为紧挨 @(避免「找@ 阿浪」无法匹配人名)
|
||||
line = line.replace(/([@@])\s+(?=[\u4e00-\u9fffA-Za-z0-9_\u00b7])/g, '$1')
|
||||
const normalize = s => (s || '').trim().toLowerCase()
|
||||
const personMap = {}
|
||||
const tagMap = {}
|
||||
for (const p of (config.persons || [])) {
|
||||
const keys = [p.name, ...(p.aliases ? p.aliases.split(',') : [])].map(normalize).filter(Boolean)
|
||||
for (const k of keys) { if (!personMap[k]) personMap[k] = p }
|
||||
const token = (p.token || '').trim()
|
||||
if (!token) continue
|
||||
const display = (p.name || p.label || '').trim()
|
||||
const aliasStr = p.aliases != null ? String(p.aliases) : ''
|
||||
const keys = [display, p.label, ...(aliasStr ? aliasStr.split(',') : [])]
|
||||
.map((x) => (x != null ? String(x) : '').trim())
|
||||
.filter(Boolean)
|
||||
.map(normalize)
|
||||
.filter(Boolean)
|
||||
for (const k of keys) {
|
||||
if (!personMap[k]) personMap[k] = p
|
||||
}
|
||||
}
|
||||
for (const t of (config.linkTags || [])) {
|
||||
const keys = [t.label, ...(t.aliases ? t.aliases.split(',') : [])].map(normalize).filter(Boolean)
|
||||
@@ -204,8 +245,8 @@ function matchLineToSegments(line, config) {
|
||||
if (!personNames.length && !tagLabels.length) return [{ type: 'text', text: line }]
|
||||
|
||||
const parts = []
|
||||
if (personNames.length) parts.push('[@@](' + personNames.join('|') + ')')
|
||||
if (tagLabels.length) parts.push('[##](' + tagLabels.join('|') + ')')
|
||||
if (personNames.length) parts.push('[@@]\\s*(' + personNames.join('|') + ')')
|
||||
if (tagLabels.length) parts.push('[##]\\s*(' + tagLabels.join('|') + ')')
|
||||
const pattern = new RegExp(parts.join('|'), 'gi')
|
||||
|
||||
const segs = []
|
||||
@@ -216,16 +257,22 @@ function matchLineToSegments(line, config) {
|
||||
segs.push({ type: 'text', text: line.slice(lastEnd, m.index) })
|
||||
}
|
||||
const full = m[0]
|
||||
const prefix = full[0]
|
||||
const body = full.slice(1)
|
||||
if (prefix === '@' || prefix === '@') {
|
||||
if (/^[@@]/u.test(full)) {
|
||||
const body = full.replace(/^[@@]\s*/u, '')
|
||||
const person = personMap[normalize(body)]
|
||||
if (person) {
|
||||
segs.push({ type: 'mention', userId: person.personId || '', nickname: person.name || body })
|
||||
const nick = cleanSingleLineField(person.name || person.label || body)
|
||||
const uid = (person.token || '').trim()
|
||||
if (uid) {
|
||||
segs.push({ type: 'mention', userId: uid, nickname: nick, mentionDisplay: '@' + nick })
|
||||
} else {
|
||||
segs.push({ type: 'text', text: full })
|
||||
}
|
||||
} else {
|
||||
segs.push({ type: 'text', text: full })
|
||||
}
|
||||
} else {
|
||||
const body = full.replace(/^[##]\s*/u, '')
|
||||
const tag = tagMap[normalize(body)]
|
||||
if (tag) {
|
||||
segs.push({
|
||||
@@ -285,5 +332,6 @@ function parseContent(rawContent, config) {
|
||||
|
||||
module.exports = {
|
||||
parseContent,
|
||||
isHtmlContent
|
||||
isHtmlContent,
|
||||
cleanSingleLineField,
|
||||
}
|
||||
|
||||
@@ -26,7 +26,8 @@ function getAppInstance() {
|
||||
}
|
||||
|
||||
const RULE_COOLDOWN_KEY = 'rule_engine_cooldown'
|
||||
const COOLDOWN_MS = 60 * 1000
|
||||
// 0 = 关闭冷却(需求:去掉「操作频繁 / N 分钟」类体感限制)
|
||||
const COOLDOWN_MS = 0
|
||||
let _cachedRules = null
|
||||
let _cacheTs = 0
|
||||
const CACHE_TTL = 5 * 60 * 1000
|
||||
@@ -45,6 +46,7 @@ const TRIGGER_SCENE_MAP = {
|
||||
}
|
||||
|
||||
function isInCooldown(ruleId) {
|
||||
if (!COOLDOWN_MS || COOLDOWN_MS <= 0) return false
|
||||
try {
|
||||
const map = wx.getStorageSync(RULE_COOLDOWN_KEY) || {}
|
||||
const ts = map[ruleId]
|
||||
@@ -57,6 +59,7 @@ function isInCooldown(ruleId) {
|
||||
}
|
||||
|
||||
function setCooldown(ruleId) {
|
||||
if (!COOLDOWN_MS || COOLDOWN_MS <= 0) return
|
||||
try {
|
||||
const map = wx.getStorageSync(RULE_COOLDOWN_KEY) || {}
|
||||
map[ruleId] = Date.now()
|
||||
|
||||
Reference in New Issue
Block a user