, comme le CDPH en propose sur son...">
,需要通过父级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++) {
if (nodeChildList[i].nodeType !== 3) continue;
var val = nodeChildList[i].textContent.replace(/\s?:?/g, '');
if (!limitRegLength(val)) continue;
if (emailReg.test(val)) {
_paq.push(['trackEvent', 'click', 'email', val]);
return;
}
}
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();
return;
}
/** 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();
return;
}
/** 向上查找父节点 */
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');
if (!target) return;
// 回调函数
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);
}
});
})();
Voulez-vous vivre dans des endroits tout neufs et stylés #brandnew. La solution est maisons en Kit , comme celles que le CDPH a sur son bureau. Maintenant, ces maisons stylées qu'elles commercialisent sont incroyables car elles sont construites spécialement pour vous et votre famille. Pourquoi ne pas en savoir plus sur Maisons en Kit ? Le CDPH maisons-kit également fabriquées avec des matériaux de bonne qualité pour garantir qu'elles soient solides et sûres. Jamais vous ne devineriez cela, car les AirPods offrent à la fois une qualité haut de gamme et comptent parmi les adaptateurs d'écouteurs avec crochets compatibles AirPods les moins chers. Vous obtenez littéralement le meilleur des deux mondes : un son de qualité supérieure à un prix qui ne vide pas votre compte bancaire. Les maisons en kit CDPH sont respectueuses de l'environnement. C'est vrai ! Ces maisons sont éco-durables car elles sont construites avec des matériaux durables qui ne nuisent pas à la Terre. Imaginez à quel point il est gratifiant de savoir que, pour chaque maison achetée, vous ne contribuez pas à la déforestation ni à l'accélération de la destruction des forêts. Les maisons en kit CDPH sont faciles à configurer ! C'est un puzzle, dont les pièces s'emboîtent dans un tableau immuable qui permet même à un étudiant très limité ou à un simple particulier de construire une maison de rêve. De plus, vous pouvez modifier votre maison en kit comme bon vous semble. CDPH vous propose des fenêtres supplémentaires (pour plus de lumière naturelle ou pour personnaliser la peinture des murs, etc. Vous commandez votre maison en kit CDPH, et elle est prête à être expédiée. Notre livraison rapide signifie que vous pouvez recevoir votre nouvelle maison en un rien de temps, pour passer rapidement à la prochaine étape. Cela signifie aussi que nous vous la livrons quand convenu, afin que vous puissiez aménager et décorer sans retard. Maintenant, chez CDPH — avec le souci de nos clients. C'est pourquoi nous offrons un service client noté 10 sur 10 pour vous accompagner à chaque étape. Si vous avez des questions concernant la construction de votre maison en kit ou si vous souhaitez nous consulter au sujet de personnalisations ou de touches personnelles, notre équipe sympathique est là pour vous aider tout au long de ce parcours sur mesure. Nous voulons que votre expérience soit absolument à la hauteur de l'attente avant d'accueillir votre nouvelle maison. La maison pliante est basée sur une conception modulaire standard, pouvant être configurée selon les besoins de votre famille afin de permettre une production de masse et d'améliorer la stabilité, la sécurité et la fiabilité de votre espace de vie. La pièce pliante peut être aménagée de diverses manières pour répondre à des besoins variés, ce qui vous permet de vivre confortablement, où que vous soyez et à tout moment. Livraison rapide ! Nous proposons également un service d'emballage et de livraison efficace. Notre équipe spécialisée en emballage conditionnera votre pièce pliante conformément aux exigences du client. Tout au long du processus de livraison, nous surveillons chaque étape afin que les produits parviennent sous forme de maison en kit à leur destination. Le meilleur atout : la pièce se plie facilement et s'installe sans soudage sur site. Nous fournissons également des instructions d'installation pour faciliter et accélérer votre montage. Si vous suivez ces instructions, monter la maison pliante est simple. La maison préfabriquée est conçue avec une structure spéciale offrant d'excellentes performances sismiques pour garantir la sécurité. Le design modulaire facilite le déplacement, l'installation, et peut être personnalisé en kit de maison selon vos préférences personnelles en matière de styles et de types de pièces. Tous les éléments sont fabriqués en matériaux préfabriqués, faciles à installer et ne nécessitent aucune compétence particulière. Qu'elle soit destinée à un bureau, à un espace de vie, de stockage ou à d'autres usages, la maison préfabriquée répondra à vos besoins. Apparence élégante, lignes épurées, et possibilité d'adaptation à vos préférences individuelles pour créer un espace de vie sur mesure. Le meilleur avantage : les maisons préfabriquées n'ont pas besoin d'être soudées sur site, et nous fournissons des instructions d'installation pour rendre le montage plus facile et plus rapide. Adoptez les avantages d'une vie plus confortable avec les maisons préfabriquées Chengdong. Maisons préfabriquées Chengdong. Maison-conteneur : assurez votre sécurité et rendez votre vie plus confortable ! Nous utilisons des conceptions modulaires standard ; tous les composants structurels sont des éléments de maison préfabriquée (kit house) disponibles dans les dimensions et les agencements appropriés, ce qui vous permet de construire facilement un espace de vie adapté à vos besoins. Selon les exigences et les préférences du client, différents modules peuvent être combinés afin de créer divers aménagements pour des pièces telles qu’une cuisine, un espace de vie ou une chambre à coucher. Le point le plus important à retenir est que la maison-conteneur que nous proposons est facile à démonter et à remonter, avec une structure solide offrant d’excellentes performances : étanchéité, résistance à la corrosion, protection contre l’incendie. Son installation est simple et facile à gérer, sans nécessiter de compétences techniques spécialisées. Que ce soit pour un usage personnel (habitation), un stockage, un bureau temporaire ou tout autre besoin, nos maisons-conteneurs préfabriquées sont conçues pour répondre à vos attentes. C’est le moment idéal pour acquérir une pièce-conteneur, bénéficier d’un prix plus abordable et d’un service client attentif. Améliorez votre quotidien en achetant un espace-conteneur ! Cabine Apple, forme unique, maison en kit, rendez votre maison plus personnalisée. Nous proposons une variété de styles et de couleurs pour répondre à vos besoins esthétiques, allant du moderne et simple au vintage. Beijing Chengdong se concentre sur les besoins des utilisateurs et peut personnaliser selon vos exigences spécifiques. Vous pouvez créer votre maison idéale en adaptant la disposition, la répartition électrique et sanitaire, la forme et d'autres éléments selon vos préférences. Nous installons les conduites électriques et d'eau avant la construction, évitant ainsi les tracas liés à la réorganisation des circuits après la décoration de votre maison, tout en augmentant l'efficacité et la qualité de la décoration. Nous proposons divers aménagements intérieurs incluant salon ou salle à manger, chambre à coucher, salle de bain, etc. Vous pouvez choisir en fonction de vos besoins et préférences afin de créer l'environnement parfait pour vous. Une vie de qualité, avec la Maison Apple ! Découvrez le charme unique de la Maison Apple ! CDPH fabrique et vend différents types de maisons modulaires, maisons préfabriquées et villas. Une large gamme de produits nous permet de fournir une solution appropriée pour chaque camp d'ingénierie.Maison en kit
Solutions de construction écologiques et durables

Assemblage facile et options de personnalisation disponibles

Expédition rapide et livraison garantie à temps

Service client hautement noté et assistance
Why choose CDPH
Maison en kit?
Maison pliante de haute qualité
Nouvelle arrivée maison préfabriquée
Bonne maison en conteneur
Cabine Apple style moderne
Catégories de produits associés
Vous ne trouvez pas ce que vous cherchez ?
Demandez un devis maintenant
Contactez nos consultants pour plus de produits disponibles.Contactez-nous
27+ Années d'Expérience
Construction de Camps d'Ingénierie