">
,需要通过父级DOM结构来判断
*/
var trackActionPhone = function (node) {
var nodeInnerText = node.innerText || ''
if (!limitRegLength(nodeInnerText)) return
var nodeText = trimText(nodeInnerText)
if (nodeText.length < 5 || nodeText.length > 20) return false
var type = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'click'
var str = trimText(node.href || node.innerHTML || '')
if (phoneReg.test(str) && numUseReg.test(str)) {
_paq.push(['trackEvent', type, 'phone', nodeText])
return true
}
/** 排查父级嵌套非标签场景,并且对dom的正则校验做一个性能兜底,通过控制innerText的长度,来确保正则的性能 */
var fatherText = trimText(node.parentNode.innerText || '')
if (fatherText.length < 5 || fatherText.length > 20) return false
var fatherDom = trimText(node.parentNode.innerHTML || '')
if (phoneReg.test(fatherDom) && numUseReg.test(fatherDom)) {
_paq.push(['trackEvent', type, 'phone', nodeText])
return true
}
return false
}
window.addEventListener('click', function (e) {
var node = e.target
/** 社媒点击 */
var appName = ''
var getAppAriaLabel = node.ariaLabel || node.parentNode.ariaLabel || ''
if (mediaList.includes(getAppAriaLabel.toLowerCase())) {
appName = getAppAriaLabel
}
if (!appName && node.nodeName && node.nodeName.toLowerCase() === 'a') {
appName = getMediaName(node.href) || getMediaName(node.alt)
}
if (!appName && node.nodeName && node.nodeName.toLowerCase() === 'img') {
appName = getMediaName(node.alt) || getMediaName(node.src)
}
if (!appName && node.nodeName && node.nodeName.toLowerCase() === 'i') {
appName = getMediaName(node.className)
}
if (appName) {
_paq.push(['trackEvent', 'click', 'contactApp', appName])
return
}
/** 联系方式点击 */
if (trackActionPhone(node, 'click')) return
if (node.nodeName && node.nodeName.toLowerCase() === 'a') {
var val = node.href
if (!limitRegLength(val)) return
if (emailReg.test(val)) {
_paq.push(['trackEvent', 'click', 'email', val])
return
}
}
if (node.nodeName && node.nodeName.toLowerCase() === 'i') {
var val = node.className
var content = node.parentNode.href || ''
if (val.includes('email')) {
_paq.push(['trackEvent', 'click', 'email', content])
return
}
}
var nodeChildList = node.childNodes
for (var i = 0; i < nodeChildList.length; i++) {
;(function (i) {
if (nodeChildList[i].nodeType !== 3) return
var val = nodeChildList[i].textContent.replace(/\s?:?/g, '')
if (!limitRegLength(val)) return
if (emailReg.test(val)) {
_paq.push(['trackEvent', 'click', 'email', val])
return
}
})(i)
}
trackNumberData(node)
})
window.addEventListener('copy', function (e) {
if (trackActionPhone(e.target, 'copy')) return
var text = e.target.textContent
if (!text) return
var val = text.replace(/\s:?/g, '')
if (!limitRegLength(val)) return
if (emailReg.test(val)) {
_paq.push(['trackEvent', 'copy', 'email', val])
return
}
trackNumberData(e.target)
})
}
trackContactInit()
/**
* 基于custom_inquiry_form.js 以及 form.js 对于询盘表单提交的实现,来反推询盘表单的input标签触发,用来收集意向客户
* 1. 缓存的KEY:TRACK_INPUT_ID_MTM_00;
* 2. 缓存策略 - lockTrackInput:单个页面内,10分钟内,不重复上报
*/
function trackActionInput() {
const CACHE_KEY = 'TRACK_INPUT_ID_MTM_00'
const pathName = window.location.hostname + window.location.pathname
var lockTrackInput = function () {
try {
const lastCacheData = localStorage.getItem(CACHE_KEY)
if (!lastCacheData) return false
const cacheData = JSON.parse(lastCacheData)
const cacheTime = cacheData[pathName]
if (!cacheTime) return false
return Date.now() - cacheTime < 1000 * 60 * 10 // 10分钟内,不重复上报
} catch (error) {
console.error('lockTrackInput Error', error)
return false
}
}
var setInputTrackId = function () {
try {
const curCacheData = localStorage.getItem(CACHE_KEY)
if (curCacheData) {
const cacheData = JSON.parse(curCacheData)
cacheData[pathName] = Date.now()
localStorage.setItem(CACHE_KEY, JSON.stringify(cacheData))
return
}
const cacheData = {
[pathName]: Date.now(),
}
localStorage.setItem(CACHE_KEY, JSON.stringify(cacheData))
} catch (error) {
console.error('setInputTrackId Error', error)
}
}
var getInputDom = function (initDom) {
var ele = initDom
while (ele) {
/**
* isWebSiteForm 是站点的表单
* isChatWindowForm 是聊天窗口的表单
*/
/** 旧模板表单 */
var isWebSiteForm = !!(/crm-form/i.test(ele.className) && ele.querySelector('form'))
/** 1:新模板自定义表单、2:Get a Quote 弹框表单 */
var isWebSiteFormNew = !!(/inquiry/i.test(ele.className) && ele.querySelector('form'))
if (isWebSiteForm || isWebSiteFormNew) {
_paq.push(['trackEvent', 'formInquiry', 'formInput', 'page'])
setInputTrackId()
break
}
/** Mkt会话触达-聊天弹框的表单输入: MKT由于是iframe嵌入,所以MKT的上报,会单独写到MKT-form代码上 */
var isInquiryChatForm = !!(/comp-form/i.test(ele.className) && ele.querySelector('form'))
if (isInquiryChatForm) {
_paq.push(['trackEvent', 'formInquiry', 'formInput', 'chat'])
setInputTrackId()
break
}
/** 向上查找父节点 */
ele = ele.parentNode
}
}
function initInputListener() {
var inputUseDebounce = function (fn, delay) {
var timer = null
var that = this
return function () {
var args = Array.prototype.slice.call(arguments)
if (timer) clearTimeout(timer)
timer = setTimeout(function () {
fn.apply(that, args)
}, delay)
}
}
var optimizeGetInputDom = inputUseDebounce(getInputDom, 300)
window.addEventListener('input', function (e) {
/** 如果已经上报过,则不再上报 */
if (lockTrackInput()) return
optimizeGetInputDom(e.target)
})
}
try {
initInputListener()
} catch (error) {
console.log('initInputListener Error', error)
}
}
trackActionInput()
}
/** 第三方消息上报:目前主要是针对全点托管会话;在msgCollect/index.js中调试,访问test.html */
function thirdMsgCollect() {
/** 先检测是否是stayReal托管:如果stayReal脚本都没有,那么说明当前站点未开启stayReal会话托管 */
const scriptList = Array.prototype.slice.call(document.querySelectorAll('script'))
const checkStayReal = () => !!scriptList.find((s) => s.src.includes('stayreal.xiaoman.cn'))
if (!checkStayReal()) return
/** 缓存当前消息队列的最后一条消息id */
const CACHE_KEY = 'CACHE_KEY_MONITOR'
const setCache = (msgIndex) => {
/** 对缓存KEY进行base64转码处理 */
const cacheMsgIndex = btoa(msgIndex)
localStorage.setItem(CACHE_KEY, cacheMsgIndex)
}
const getCache = () => {
const cacheMsgIndex = localStorage.getItem(CACHE_KEY)
if (cacheMsgIndex) return Number(atob(cacheMsgIndex))
return -1
}
/** 拉取最新msg列表 */
const pullMsgList = () => {
const msgEleList = Array.prototype.slice.call(document.querySelectorAll('#chat-list li'))
const msgIds = []
const msgMap = msgEleList.reduce((acc, item) => {
const sendTime = item.querySelector('.message-data-time').textContent.trim()
const sendContent = item.querySelector('.message').textContent.trim()
/** msg带有class:other-message的是访客消息,my-message的是客服消息 */
const isOtherMessage = item.querySelector('.message').classList.contains('other-message')
const msgId = item.querySelector('.message').getAttribute('id')
const msgItemData = {
msgId,
user: isOtherMessage ? 'visitor' : 'official',
time: sendTime,
content: sendContent,
}
msgIds.push(msgId)
acc[msgId] = msgItemData
return acc
}, {})
return {
ids: msgIds,
dataMap: msgMap,
}
}
/** 加密并上传消息数据 */
let ENCRYPT_KEY = 'de29f1aab63ab033'
let ENCRYPT_IV = 'b8d2badf875e76ac'
const baseUrl = 'https://cms.xiaoman.cn'
// var getEncryptConfig = function () {
// const url = baseUrl + '/shop-api/innerApi/getKeyIv'
// $.get(
// url,
// function (result) {
// console.log('result', result)
// if (Number(result.code) === 0 && result.data.key && result.data.iv) {
// ENCRYPT_KEY = result.data.key
// ENCRYPT_IV = result.data.iv
// uploadMsgData()
// } else {
// /** 如果获取失败,则重试 */
// setTimeout(() => {
// getEncryptConfig()
// }, 1000)
// }
// },
// 'json'
// )
// }
// getEncryptConfig()
const encryptMsg = function (msgData) {
const enc = new TextEncoder()
// 转字节
const keyBytes = enc.encode(ENCRYPT_KEY)
const ivBytes = enc.encode(ENCRYPT_IV)
const plainBytes = enc.encode(msgData)
// 导入密钥并加密
return crypto.subtle
.importKey('raw', keyBytes, { name: 'AES-CBC' }, false, ['encrypt'])
.then(function (cryptoKey) {
return crypto.subtle.encrypt({ name: 'AES-CBC', iv: ivBytes }, cryptoKey, plainBytes)
})
.then(function (encryptedBuffer) {
// 转 base64 返回
return btoa(String.fromCharCode(...new Uint8Array(encryptedBuffer)))
})
.catch((err) => {
return Promise.reject(err)
})
}
let uploadFlag = false
const uploadMsgData = function () {
if (uploadFlag) return
uploadFlag = true
const { ids, dataMap } = pullMsgList()
let cacheMsgIndex = getCache()
const msgLen = ids.length
if (!msgLen) {
// 消息DOM未挂载 || 消息DOM已挂载,但是消息列表为空
uploadFlag = false
return
}
if (msgLen - 1 < cacheMsgIndex) {
/** 针对站点挂后台一段时间,消息列表会自动塞入重复消息,导致消息有重复,刷新后又重置回正常消息列表,所以这里需要更新锚点下标 */
cacheMsgIndex = msgLen - 1
setCache(cacheMsgIndex)
uploadFlag = false
return
}
if (msgLen - 1 === cacheMsgIndex) {
// 缓存的最后一次发送的消息ID是最后一条(说明当前消息均已经上报),则不跳过本地上报
uploadFlag = false
return
}
const currentMsgIds = ids.slice(cacheMsgIndex + 1, msgLen)
const currentMsgData = currentMsgIds.map((id) => dataMap[id])
const mtmId = window.matomo_site_id_cookie_key || '' // 获取mtm会话id
const msgBody = {
mtmId,
curl: window.location.href,
msgList: currentMsgData,
}
const msgBodyStr = JSON.stringify(msgBody)
encryptMsg(msgBodyStr)
.then(function (encryptedMsg) {
console.log('encryptedMsg:', encryptedMsg, msgBodyStr)
const url = baseUrl + '/shop-api/External/ListenSiteActiveStatus'
$.ajax({
type: 'POST',
url,
data: JSON.stringify({ d_v: encryptedMsg }),
contentType: 'application/json',
success: function (result) {
if (Number(result.code) === 0) {
// 更新消息队列
setCache(msgLen - 1)
}
uploadFlag = false
},
error: function (err) {
console.error(err, '请求异常')
uploadFlag = false
},
})
})
.catch((err) => {
console.error(err, '数据加密失败')
uploadFlag = false
})
}
/** 监控chat-list的DOM变更 */
const initChatListObserver = () => {
// 需要监听的 DOM 节点
const target = document.getElementById('chat-list')
// 回调函数
const callback = function (mutationsList, observer) {
for (const mutation of mutationsList) {
console.log('mutation', mutation)
if (mutation.type === 'childList') {
uploadMsgData()
}
}
}
// 配置
const config = {
childList: true, // 监听子节点的增删
subtree: true, // 是否也监听后代节点
}
// 创建 observer
const observer = new MutationObserver(callback)
// 开始监听
observer.observe(target, config)
}
let testCount = 30
let itv = null
const checkChatDom = () => !!document.querySelector('#vc-model')
const initTalkCheck = () => {
itv = setTimeout(() => {
console.log('checkChatDom', checkChatDom(), testCount)
if (!checkChatDom() && testCount > 0) {
testCount--
initTalkCheck()
return
}
clearTimeout(itv)
uploadMsgData()
initChatListObserver()
}, 1500)
}
initTalkCheck()
}
try {
gtmTrack()
thirdMsgCollect()
console.log('inserted gtm code')
} catch (error) {
console.error('gtmTrack Error', error)
}
})
})();
Title: "Pale Papale Dapper no Yiwu Qianxun Wahine Kaulana" I ka ʻimi ʻana i kahi hiʻohiʻona ʻaʻole hoʻi e hoʻonui i nā lole ʻoluʻolu? E nānā i nā pāpale papa no nā wāhine! ʻO kēia mau mea Yiwu Qianxun ua lilo nā pāpale i mea kaulana i waena o nā wāhine akamai i ke ʻano no kā lākou ʻano a me nā ʻano i hiki ke nani. No ke aha mākou e ʻimi ʻole ai i ka papale papa no na kane nā mea maikaʻi o nā pāpale makua kāne a pehea e hoʻohana ʻia ai e kekahi e hoʻonui i kou ʻano Pono pono nā pāpale papa no nā wāhine makemake e mālama i kā lākou mau laka i ka nānā ʻana i ka nani. ʻO kēia mau mea Yiwu Qianxun hiki i nā pāpale ke kūpono i kēlā me kēia nui o ka noʻonoʻo me ka puʻupuʻu mālie a hiki ke hoʻololi ʻia. Eia kekahi, hele mai nā pāpale papa i nā kala, nā maʻamau, a me nā pāpale papa kahiko nā mea e nānā ai i ke ʻano a i ʻole ka lole. Mai ka denim a hiki i ka pulupulu, ka ʻili i hoʻokumu ʻia i ka suede, ʻo nā pāpale papa he mea kūpono a me ke ʻano. Ua hōʻike ʻia nā mea hoʻomohala me nā manaʻo he mea hou e hoʻolilo i nā pāpale papa i nā wahine he nui e hoihoi. ʻO kekahi mau pāpale makua kāne i kāhiko maoli ʻia a paʻi ʻia paha i nā ʻano hoʻolālā e hoʻohālikelike pinepine ʻia i ke alo o ke alo. Yiwu Qianxun hui. ʻO nā pāpale papa ʻē aʻe e hana ana i nā sequins ʻālohilohi, nā ʻāpana ʻālohilohi, a i ʻole nā kikoʻī ʻālohilohi he metallic e hoʻohui i kahi paʻi nani i kou ʻano pilikino. ʻO ka papale papa no na wahine pau ole na koho ʻO ka hoʻohana ʻana i ka pāpale makua ʻaʻole e hoʻokiʻekiʻe wale i kāu mea akā e mālama pono iā ʻoe mai ka lā. ʻO nā pāpale Pāpā ma ke ʻano he lula, ʻo ia nō ka uhi ākea i ka maka a hoʻolele i kou mau maka mai nā kukuna UV ʻino. Yiwu Qianxun i hanaia e keia papale papa maikai loa he hiʻohiʻona kūpono no nā hana waho e like me ka hele wāwae, ka hoʻomoana, a i ʻole ka pīkī. Ma waho aʻe, ʻoluʻolu maikaʻi nā pāpale makua kāne ʻaʻole ʻoe e manaʻo e kau ana ʻoe i kahi pāpale. e haʻaheo i ka hana ʻana i nā pāpale kiʻekiʻe he ʻano hana lōʻihi. ʻO nā pāpale Baseball i kūkulu ʻia i nā mea waiwai kiʻekiʻe e hāʻawi i ka hōʻoluʻolu kiʻekiʻe. Hiki ke hoʻololi ʻia ke ʻano o ke ʻano o nā papale i ulana ʻia i nā pono o nā mea kūʻai aku. ʻO nā pāpale mauʻu i ulana maikaʻi ʻia. hiki ke koho i nā pāpale papa no nā pāpale wahine. papale papa no na wahine e kauoha mau i na mea kuai e ike i ko lakou makemake. ʻo ia hoʻi ka hana ʻana i ka noiʻi mākeke, ke hele ʻana i nā hōʻike kālepa, ke noi ʻana i nā manaʻo. Hāʻawi ka hale hana i nā koho pāpale maʻamau e hiki i nā mea kūʻai ke hana i nā pāpale e pili ana i nā koi kikoʻī, e like me LOGO embroidery color customization. ʻO nā pāpale papa no nā wāhine he hui akamai kūpono e hana pū ana e hoʻokō i nā pahuhopu hana. ʻO kēlā me kēia lālā o ka hui ka ʻike mākau kūikawā, a mai ka hoʻolālā mua ʻana i ka mana o ka hana ʻana i ka maikaʻi, hana pū ka hui e hana i nā pāpale e kū i nā kūlana kiʻekiʻe. ʻO nā pāpale papa no nā wahine logistics kaʻina e hōʻoia i nā huahana i hāʻawi ʻia i nā mea kūʻai aku ma ka honua holoʻokoʻa. hana nā ʻoihana kaulana e hoʻouna i nā pāpale ma ke kai, ka ʻāina lewa. Maikaʻi ka hui Logistics i ka lawelawe ʻana i nā moku honua, e hōʻoia ana i ka hōʻea ʻana o nā pāpale i ka manawa kūpono i ke kūlana kūpono. Ua kūpono nā pāpale Pāpā i kēlā me kēia lā kūikawā, inā paha ʻoe e holo ana i nā ʻoihana, hoʻokō i nā hoaaloha, e hoʻolālā ana i kahi ʻahamele, a i ʻole ka huakaʻi. Hiki iā ʻoe ke hoʻonohonoho iā lākou a pau me nā jeans, nā pōkole, nā ʻaʻahu, nā ʻaʻahu, a i ʻole nā lole ʻauʻau e hana i kahi hiʻohiʻona. Yiwu Qianxun ʻoiaʻiʻo ke ʻano. Hoʻohui pū ʻia nā pāpale makua kāne i nā lauoho like ʻole, e like me nā huelo pony, braids, buns, a i ʻole nā nalu papale papale ua hemo. Hiki ke ʻaʻahu ʻia lākou no ka mea hiki iā ʻoe ke mālama i kou poʻo i ka hoʻoilo a i ʻole ke anuanu i ke kauwela e ʻoe i kēlā me kēia manawa. Maikaʻi a maʻalahi ka hoʻohana ʻana i nā pāpale papa. ʻO ka mea mua, e koho i kahi pāpale makua i kūpono i kou ʻano a i ʻole ka hui. A laila, hoʻoponopono i ka Yiwu Qianxun kāʻei e kūpono i kou manaʻo. A laila e hoʻoponopono i ke kaula inā makemake ʻoe e ʻaʻahu i ka pāpale i hope, e hoʻohuli mua i ke kaʻe a. Inā ʻoe e paʻa i nā maka aniani, e ʻike pono i ka pāpale pōhili papa ʻaʻole keʻakeʻa iā lākou. ʻO ka hope loa, e hauʻoli i ka hoʻomāhuahua ʻana i kou makuakāne he pāpale hou kēia ʻOiai e kūʻai ana i nā pāpale papa, e koho i kahi inoa inoa kaulana e hāʻawi i ka hopena i kēia Yiwu Qianxun he mea maikai loa. E ʻimi i nā hōʻailona e hāʻawi ana i ka laulā o nā hoʻolālā|ʻano ākea o}, nā mea a me nā kala, me ka lawe wikiwiki ʻana, nā hoʻihoʻi maʻalahi, a me ka lawelawe mea kūʻai aku i kēia papale kaʻa kaʻa he aloha aloha. Hiki iā ʻoe ke heluhelu maʻalahi i nā loiloi o ka mea kūʻai aku a i ʻole e noi i nā alakaʻi mai nā hoaaloha e hōʻoia i ka loaʻa ʻana o kahi mea maikaʻi. Papale papa no na wahine
Hoʻolauna:

Naʻi:

Hoʻoulu ʻana:

Kākoʻo ʻelua:
Why choose Yiwu Qianxun Papale papa no na wahine?
Nā Kūkaepoka Paʻa Me Nā Hopena ʻElua
Ke Anakohu ʻAna I Mua Hoʻoponopono
Kumuwaiwai A Me Ka Hoʻomohala Noʻonoʻo
Ka Pāhana Hoʻokuʻu Paʻa A Me Ka Hoʻohana Heluhelu
Nā Kepanī Helu Ana Mea Paʻa Kapu
Hoʻohana:
Pepehi Ia Me He Aha:
Hōʻea:
ʻAʻole i ʻōlelo ʻou i kekahi mea e nānā ana ʻoe?
Hoʻomehana Hoʻokō ʻElua
Hoʻokuʻu i kēlā mea hoʻoponopono no nā mea paʻa e hiki mai ana.