index.js 82 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194
  1. // var Parse = getApp().Parse;
  2. // var app = getApp()
  3. // const { wxLogin } = require('./utils/login')
  4. const CONFIG = require("config.js");
  5. let config = {
  6. appid: CONFIG.default.appid,
  7. company: CONFIG.default.company,
  8. rootPage: CONFIG.default.rootPage,
  9. }
  10. const plugin = requirePlugin('fm-plugin')
  11. const { Parse, checkAuth } = plugin
  12. Page({
  13. /**
  14. * 页面的初始数据
  15. */
  16. data: {
  17. splashUrl: wx.getStorageSync("enabledOptions")[0],
  18. loading:true
  19. },
  20. /**
  21. * 生命周期函数--监听页面加载
  22. */
  23. onLoad: async function (options) {
  24. // 拦截插件的"登录信息已过期"提示
  25. const originalShowModal = wx.showModal;
  26. wx.showModal = function(options) {
  27. // 如果是"登录信息已过期"的提示,改为友好的重新登录提示
  28. if (options.content && options.content.includes('登录信息过期')) {
  29. console.log('⚠️ 拦截到"登录信息已过期"提示,改为友好提示');
  30. return originalShowModal.call(this, {
  31. title: '提示',
  32. content: '登录状态异常,是否重新登录?',
  33. showCancel: true,
  34. cancelText: '取消',
  35. confirmText: '重新登录',
  36. success: (result) => {
  37. if (result.confirm) {
  38. // 清除登录状态
  39. wx.removeStorageSync("sessionToken");
  40. wx.removeStorageSync("userLogin");
  41. // 重新加载首页
  42. const rootPage = getApp().globalData.rootPage || getApp().globalData.defaultTabBar?.list?.[0]?.pagePath || '/pages/index/index';
  43. wx.reLaunch({
  44. url: rootPage
  45. });
  46. }
  47. // 调用原始的 success 回调
  48. if (options.success) {
  49. options.success(result);
  50. }
  51. }
  52. });
  53. }
  54. // 其他提示正常显示
  55. return originalShowModal.call(this, options);
  56. };
  57. wx.login({
  58. success: function (res) {
  59. if (res.code) {
  60. console.log(res);
  61. // wx.request({
  62. // url: "https://server.fmode.cn/api/wxapp/auth_wxapp",
  63. // data: {
  64. // c: getApp().globalData.company,
  65. // code: res.code,
  66. // },
  67. // async success(res) {
  68. // wx.setStorageSync("userInfo", res.data);
  69. // resolve(res)
  70. // },
  71. // });
  72. }
  73. },
  74. fail: function (err) {
  75. wx.showToast({
  76. title: '服务器繁忙,请稍后重试',
  77. })
  78. }
  79. });
  80. wx.setStorageSync("invite", null);
  81. // 处理扫码链接(options.q 包含完整的URL或activityId)
  82. if (options.q) {
  83. let str = decodeURIComponent(options.q); // 扫描二维码获得的跳转链接
  84. // 兼容一些环境中把 & 编码成 & 的情况,防止参数解析错误(如 scanCount=0&storeId=...)
  85. if (str.indexOf('&') !== -1) {
  86. console.log('🔧 检测到 URL 中包含 &,自动还原为 &');
  87. str = str.replace(/&/g, '&');
  88. }
  89. // 检查是否是员工邀请链接(app.fmode.cn/dev/pobingfeng/manager/staff?invite=xxx)
  90. if (str.includes('app.fmode.cn/dev/pobingfeng/manager/staff')) {
  91. let obj = this.getParaName(str);
  92. if (obj && obj.invite) {
  93. wx.setStorageSync("invite", obj.invite);
  94. console.log('✅ 检测到员工邀请链接,invite:', obj.invite);
  95. }
  96. // 员工邀请链接不需要跳转到店铺,直接返回
  97. this.setData({ options: options });
  98. plugin.init(config, wx.getStorageSync('invite'));
  99. return;
  100. }
  101. // 检查是否是活动海报二维码(activityId作为qrCode参数值)
  102. // 如果str不包含http/https,且看起来像是一个ID,则可能是activityId
  103. if (!str.includes('http://') && !str.includes('https://') && !str.includes('?')) {
  104. // 可能是活动ID,尝试作为activityId处理
  105. if (str && str.length > 0 && !isNaN(str)) {
  106. options.activityId = str;
  107. console.log('✅ 检测到活动海报二维码,activityId:', str);
  108. // 活动二维码需要特殊处理,先保存activityId
  109. wx.setStorageSync("activityId", str);
  110. }
  111. }
  112. // 处理店铺相关的二维码链接(pwa.fmode.cn/gomini/pmd/)
  113. // 从完整URL中提取参数
  114. if (str.includes('pwa.fmode.cn/gomini/pmd') || str.includes('?')) {
  115. let obj = this.getParaName(str);
  116. if (obj && obj.invite) {
  117. wx.setStorageSync("invite", obj.invite);
  118. }
  119. // 将所有参数合并到 options 中
  120. obj && Object.keys(obj).forEach(key=> options[key] = obj[key]);
  121. }
  122. }
  123. // 兼容从编译参数或页面直达传入的 storeId
  124. if (options && options.scene && !options.storeId) {
  125. try {
  126. const sceneStr = decodeURIComponent(options.scene)
  127. if (sceneStr) {
  128. const pairs = sceneStr.split('&')
  129. for (const p of pairs) {
  130. const [k, v] = p.split('=')
  131. if (k === 'storeId' && v) {
  132. options.storeId = v
  133. break
  134. }
  135. }
  136. }
  137. } catch (e) {
  138. console.warn('解析 scene 失败:', e)
  139. }
  140. }
  141. this.setData({
  142. options: options
  143. })
  144. let {
  145. time,
  146. dramaId,
  147. roomId,
  148. orderId,
  149. shopId,
  150. invite,
  151. activityId,
  152. company,
  153. inviteHost,
  154. storeId
  155. } = options
  156. time && wx.setStorageSync("time", time);
  157. dramaId && wx.setStorageSync("dramaId", dramaId);
  158. roomId && wx.setStorageSync("roomId", roomId);
  159. orderId && wx.setStorageSync("orderId", orderId);
  160. shopId && wx.setStorageSync("shopId", shopId);
  161. invite && wx.setStorageSync("invite", invite);
  162. activityId && wx.setStorageSync("activityId", activityId);
  163. inviteHost && wx.setStorageSync("inviteHost", true);
  164. if (storeId) {
  165. // 扫码进入时,强制设置店铺 ID,标记为扫码来源
  166. wx.setStorageSync('storeId', storeId)
  167. getApp().globalData.storeId = storeId
  168. wx.setStorageSync('storeId_from_scan', true)
  169. console.log('✅ 入口页扫码进入,已设置店铺 ID(优先级最高):', storeId)
  170. }
  171. if (company) getApp().globalData.toCompany = true;
  172. // 检查是否是扫码进入(需要统计扫码次数)
  173. this.checkAndHandleScan(options);
  174. plugin.init(config, wx.getStorageSync('invite'))
  175. try {
  176. getApp().checkTrafficDeductPending = this.checkTrafficDeductPending.bind(this);
  177. console.log('🚦 [traffic] 已注册全局复判方法 checkTrafficDeductPending');
  178. } catch (e) {
  179. console.warn('🚦 [traffic] 注册全局复判方法失败:', e?.message || e);
  180. }
  181. },
  182. /**
  183. * 生命周期函数--监听页面初次渲染完成
  184. */
  185. onReady: async function () { },
  186. /**
  187. * 生命周期函数--监听页面显示
  188. */
  189. onShow: async function () {
  190. await this.review()
  191. try {
  192. if (typeof this.checkTrafficDeductPending === 'function') {
  193. console.log('🚦 [traffic] onShow 触发暂缓扣减复判');
  194. await this.checkTrafficDeductPending();
  195. }
  196. } catch (e) {
  197. console.warn('🚦 [traffic] onShow 复判触发失败:', e?.message || e);
  198. }
  199. },
  200. async review(force){
  201. try {
  202. let options = this.data.options
  203. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath
  204. if (options) {
  205. let objArr = Object.keys(options)
  206. if (objArr && objArr.length > 0) {
  207. let parms = '?'
  208. objArr.forEach((o, index) => {
  209. if (index > 0) {
  210. parms += '&' + o + '=' + options[o]
  211. } else {
  212. parms += o + '=' + options[o]
  213. }
  214. })
  215. url += parms
  216. }
  217. }
  218. let currentUser = Parse.User.current()
  219. console.log('===========================================');
  220. console.log('======= index.js review 方法 =======');
  221. console.log('当前用户:', currentUser ? currentUser.id : '无');
  222. console.log('用户手机号:', currentUser?.get('mobile') || '无');
  223. console.log('用户名:', currentUser?.get('username') || '无');
  224. console.log('Session Token:', currentUser?.getSessionToken()?.substring(0, 20) || '无');
  225. console.log('userLogin 存储:', wx.getStorageSync('userLogin') || '无');
  226. console.log('force 参数:', force);
  227. console.log('===========================================');
  228. // 查询 Company 的 isPublishing 字段
  229. let isPublishing = false;
  230. try {
  231. const companyQuery = new Parse.Query('Company');
  232. companyQuery.equalTo('objectId', getApp().globalData.company);
  233. companyQuery.select('isPublishing');
  234. try {
  235. const companyObj = await companyQuery.first();
  236. if (companyObj) {
  237. isPublishing = companyObj.get('isPublishing') === true;
  238. console.log('📋 Company isPublishing:', isPublishing);
  239. } else {
  240. console.log('⚠️ 未找到 Company 记录,默认 isPublishing = false');
  241. isPublishing = false;
  242. }
  243. } catch (queryError) {
  244. console.error('❌ 查询 Company 失败(可能 token 无效):', queryError.message);
  245. // 查询失败时,默认为 false(强制登录)
  246. isPublishing = false;
  247. }
  248. // 保存到全局,供其他页面使用
  249. getApp().globalData.isPublishing = isPublishing;
  250. wx.setStorageSync('isPublishing', isPublishing);
  251. } catch (error) {
  252. console.error('❌ 查询 Company isPublishing 失败:', error);
  253. // 查询失败时,默认为 false(强制登录)
  254. isPublishing = false;
  255. }
  256. // 根据 isPublishing 决定是否强制登录
  257. if (!currentUser || force) {
  258. console.log('🔄 开始调用 checkAuth...');
  259. // isPublishing == true 时不强制授权,否则强制授权
  260. const forceAuth = !isPublishing;
  261. console.log('🔐 是否强制登录:', forceAuth);
  262. let r = await checkAuth(forceAuth);
  263. console.log('===========================================');
  264. console.log('======= checkAuth 返回结果 =======');
  265. console.log('返回值:', r);
  266. // 重新获取用户信息
  267. currentUser = Parse.User.current();
  268. console.log('checkAuth 后的用户:', currentUser ? currentUser.id : '无');
  269. console.log('checkAuth 后的手机号:', currentUser?.get('mobile') || '无');
  270. console.log('checkAuth 后的 Session Token:', currentUser?.getSessionToken()?.substring(0, 20) || '无');
  271. console.log('===========================================');
  272. // 如果强制登录但用户未登录,不允许继续访问
  273. if (forceAuth && !r) {
  274. console.log('❌ 强制登录模式,用户未登录,停止访问');
  275. return;
  276. }
  277. // 即使登录失败,也允许继续访问(仅在非强制登录模式下)
  278. if(!r) {
  279. console.log('⚠️ 用户未登录或拒绝授权,允许游客访问');
  280. if (wx.getStorageSync('need_scan_redirect') === true) {
  281. await this.ensureTrackingUser();
  282. await this.checkAndRecordUserSourceOnLogin();
  283. }
  284. // 不要 return,继续执行后面的跳转逻辑
  285. } else {
  286. // 登录成功,设置 userLogin
  287. if (currentUser && currentUser.get('mobile')) {
  288. wx.setStorageSync("userLogin", currentUser.id);
  289. console.log('✅ 授权登录成功,已设置 userLogin:', currentUser.id);
  290. console.log('✅ 用户手机号:', currentUser.get('mobile'));
  291. } else {
  292. console.warn('⚠️ checkAuth 返回成功,但用户没有手机号!');
  293. console.warn(' 用户对象:', currentUser);
  294. }
  295. // 检查是否有待记录的扫码信息
  296. await this.checkAndRecordPendingScan();
  297. await this.checkTrafficDeductPending();
  298. // 如果用户没有来源信息,且当前有扫码参数,记录来源
  299. await this.checkAndRecordUserSourceOnLogin();
  300. }
  301. } else {
  302. console.log('✅ 用户已登录,跳过 checkAuth');
  303. // 用户已登录,确保 userLogin 已设置
  304. if (currentUser.get('mobile')) {
  305. wx.setStorageSync("userLogin", currentUser.id);
  306. console.log('✅ 已确认 userLogin:', currentUser.id);
  307. }
  308. this.updateUser(currentUser.id);
  309. // 用户已登录,检查是否有待记录的扫码信息
  310. await this.checkAndRecordPendingScan();
  311. await this.checkTrafficDeductPending();
  312. // 如果用户没有来源信息,且当前有扫码参数,记录来源
  313. await this.checkAndRecordUserSourceOnLogin();
  314. }
  315. getApp().Parse = Parse
  316. getApp().checkAuth = checkAuth
  317. getApp().checkTrafficDeductPending = this.checkTrafficDeductPending
  318. if (!await this.getCompanyServerExpire(url)) {
  319. return
  320. }
  321. // 检查是否需要跳转到活动页面
  322. if (wx.getStorageSync('need_activity_redirect') === true) {
  323. await this.redirectToActivityPage();
  324. return;
  325. }
  326. // 检查是否需要跳转到扫码统计页面
  327. if (this.shouldRedirectToScanPage()) {
  328. await this.redirectToScanPage();
  329. return;
  330. }
  331. console.log('✅ 准备跳转到:', url);
  332. wx.redirectTo({
  333. url: url,
  334. success: () => {
  335. console.log('✅ redirectTo 跳转成功');
  336. },
  337. fail: (err) => {
  338. console.error('❌ redirectTo 失败:', err);
  339. console.log('⚠️ 尝试使用 reLaunch');
  340. // 降级:尝试使用 reLaunch
  341. wx.reLaunch({
  342. url: url,
  343. success: () => {
  344. console.log('✅ reLaunch 跳转成功');
  345. },
  346. fail: (err2) => {
  347. console.error('❌ reLaunch 也失败:', err2);
  348. // 显示错误提示
  349. this.setData({ loading: false });
  350. wx.showModal({
  351. title: '温馨提示',
  352. content: '页面加载失败,请检查网络后重试。',
  353. showCancel: true,
  354. cancelText: '退出',
  355. confirmText: '重试',
  356. success: (result) => {
  357. if (result.confirm) {
  358. this.review(true);
  359. } else {
  360. wx.exitMiniProgram();
  361. }
  362. }
  363. });
  364. }
  365. });
  366. }
  367. });
  368. }
  369. catch (err) {
  370. console.log('❌ review 方法出错:', err);
  371. /* 登录身份信息到期,重新登陆 */
  372. if((err?.message?.indexOf('Session token is expired') != -1 || err?.message?.indexOf('Invalid session token') != -1) && !force){
  373. console.log('⚠️ Session Token 过期,准备重新登录');
  374. // 保存需要保留的数据
  375. const invite = wx.getStorageSync('invite');
  376. const agreementAccepted = wx.getStorageSync('user_agreement_accepted');
  377. const storeId = wx.getStorageSync('storeId');
  378. const isPublishing = wx.getStorageSync('isPublishing');
  379. // 先登出 Parse 用户
  380. try {
  381. await Parse.User.logOut();
  382. console.log('✅ Parse 用户已登出');
  383. } catch (logoutErr) {
  384. console.warn('⚠️ Parse 登出失败:', logoutErr);
  385. }
  386. // 清除所有存储
  387. wx.clearStorageSync();
  388. // 恢复需要保留的数据
  389. if (invite) wx.setStorageSync('invite', invite);
  390. if (agreementAccepted) wx.setStorageSync('user_agreement_accepted', agreementAccepted);
  391. if (storeId) wx.setStorageSync('storeId', storeId);
  392. if (isPublishing !== undefined) wx.setStorageSync('isPublishing', isPublishing);
  393. console.log('✅ 已清除过期登录信息,保留必要数据');
  394. console.log(' 保留的 invite:', invite || '无');
  395. console.log(' 保留的协议状态:', agreementAccepted || '无');
  396. console.log(' 保留的店铺ID:', storeId || '无');
  397. /* 强制重新登录 */
  398. this.review(true);
  399. return;
  400. }
  401. // 如果出错,尝试继续跳转到主页(游客模式)
  402. console.log('⚠️ 登录出错,尝试游客模式访问');
  403. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath;
  404. if (this.data.options) {
  405. let objArr = Object.keys(this.data.options)
  406. if (objArr && objArr.length > 0) {
  407. let parms = '?'
  408. objArr.forEach((o, index) => {
  409. if (index > 0) {
  410. parms += '&' + o + '=' + this.data.options[o]
  411. } else {
  412. parms += o + '=' + this.data.options[o]
  413. }
  414. })
  415. url += parms
  416. }
  417. }
  418. wx.redirectTo({
  419. url: url,
  420. fail: () => {
  421. // 如果跳转失败,显示错误提示
  422. this.setData({
  423. loading:false
  424. })
  425. wx.showModal({
  426. title: '温馨提示',
  427. content: '页面加载失败,请检查网络后重试。',
  428. showCancel: true,
  429. cancelText: '退出',
  430. confirmText: '重试',
  431. success: (result) => {
  432. if (result.confirm) {
  433. // 用户选择重试
  434. this.review(true)
  435. } else {
  436. // 用户选择退出
  437. wx.exitMiniProgram()
  438. }
  439. },
  440. });
  441. }
  442. });
  443. }
  444. },
  445. /**
  446. * 检查并记录待处理的扫码统计
  447. * 同时记录用户来源信息到 _User 表
  448. */
  449. async checkAndRecordPendingScan() {
  450. try {
  451. const pendingScan = wx.getStorageSync('pending_scan_record');
  452. if (!pendingScan) {
  453. return;
  454. }
  455. console.log('===========================================');
  456. console.log('======= 发现待记录的扫码信息 =======');
  457. console.log('扫码信息:', pendingScan);
  458. console.log('===========================================');
  459. // 检查是否超时(24小时)
  460. const now = Date.now();
  461. const scanTime = pendingScan.timestamp || 0;
  462. const hoursPassed = (now - scanTime) / (1000 * 60 * 60);
  463. if (hoursPassed > 24) {
  464. console.log('⚠️ 扫码信息已超过24小时,不再记录');
  465. wx.removeStorageSync('pending_scan_record');
  466. return;
  467. }
  468. // 记录用户来源信息
  469. await this.recordUserSource(pendingScan);
  470. // 记录扫码统计
  471. await this.recordScanStatistics({
  472. storeId: pendingScan.storeId,
  473. sourceType: pendingScan.sourceType,
  474. sourceId: pendingScan.sourceId,
  475. ownerId: pendingScan.ownerId,
  476. employeeId: pendingScan.employeeId,
  477. partnerId: pendingScan.partnerId,
  478. userId: pendingScan.userId,
  479. productId: pendingScan.productId,
  480. scanCount: pendingScan.scanCount
  481. });
  482. // 清除待记录的扫码信息
  483. wx.removeStorageSync('pending_scan_record');
  484. console.log('✅ 扫码统计已记录并清除');
  485. } catch (error) {
  486. console.error('❌ 记录待处理扫码信息失败:', error);
  487. }
  488. },
  489. /**
  490. * 检查并处理暂缓扣减任务(在获取到手机号后复判白名单并决定是否扣减)
  491. */
  492. async checkTrafficDeductPending() {
  493. try {
  494. const pending = wx.getStorageSync('traffic_deduct_pending');
  495. if (!pending) return;
  496. const currentUser = Parse.User.current();
  497. if (!currentUser || !currentUser.id) return;
  498. const beforeNumbers = collectUserMobiles(currentUser);
  499. console.log('🚦 [traffic] 检测到暂缓扣减任务:', {
  500. storeId: pending && pending.storeId,
  501. userId: currentUser.id,
  502. userNumbers: beforeNumbers
  503. });
  504. let userNumbers = beforeNumbers;
  505. try {
  506. const sessionToken = typeof currentUser.getSessionToken === 'function' ? currentUser.getSessionToken() : null;
  507. const q = new Parse.Query('_User');
  508. q.select('mobile', 'phone');
  509. const freshUser = await q.get(currentUser.id, sessionToken ? { sessionToken } : undefined);
  510. const afterNumbers = collectUserMobiles(freshUser);
  511. console.log('🚦 [traffic] 刷新用户后号码:', { before: beforeNumbers, after: afterNumbers });
  512. if (Array.isArray(afterNumbers) && afterNumbers.length) {
  513. userNumbers = afterNumbers;
  514. }
  515. } catch (e) {
  516. console.warn('🚦 [traffic] 刷新用户手机号失败:', e?.message || e);
  517. }
  518. if (!userNumbers || !userNumbers.length) {
  519. const storageMobile = wx.getStorageSync('user_mobile') || '';
  520. console.log('🚦 [traffic] 仍未获取到手机号,继续暂缓,并安排 800ms 后重试', {
  521. storageMobile
  522. });
  523. try {
  524. const retryInfo = wx.getStorageSync('traffic_deduct_retry') || { count: 0 };
  525. if ((retryInfo.count || 0) < 3) {
  526. wx.setStorageSync('traffic_deduct_retry', { count: (retryInfo.count || 0) + 1 });
  527. setTimeout(() => {
  528. try {
  529. if (typeof getApp().checkTrafficDeductPending === 'function') {
  530. console.log('🚦 [traffic] 重试触发暂缓扣减复判');
  531. getApp().checkTrafficDeductPending();
  532. }
  533. } catch (e) {}
  534. }, 800);
  535. }
  536. } catch (e) {}
  537. return;
  538. }
  539. const storeId = pending && pending.storeId ? pending.storeId : null;
  540. if (!storeId) {
  541. console.warn('🚦 [traffic] 暂缓扣减任务缺少 storeId,清除任务');
  542. wx.removeStorageSync('traffic_deduct_pending');
  543. return;
  544. }
  545. const isExempt = await isTrafficExemptForStore(Parse, storeId, currentUser);
  546. if (isExempt) {
  547. console.log('🚦 [traffic] 暂缓扣减复判:命中白名单,取消扣减并清除暂缓任务');
  548. try {
  549. currentUser.set('trafficExempt', true);
  550. currentUser.set('trafficExemptAt', new Date());
  551. currentUser.set('trafficExemptStoreId', storeId);
  552. await currentUser.save();
  553. } catch (e) {
  554. console.warn('🚦 [traffic] 写入 trafficExempt 失败(不影响继续访问):', e?.message || e);
  555. }
  556. wx.removeStorageSync('traffic_deduct_pending');
  557. return;
  558. }
  559. console.log('🚦 [traffic] 暂缓扣减复判:未命中白名单,执行扣减');
  560. try {
  561. await decrementStoreTrafficImpl(Parse, storeId);
  562. const deductedStores = Array.isArray(currentUser.get('trafficDeductedStores')) ? currentUser.get('trafficDeductedStores') : [];
  563. const updatedStores = Array.isArray(deductedStores) ? deductedStores.slice() : [];
  564. if (!updatedStores.includes(storeId)) updatedStores.push(storeId);
  565. currentUser.set('trafficDeductedStores', updatedStores);
  566. currentUser.set('trafficDeductedStoreId', storeId);
  567. const atMap = currentUser.get('trafficDeductedAtMap') || {};
  568. atMap[storeId] = new Date();
  569. currentUser.set('trafficDeductedAtMap', atMap);
  570. await currentUser.save();
  571. console.log('🚦✅ [traffic] 暂缓扣减复判:已为店铺扣减 1 个流量,storeId:', storeId);
  572. } catch (decErr) {
  573. console.error('🚦❌ [traffic] 暂缓扣减复判:扣减失败:', decErr?.message || decErr);
  574. }
  575. wx.removeStorageSync('traffic_deduct_pending');
  576. } catch (error) {
  577. console.error('🚦❌ [traffic] 处理暂缓扣减任务失败:', error);
  578. }
  579. },
  580. async ensureTrackingUser() {
  581. try {
  582. let currentUser = Parse.User.current();
  583. if (currentUser) {
  584. return currentUser;
  585. }
  586. if (Parse.AnonymousUtils && typeof Parse.AnonymousUtils.logIn === 'function') {
  587. await Parse.AnonymousUtils.logIn();
  588. currentUser = Parse.User.current();
  589. return currentUser || null;
  590. }
  591. return null;
  592. } catch (e) {
  593. console.warn('ensureTrackingUser 失败:', e?.message || e);
  594. return null;
  595. }
  596. },
  597. /**
  598. * 记录用户来源信息到 _User 表
  599. * 根据扫码参数判断来源类型并保存到用户的 source 字段
  600. *
  601. * 来源类型:
  602. * 1. 渠道xxx→异业xxx(老板后台添加的异业)
  603. * 2. 渠道xxx→异业xxx→员工xxx(员工后台添加的异业)
  604. * 3. 员工xxx
  605. * 4. 老板
  606. * 5. 自主进入(无任何推荐)
  607. */
  608. async recordUserSource(scanInfo) {
  609. try {
  610. const currentUser = Parse.User.current();
  611. if (!currentUser) {
  612. console.log('⚠️ 用户未登录,无法记录来源');
  613. return;
  614. }
  615. const existingSource = currentUser.get('source');
  616. console.log('📊 ===========================================');
  617. console.log('📊 [记录来源] 开始记录用户来源信息');
  618. console.log('📊 用户ID:', currentUser.id);
  619. console.log('📊 ===========================================');
  620. const { partnerId, employeeId, ownerId, userId, storeId } = scanInfo;
  621. console.log('📌 [扣减前置] storeId:', storeId || '无');
  622. console.log('📌 [扣减前置] existingSource:', existingSource ? '有' : '无');
  623. console.log('📌 [扣减前置] trafficDeducted:', currentUser.get('trafficDeducted') === true ? 'true' : 'false');
  624. console.log('📌 [扣减前置] trafficDeductedStoreId:', currentUser.get('trafficDeductedStoreId') || '无');
  625. console.log('📌 [扣减前置] trafficDeductedAt:', currentUser.get('trafficDeductedAt') || '无');
  626. if (!existingSource) {
  627. let sourceInfo = {
  628. type: 'self_entry',
  629. label: '自主进入',
  630. timestamp: new Date(),
  631. storeId: storeId
  632. };
  633. if (partnerId && employeeId) {
  634. console.log('🔍 [来源类型] 渠道→异业→员工');
  635. try {
  636. const partnerQuery = new Parse.Query('Partner');
  637. const partner = await partnerQuery.get(partnerId);
  638. const partnerName = partner.get('name') || '未知异业';
  639. const channelName = partner.get('channelName') || partner.get('category') || '未知渠道';
  640. const employeeQuery = new Parse.Query('Employee');
  641. const employee = await employeeQuery.get(employeeId);
  642. const employeeName = employee.get('name') || '未知员工';
  643. sourceInfo = {
  644. type: 'channel_partner_employee',
  645. label: `${channelName},${partnerName},员工${employeeName}`,
  646. channelName: channelName,
  647. partnerId: partnerId,
  648. partnerName: partnerName,
  649. employeeId: employeeId,
  650. employeeName: employeeName,
  651. timestamp: new Date(),
  652. storeId: storeId
  653. };
  654. console.log('✅ [来源信息] 渠道→异业→员工:', sourceInfo.label);
  655. } catch (error) {
  656. console.error('❌ 查询异业或员工信息失败:', error);
  657. }
  658. } else if (partnerId && !employeeId) {
  659. console.log('🔍 [来源类型] 渠道→异业');
  660. try {
  661. const partnerQuery = new Parse.Query('Partner');
  662. const partner = await partnerQuery.get(partnerId);
  663. const partnerName = partner.get('name') || '未知异业';
  664. const channelName = partner.get('channelName') || partner.get('category') || '未知渠道';
  665. sourceInfo = {
  666. type: 'channel_partner',
  667. label: `${channelName},${partnerName}`,
  668. channelName: channelName,
  669. partnerId: partnerId,
  670. partnerName: partnerName,
  671. timestamp: new Date(),
  672. storeId: storeId
  673. };
  674. console.log('✅ [来源信息] 渠道→异业:', sourceInfo.label);
  675. } catch (error) {
  676. console.error('❌ 查询异业信息失败:', error);
  677. }
  678. } else if (employeeId && !partnerId) {
  679. console.log('🔍 [来源类型] 员工');
  680. try {
  681. const employeeQuery = new Parse.Query('Employee');
  682. const employee = await employeeQuery.get(employeeId);
  683. const employeeName = employee.get('name') || '未知员工';
  684. sourceInfo = {
  685. type: 'employee',
  686. label: employeeName,
  687. employeeId: employeeId,
  688. employeeName: employeeName,
  689. timestamp: new Date(),
  690. storeId: storeId
  691. };
  692. console.log('✅ [来源信息] 员工:', sourceInfo.label);
  693. } catch (error) {
  694. console.error('❌ 查询员工信息失败:', error);
  695. }
  696. } else if (ownerId) {
  697. console.log('🔍 [来源类型] 老板');
  698. sourceInfo = {
  699. type: 'owner',
  700. label: '老板',
  701. ownerId: ownerId,
  702. timestamp: new Date(),
  703. storeId: storeId
  704. };
  705. console.log('✅ [来源信息] 老板');
  706. } else if (userId) {
  707. console.log('🔍 [来源类型] 推广员');
  708. try {
  709. const userQuery = new Parse.Query('_User');
  710. const promoter = await userQuery.get(userId);
  711. const promoterName = promoter.get('username') || promoter.get('mobile') || '未知推广员';
  712. sourceInfo = {
  713. type: 'promoter',
  714. label: `推广员${promoterName}`,
  715. userId: userId,
  716. promoterName: promoterName,
  717. timestamp: new Date(),
  718. storeId: storeId
  719. };
  720. console.log('✅ [来源信息] 推广员:', sourceInfo.label);
  721. } catch (error) {
  722. console.error('❌ 查询推广员信息失败:', error);
  723. }
  724. } else {
  725. console.log('🔍 [来源类型] 自主进入');
  726. console.log('✅ [来源信息] 自主进入');
  727. }
  728. currentUser.set('source', sourceInfo);
  729. await currentUser.save();
  730. console.log('✅ [保存成功] 用户来源信息已保存');
  731. console.log(' - 来源类型:', sourceInfo.type);
  732. console.log(' - 来源标签:', sourceInfo.label);
  733. console.log('📊 ===========================================');
  734. } else {
  735. console.log('ℹ️ 用户已有来源信息,跳过覆盖:', existingSource);
  736. }
  737. const deductedStores = Array.isArray(currentUser.get('trafficDeductedStores')) ? currentUser.get('trafficDeductedStores') : [];
  738. const hasDeductedForThisStore =
  739. (Array.isArray(deductedStores) && deductedStores.includes(storeId)) ||
  740. (currentUser.get('trafficDeductedStoreId') && currentUser.get('trafficDeductedStoreId') === storeId);
  741. if (storeId && !hasDeductedForThisStore) {
  742. const internal = await isInternalVisit(Parse, storeId, Parse.User.current(), { ownerId, employeeId, partnerId });
  743. if (internal) {
  744. console.log('ℹ️ [扣减跳过] 内部角色访问(老板/异业/员工),不扣减流量');
  745. } else {
  746. const nums = collectUserMobiles(currentUser);
  747. if (!nums || !nums.length) {
  748. console.log('ℹ️ [扣减暂停] 用户未提供手机号,暂缓扣减', { storeId, userId: currentUser.id });
  749. try {
  750. wx.setStorageSync('traffic_deduct_pending', { storeId, userId: currentUser.id, time: Date.now() });
  751. } catch (e) {}
  752. } else {
  753. const isExempt = await isTrafficExemptForStore(Parse, storeId, currentUser);
  754. if (isExempt) {
  755. console.log('ℹ️ [扣减跳过] 命中白名单手机号,跳过扣减流量');
  756. try {
  757. currentUser.set('trafficExempt', true);
  758. currentUser.set('trafficExemptAt', new Date());
  759. currentUser.set('trafficExemptStoreId', storeId);
  760. await currentUser.save();
  761. } catch (e) {
  762. console.warn('⚠️ [扣减跳过] 写入 trafficExempt 失败(不影响继续访问):', e?.message || e);
  763. }
  764. } else {
  765. console.log('🧮 [扣减流量] 检测到新用户首次扣减,准备扣减门店流量');
  766. try {
  767. await decrementStoreTrafficImpl(Parse, storeId);
  768. const updatedStores = Array.isArray(deductedStores) ? deductedStores.slice() : [];
  769. if (!updatedStores.includes(storeId)) updatedStores.push(storeId);
  770. currentUser.set('trafficDeductedStores', updatedStores);
  771. currentUser.set('trafficDeductedStoreId', storeId);
  772. const atMap = currentUser.get('trafficDeductedAtMap') || {};
  773. atMap[storeId] = new Date();
  774. currentUser.set('trafficDeductedAtMap', atMap);
  775. await currentUser.save();
  776. console.log('✅ [扣减成功] 已为店铺扣减 1 个流量,storeId:', storeId);
  777. } catch (decErr) {
  778. console.error('❌ [扣减失败] 扣减门店流量失败:', decErr?.message || decErr);
  779. }
  780. }
  781. }
  782. }
  783. } else if (!storeId) {
  784. console.warn('⚠️ [扣减跳过] storeId 为空,无法扣减 ShopStore.traffic');
  785. } else if (hasDeductedForThisStore) {
  786. console.log('ℹ️ [扣减跳过] 该用户已在该门店扣减过流量,不重复扣减');
  787. }
  788. } catch (error) {
  789. console.error('❌ 记录用户来源失败:', error);
  790. }
  791. },
  792. /**
  793. * 用户登录后检查并记录来源信息
  794. * 如果用户是首次登录且没有来源信息,根据当前的扫码参数记录来源
  795. */
  796. async checkAndRecordUserSourceOnLogin() {
  797. try {
  798. const currentUser = Parse.User.current();
  799. if (!currentUser) {
  800. return;
  801. }
  802. // 检查是否有扫码参数
  803. const scanStoreId = wx.getStorageSync('scan_storeId');
  804. const scanPartnerId = wx.getStorageSync('scan_partnerId');
  805. const scanEmployeeId = wx.getStorageSync('scan_employeeId');
  806. const scanOwnerId = wx.getStorageSync('scan_ownerId');
  807. const scanUserId = wx.getStorageSync('scan_userId');
  808. // 如果有任何扫码参数,记录来源
  809. if (scanStoreId || scanPartnerId || scanEmployeeId || scanOwnerId || scanUserId) {
  810. console.log('📊 检测到扫码参数,记录用户来源');
  811. await this.recordUserSource({
  812. storeId: scanStoreId,
  813. partnerId: scanPartnerId,
  814. employeeId: scanEmployeeId,
  815. ownerId: scanOwnerId,
  816. userId: scanUserId
  817. });
  818. } else {
  819. // 没有任何扫码参数,标记为自主进入
  820. console.log('📊 无扫码参数,标记为自主进入');
  821. await this.recordUserSource({
  822. storeId: wx.getStorageSync('storeId') || null
  823. });
  824. }
  825. } catch (error) {
  826. console.error('❌ 检查并记录用户来源失败:', error);
  827. }
  828. },
  829. async updateUser(id) {
  830. try {
  831. let User = new Parse.Query('_User')
  832. let user = await User.get(id)
  833. let invite = wx.getStorageSync('invite')
  834. //查询邀请人user
  835. let query = new Parse.Query("_User")
  836. query.equalTo('objectId', invite)
  837. let result = await query.first()
  838. if (result && result.id && result.get("invite")?.id == user.id) {
  839. console.error('邀请人不能是自己的下级')
  840. return
  841. }
  842. if (invite && !user.get('invite') && user.id != invite && !user.get('agentLevel')) {
  843. console.log('上下级绑定成功');
  844. user.set('invite', {
  845. __type: "Pointer",
  846. className: "_User",
  847. objectId: invite
  848. })
  849. user.set('agent', {
  850. __type: "Pointer",
  851. className: "_User",
  852. objectId: invite
  853. })
  854. await Parse.Cloud.run('user_save', {
  855. userJson: user.toJSON()
  856. })
  857. }
  858. } catch (error) {
  859. console.error('❌ updateUser 失败:', error.message);
  860. // 不阻断流程,继续执行
  861. }
  862. },
  863. async getCompanyServerExpire(url) {
  864. try {
  865. let query = new Parse.Query('Company')
  866. query.equalTo('objectId', getApp().globalData.company)
  867. query.select('expireDate', 'expireMap')
  868. let com = await query.first()
  869. if (com?.id && com?.get('expireDate')) {
  870. let now = + new Date()
  871. let expireTime = + new Date(com?.get('expireDate'))
  872. if (com?.get('expireMap') && com.get('expireMap')[getApp().globalData.appid]) {
  873. expireTime = + new Date(com.get('expireMap')[getApp().globalData.appid])
  874. }
  875. if (now >= expireTime) {
  876. console.log('服务器到期');
  877. wx.reLaunch({
  878. url: `common-page/pages/loading/index?url=${url}`,
  879. });
  880. return
  881. }
  882. }
  883. return true
  884. } catch (error) {
  885. console.error('❌ getCompanyServerExpire 失败:', error.message);
  886. // 查询失败时,允许继续访问
  887. return true
  888. }
  889. },
  890. onUnload: function () {
  891. wx.setStorageSync("active", 0);
  892. },
  893. getParaName(url) {
  894. if (!url || url.indexOf('?') == -1) {
  895. return
  896. }
  897. // 兼容 URL 中 &amp; 的情况,先统一还原为 &
  898. if (url.indexOf('&amp;') !== -1) {
  899. url = url.replace(/&amp;/g, '&');
  900. }
  901. // 提取查询参数部分(去除可能的 hash 部分)
  902. let queryString = url.split('?')[1];
  903. // 如果包含 #,只取 # 之前的部分
  904. if (queryString.indexOf('#') !== -1) {
  905. queryString = queryString.split('#')[0];
  906. }
  907. return this.setObject(queryString) //封装成对象
  908. },
  909. setObject(paraArr) {
  910. let obj = {}
  911. let arr1 = paraArr.split('&')
  912. arr1.forEach(item => {
  913. let str = item.split('=')
  914. let key = str[0]
  915. let val = str[1]
  916. obj[key] = val
  917. })
  918. return obj
  919. },
  920. /**
  921. * 检查并处理扫码参数
  922. * 支持所有类型的二维码:
  923. * 1. 推广员二维码: ?scanCount=0&storeId=xxx&userId=xxx
  924. * 2. 产品二维码: ?scanCount=0&storeId=xxx&productId=xxx
  925. * 3. 案例二维码: ?scanCount=0&storeId=xxx&caseId=xxx
  926. * 4. 活动海报二维码: activityId作为qrCode参数值
  927. * 5. 异业合作伙伴二维码(移动端 / PC端统一): ?scanCount=xxx&storeId=xxx&partnerId=xxx
  928. * 6. 员工邀请二维码(移动端): ?scanCount=0&storeId=xxx&employeeId=xxx
  929. * 7. 我的二维码(老板): ?scanCount=0&storeId=xxx&ownerId=xxx
  930. * 8. 我的二维码(员工): ?scanCount=0&storeId=xxx&employeeId=xxx
  931. * 9. 方案分享二维码: ?scanCount=0&storeId=xxx&planId=xxx&shareUserId=xxx
  932. */
  933. checkAndHandleScan(options) {
  934. console.log('🔍🔍🔍 ========================================');
  935. console.log('🔍 [扫码检测] 开始检查扫码参数');
  936. console.log('🔍 完整 options 对象:', JSON.stringify(options, null, 2));
  937. console.log('🔍🔍🔍 ========================================');
  938. const {
  939. scanCount,
  940. ownerId,
  941. employeeId,
  942. partnerId,
  943. storeId,
  944. productId,
  945. caseId,
  946. userId, // 推广员二维码使用userId
  947. activityId,
  948. schemeId, // 方案ID(旧参数名)
  949. planId, // 方案ID(新参数名,优先使用)
  950. shareUserId, // 分享方案的用户ID
  951. q // 扫码链接
  952. } = options;
  953. // planId 和 schemeId 兼容处理,优先使用 planId
  954. const finalPlanId = planId || schemeId;
  955. console.log('📋 [扫码参数] 提取的参数:');
  956. console.log(' - scanCount:', scanCount || '无');
  957. console.log(' - storeId:', storeId || '无');
  958. console.log(' - ownerId:', ownerId || '无');
  959. console.log(' - employeeId:', employeeId || '无');
  960. console.log(' - partnerId:', partnerId || '无');
  961. console.log(' - userId:', userId || '无');
  962. console.log(' - productId:', productId || '无');
  963. console.log(' - caseId:', caseId || '无');
  964. console.log(' - activityId:', activityId || '无');
  965. console.log(' - planId:', planId || '无');
  966. console.log(' - schemeId:', schemeId || '无');
  967. console.log(' - finalPlanId (最终使用):', finalPlanId || '无');
  968. console.log(' - shareUserId:', shareUserId || '无');
  969. console.log(' - q (扫码链接):', q || '无');
  970. // 处理活动海报二维码(activityId作为qrCode参数值)
  971. if (activityId && !storeId) {
  972. console.log('🎯 ===========================================');
  973. console.log('🎯 [活动二维码] 检测到活动海报二维码');
  974. console.log('🎯 活动ID (activityId):', activityId);
  975. console.log('🎯 来源类型: 活动海报');
  976. console.log('🎯 ===========================================');
  977. // 活动二维码需要跳转到活动页面,不需要跳转到店铺
  978. wx.setStorageSync('scan_activityId', activityId);
  979. wx.setStorageSync('need_activity_redirect', true);
  980. return;
  981. }
  982. // 如果存在 storeId,说明是扫码进入具体门店(包括异业分享)
  983. if (storeId) {
  984. console.log('🏪 ===========================================');
  985. console.log('🏪 [店铺扫码] 检测到扫码参数');
  986. console.log('🏪 店铺ID (storeId):', storeId);
  987. console.log('🏪 扫码次数 (scanCount):', scanCount || '0');
  988. // 确定来源类型
  989. // 规则:
  990. // 1. 扫描异业码(partnerId)→ sourceType = "partner"
  991. // 2. 扫描员工码(employeeId、老板/员工)→ sourceType = "sales" 或 "promoter"
  992. // 3. 扫描其他码(产品码、案例码等)→ sourceType = "scan"
  993. let sourceType = 'scan'; // 默认为通用扫码
  994. let sourceId = null;
  995. if (partnerId) {
  996. // 异业合作伙伴二维码
  997. sourceType = 'partner';
  998. sourceId = partnerId;
  999. console.log('🤝 [来源识别] 来源类型: 异业合作伙伴 (partner)');
  1000. console.log('🤝 合作伙伴ID (partnerId):', partnerId);
  1001. } else if (employeeId) {
  1002. // 员工二维码(包括老板和员工)
  1003. sourceType = 'sales';
  1004. sourceId = employeeId;
  1005. console.log('👤 [来源识别] 来源类型: 员工展业 (sales)');
  1006. console.log('👤 员工ID (employeeId):', employeeId);
  1007. } else if (ownerId) {
  1008. // 老板二维码
  1009. sourceType = 'sales';
  1010. sourceId = ownerId;
  1011. console.log('👔 [来源识别] 来源类型: 老板展业 (sales)');
  1012. console.log('👔 老板ID (ownerId):', ownerId);
  1013. } else if (userId) {
  1014. // 推广员二维码
  1015. sourceType = 'promoter';
  1016. sourceId = userId;
  1017. console.log('📢 [来源识别] 来源类型: 推广员展业 (promoter)');
  1018. console.log('📢 推广员ID (userId):', userId);
  1019. } else {
  1020. // 其他情况(产品码、案例码、店铺分享等)
  1021. sourceType = 'scan';
  1022. console.log('🏬 [来源识别] 来源类型: 通用扫码 (scan)');
  1023. }
  1024. if (productId) {
  1025. console.log('📦 [目标内容] 产品ID (productId):', productId);
  1026. }
  1027. if (caseId) {
  1028. console.log('📸 [目标内容] 案例ID (caseId):', caseId);
  1029. }
  1030. if (finalPlanId) {
  1031. console.log('📋 [目标内容] 方案ID (planId/schemeId):', finalPlanId);
  1032. console.log('👤 [分享人] 分享用户ID (shareUserId):', shareUserId || '无');
  1033. }
  1034. console.log('🏪 ===========================================');
  1035. // 保存到临时存储,用于后续跳转和统计
  1036. console.log('💾 [保存扫码信息] 开始保存到本地存储...');
  1037. wx.setStorageSync('scan_storeId', storeId);
  1038. wx.setStorageSync('scan_scanCount', scanCount || '0');
  1039. wx.setStorageSync('scan_sourceType', sourceType);
  1040. wx.setStorageSync('scan_sourceId', sourceId || '');
  1041. wx.setStorageSync('scan_ownerId', ownerId || '');
  1042. wx.setStorageSync('scan_employeeId', employeeId || '');
  1043. wx.setStorageSync('scan_partnerId', partnerId || '');
  1044. wx.setStorageSync('scan_userId', userId || '');
  1045. if (productId) {
  1046. wx.setStorageSync('scan_productId', productId);
  1047. }
  1048. if (caseId) {
  1049. wx.setStorageSync('scan_caseId', caseId);
  1050. }
  1051. if (finalPlanId) {
  1052. wx.setStorageSync('scan_planId', finalPlanId);
  1053. if (shareUserId) {
  1054. wx.setStorageSync('scan_shareUserId', shareUserId);
  1055. }
  1056. }
  1057. wx.setStorageSync('need_scan_redirect', true);
  1058. console.log('✅ [保存完成] 扫码信息已保存');
  1059. console.log('📋 [保存内容摘要]:');
  1060. console.log(' - 来源类型:', sourceType);
  1061. console.log(' - 来源ID:', sourceId || '无');
  1062. console.log(' - 店铺ID:', storeId);
  1063. console.log(' - 需要跳转:', true);
  1064. } else if (partnerId) {
  1065. // 理论上异业二维码现在也必须带 storeId,这里仅作为兜底保护
  1066. console.warn('⚠️ 检测到异业合作伙伴二维码缺少 storeId,无法确定门店,请检查二维码生成逻辑');
  1067. }
  1068. },
  1069. /**
  1070. * 判断是否需要跳转到扫码统计页面
  1071. */
  1072. shouldRedirectToScanPage() {
  1073. return wx.getStorageSync('need_scan_redirect') === true;
  1074. },
  1075. /**
  1076. * 跳转到扫码对应的店铺页面
  1077. * 根据 storeId 跳转到对应店铺,同时记录所有来源信息用于统计
  1078. */
  1079. async redirectToScanPage() {
  1080. try {
  1081. console.log('🚀 ===========================================');
  1082. console.log('🚀 [跳转处理] 开始处理扫码跳转');
  1083. console.log('🚀 ===========================================');
  1084. // 获取所有扫码相关参数
  1085. const storeId = wx.getStorageSync('scan_storeId');
  1086. const scanCount = wx.getStorageSync('scan_scanCount');
  1087. const sourceType = wx.getStorageSync('scan_sourceType');
  1088. const sourceId = wx.getStorageSync('scan_sourceId');
  1089. const ownerId = wx.getStorageSync('scan_ownerId');
  1090. const employeeId = wx.getStorageSync('scan_employeeId');
  1091. const partnerId = wx.getStorageSync('scan_partnerId');
  1092. const userId = wx.getStorageSync('scan_userId');
  1093. const productId = wx.getStorageSync('scan_productId');
  1094. const caseId = wx.getStorageSync('scan_caseId');
  1095. const planId = wx.getStorageSync('scan_planId');
  1096. const shareUserId = wx.getStorageSync('scan_shareUserId');
  1097. console.log('📖 [读取存储] 从本地存储读取扫码信息:');
  1098. console.log(' - storeId:', storeId || '无');
  1099. console.log(' - sourceType:', sourceType || '无');
  1100. console.log(' - sourceId:', sourceId || '无');
  1101. console.log(' - scanCount:', scanCount || '无');
  1102. console.log(' - ownerId:', ownerId || '无');
  1103. console.log(' - employeeId:', employeeId || '无');
  1104. console.log(' - partnerId:', partnerId || '无');
  1105. console.log(' - userId:', userId || '无');
  1106. console.log(' - productId:', productId || '无');
  1107. console.log(' - caseId:', caseId || '无');
  1108. console.log(' - planId:', planId || '无');
  1109. console.log(' - shareUserId:', shareUserId || '无');
  1110. // 清除临时存储
  1111. console.log('🧹 [清理存储] 清除临时扫码信息...');
  1112. wx.removeStorageSync('scan_storeId');
  1113. wx.removeStorageSync('scan_scanCount');
  1114. wx.removeStorageSync('scan_sourceType');
  1115. wx.removeStorageSync('scan_sourceId');
  1116. wx.removeStorageSync('scan_ownerId');
  1117. wx.removeStorageSync('scan_employeeId');
  1118. wx.removeStorageSync('scan_partnerId');
  1119. wx.removeStorageSync('scan_userId');
  1120. wx.removeStorageSync('scan_productId');
  1121. wx.removeStorageSync('scan_caseId');
  1122. wx.removeStorageSync('scan_planId');
  1123. wx.removeStorageSync('scan_shareUserId');
  1124. wx.removeStorageSync('need_scan_redirect');
  1125. console.log('✅ [清理完成] 临时存储已清除');
  1126. if (!storeId) {
  1127. console.error('❌ [错误] 缺少 storeId 参数,无法跳转');
  1128. return;
  1129. }
  1130. console.log('📊 ===========================================');
  1131. console.log('📊 [扫码来源汇总] 扫码进入店铺');
  1132. console.log('📊 店铺 ID:', storeId);
  1133. console.log('📊 来源类型:', sourceType);
  1134. console.log('📊 来源 ID:', sourceId || '无');
  1135. console.log('📊 扫码次数:', scanCount);
  1136. if (ownerId) console.log('📊 老板 ID:', ownerId);
  1137. if (employeeId) console.log('📊 员工 ID:', employeeId);
  1138. if (partnerId) console.log('📊 合作伙伴 ID:', partnerId);
  1139. if (userId) console.log('📊 推广员 ID:', userId);
  1140. if (productId) console.log('📊 产品 ID:', productId);
  1141. if (caseId) console.log('📊 案例 ID:', caseId);
  1142. if (planId) {
  1143. console.log('📊 方案 ID:', planId);
  1144. console.log('📊 分享用户 ID:', shareUserId || '无');
  1145. }
  1146. console.log('📊 ===========================================');
  1147. // 设置店铺 ID 到全局和本地存储
  1148. wx.setStorageSync('storeId', storeId);
  1149. getApp().globalData.storeId = storeId;
  1150. // 保存来源信息到本地存储(用于后续统计或绑定关系)
  1151. if (sourceId) {
  1152. wx.setStorageSync('scan_from_sourceType', sourceType);
  1153. wx.setStorageSync('scan_from_sourceId', sourceId);
  1154. console.log('✅ 已记录来源信息:', sourceType, sourceId);
  1155. }
  1156. // 检查用户是否已登录
  1157. const currentUser = Parse.User.current();
  1158. console.log('👤 [登录检查] 检查用户登录状态...');
  1159. console.log(' - 当前用户:', currentUser ? currentUser.id : '未登录');
  1160. console.log(' - 手机号:', currentUser?.get('mobile') || '无');
  1161. if (currentUser) {
  1162. // 用户已登录,立即记录扫码统计
  1163. console.log('✅ [已登录] 用户已登录,立即记录扫码统计');
  1164. await this.recordScanStatistics({
  1165. storeId,
  1166. sourceType,
  1167. sourceId,
  1168. ownerId,
  1169. employeeId,
  1170. partnerId,
  1171. userId,
  1172. productId,
  1173. scanCount
  1174. });
  1175. } else {
  1176. // 用户未登录,保存扫码信息,等登录后再记录
  1177. console.log('⚠️ [未登录] 用户未登录,保存扫码信息待登录后记录');
  1178. const pendingData = {
  1179. storeId,
  1180. sourceType,
  1181. sourceId,
  1182. ownerId,
  1183. employeeId,
  1184. partnerId,
  1185. userId,
  1186. productId,
  1187. scanCount,
  1188. timestamp: Date.now()
  1189. };
  1190. console.log('💾 [待处理记录] 保存内容:', JSON.stringify(pendingData, null, 2));
  1191. wx.setStorageSync('pending_scan_record', pendingData);
  1192. console.log('✅ [保存成功] 待处理扫码记录已保存');
  1193. }
  1194. // 如果有产品ID,直接跳转到产品详情的 H5 页面
  1195. if (productId) {
  1196. console.log('🎯 检测到产品ID,跳转到产品详情页');
  1197. await this.redirectToProductDetail(storeId, productId, partnerId);
  1198. return;
  1199. }
  1200. // 如果有案例ID,直接跳转到案例详情的 H5 页面
  1201. if (caseId) {
  1202. console.log('🎯 检测到案例ID,跳转到案例详情页');
  1203. await this.redirectToCaseDetail(storeId, caseId, partnerId);
  1204. return;
  1205. }
  1206. // 如果有方案ID,跳转到门店首页并传递方案ID
  1207. if (planId) {
  1208. console.log('🎯 检测到方案ID,跳转到门店首页展示方案');
  1209. await this.redirectToStoreWithPlan(storeId, planId, shareUserId, partnerId);
  1210. return;
  1211. }
  1212. // 获取默认首页路径并跳转
  1213. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath;
  1214. url += `?storeId=${storeId}`;
  1215. // 如果有产品ID,添加到URL参数中
  1216. if (productId) {
  1217. url += `&productId=${productId}`;
  1218. }
  1219. // 如果有异业合作伙伴ID,添加到URL参数中传回web-view
  1220. if (partnerId) {
  1221. url += `&partnerId=${partnerId}`;
  1222. }
  1223. console.log('✅ 跳转到店铺页面:', url);
  1224. wx.redirectTo({
  1225. url: url,
  1226. fail: (err) => {
  1227. console.error('❌ 跳转失败:', err);
  1228. // 如果 redirectTo 失败,尝试 reLaunch
  1229. wx.reLaunch({
  1230. url: url,
  1231. fail: (err2) => {
  1232. console.error('❌ reLaunch 也失败:', err2);
  1233. }
  1234. });
  1235. }
  1236. });
  1237. } catch (error) {
  1238. console.error('❌ 跳转到店铺页面失败:', error);
  1239. }
  1240. },
  1241. /**
  1242. * 跳转到活动页面
  1243. */
  1244. async redirectToActivityPage() {
  1245. try {
  1246. const activityId = wx.getStorageSync('scan_activityId');
  1247. // 清除临时存储
  1248. wx.removeStorageSync('scan_activityId');
  1249. wx.removeStorageSync('need_activity_redirect');
  1250. if (!activityId) {
  1251. console.error('❌ 缺少 activityId 参数,无法跳转');
  1252. return;
  1253. }
  1254. console.log('===========================================');
  1255. console.log('======= 扫码进入活动页面 =======');
  1256. console.log('活动 ID (activityId):', activityId);
  1257. console.log('===========================================');
  1258. // 保存活动ID
  1259. wx.setStorageSync('activityId', activityId);
  1260. // 获取默认首页路径并跳转(活动页面可能需要根据实际路由调整)
  1261. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath;
  1262. url += `?activityId=${activityId}`;
  1263. console.log('✅ 跳转到活动页面:', url);
  1264. wx.redirectTo({
  1265. url: url,
  1266. fail: (err) => {
  1267. console.error('❌ 跳转失败:', err);
  1268. wx.reLaunch({
  1269. url: url,
  1270. fail: (err2) => {
  1271. console.error('❌ reLaunch 也失败:', err2);
  1272. }
  1273. });
  1274. }
  1275. });
  1276. } catch (error) {
  1277. console.error('❌ 跳转到活动页面失败:', error);
  1278. }
  1279. },
  1280. /**
  1281. * 跳转到产品详情的 H5 页面
  1282. * @param {string} storeId - 店铺 ID
  1283. * @param {string} productId - 产品 ID
  1284. * @param {string} partnerId - 可选的合作伙伴 ID
  1285. */
  1286. async redirectToProductDetail(storeId, productId, partnerId = null) {
  1287. try {
  1288. console.log('===========================================');
  1289. console.log('======= 跳转到产品详情页 =======');
  1290. console.log('店铺 ID:', storeId);
  1291. console.log('产品 ID:', productId);
  1292. if (partnerId) console.log('合作伙伴 ID:', partnerId);
  1293. console.log('===========================================');
  1294. const currentUser = Parse.User.current();
  1295. const token = currentUser ? currentUser.getSessionToken() : null;
  1296. // 构建产品详情的 H5 URL(不要在这里编码,后面统一编码)
  1297. let h5Url = `https://app.fmode.cn/dev/pobingfeng/owner/nav/products?storeId=${storeId}`;
  1298. // 如果有 token,添加到 URL
  1299. if (token) {
  1300. h5Url += `&token=${token}`;
  1301. } else {
  1302. // 如果没有 token,使用游客模式
  1303. h5Url += `&guestMode=true`;
  1304. }
  1305. // 添加产品ID(不要在这里编码,避免双重编码)
  1306. h5Url += `&productId=${productId}`;
  1307. // 如果有合作伙伴ID,也添加到URL中
  1308. if (partnerId) {
  1309. h5Url += `&partnerId=${partnerId}`;
  1310. }
  1311. console.log('🌐 H5 URL:', h5Url);
  1312. // 编码 URL(统一在这里编码一次)
  1313. const encodedUrl = encodeURIComponent(h5Url);
  1314. // 构建 web-view 页面路径
  1315. let webViewPath = `/common-page/pages/web-view/index?path=${encodedUrl}&storeId=${storeId}`;
  1316. console.log('📄 web-view 页面路径:', webViewPath.substring(0, 100) + '...');
  1317. console.log('===========================================');
  1318. wx.redirectTo({
  1319. url: webViewPath,
  1320. success: () => {
  1321. console.log('✅ 跳转到产品详情页成功');
  1322. },
  1323. fail: (err) => {
  1324. console.error('❌ 跳转失败:', err);
  1325. // 如果 redirectTo 失败,尝试 reLaunch
  1326. wx.reLaunch({
  1327. url: webViewPath,
  1328. fail: (err2) => {
  1329. console.error('❌ reLaunch 也失败:', err2);
  1330. }
  1331. });
  1332. }
  1333. });
  1334. } catch (error) {
  1335. console.error('❌ 跳转到产品详情页失败:', error);
  1336. }
  1337. },
  1338. /**
  1339. * 跳转到案例详情的 H5 页面
  1340. * @param {string} storeId - 店铺 ID
  1341. * @param {string} caseId - 案例 ID
  1342. * @param {string} partnerId - 可选的合作伙伴 ID
  1343. */
  1344. async redirectToCaseDetail(storeId, caseId, partnerId = null) {
  1345. try {
  1346. console.log('===========================================');
  1347. console.log('======= 跳转到案例详情页 =======');
  1348. console.log('店铺 ID:', storeId);
  1349. console.log('案例 ID:', caseId);
  1350. if (partnerId) console.log('合作伙伴 ID:', partnerId);
  1351. console.log('===========================================');
  1352. const currentUser = Parse.User.current();
  1353. const token = currentUser ? currentUser.getSessionToken() : null;
  1354. // 构建案例详情的 H5 URL(不要在这里编码,后面统一编码)
  1355. let h5Url = `https://app.fmode.cn/dev/pobingfeng/owner/nav/cases?storeId=${storeId}`;
  1356. // 如果有 token,添加到 URL
  1357. if (token) {
  1358. h5Url += `&token=${token}`;
  1359. } else {
  1360. // 如果没有 token,使用游客模式
  1361. h5Url += `&guestMode=true`;
  1362. }
  1363. // 添加案例ID(不要在这里编码,避免双重编码)
  1364. h5Url += `&caseId=${caseId}`;
  1365. // 如果有合作伙伴ID,也添加到URL中
  1366. if (partnerId) {
  1367. h5Url += `&partnerId=${partnerId}`;
  1368. }
  1369. console.log('🌐 H5 URL:', h5Url);
  1370. // 编码 URL(统一在这里编码一次)
  1371. const encodedUrl = encodeURIComponent(h5Url);
  1372. // 构建 web-view 页面路径
  1373. let webViewPath = `/common-page/pages/web-view/index?path=${encodedUrl}&storeId=${storeId}`;
  1374. console.log('📄 web-view 页面路径:', webViewPath.substring(0, 100) + '...');
  1375. console.log('===========================================');
  1376. wx.redirectTo({
  1377. url: webViewPath,
  1378. success: () => {
  1379. console.log('✅ 跳转到案例详情页成功');
  1380. },
  1381. fail: (err) => {
  1382. console.error('❌ 跳转失败:', err);
  1383. // 如果 redirectTo 失败,尝试 reLaunch
  1384. wx.reLaunch({
  1385. url: webViewPath,
  1386. fail: (err2) => {
  1387. console.error('❌ reLaunch 也失败:', err2);
  1388. }
  1389. });
  1390. }
  1391. });
  1392. } catch (error) {
  1393. console.error('❌ 跳转到案例详情页失败:', error);
  1394. }
  1395. },
  1396. /**
  1397. * 跳转到门店首页并展示方案
  1398. * @param {string} storeId - 店铺 ID
  1399. * @param {string} planId - 方案 ID
  1400. * @param {string} shareUserId - 分享方案的用户 ID
  1401. * @param {string} partnerId - 可选的合作伙伴 ID
  1402. */
  1403. async redirectToStoreWithPlan(storeId, planId, shareUserId = null, partnerId = null) {
  1404. try {
  1405. console.log('===========================================');
  1406. console.log('======= 跳转到门店首页展示方案 =======');
  1407. console.log('店铺 ID:', storeId);
  1408. console.log('方案 ID:', planId);
  1409. if (shareUserId) console.log('分享用户 ID:', shareUserId);
  1410. if (partnerId) console.log('合作伙伴 ID:', partnerId);
  1411. console.log('===========================================');
  1412. const currentUser = Parse.User.current();
  1413. const token = currentUser ? currentUser.getSessionToken() : null;
  1414. // 构建门店首页的 H5 URL,带上方案ID
  1415. let h5Url = `https://app.fmode.cn/dev/pobingfeng/owner/nav/home?storeId=${storeId}`;
  1416. // 如果有 token,添加到 URL
  1417. if (token) {
  1418. h5Url += `&token=${token}`;
  1419. } else {
  1420. // 如果没有 token,使用游客模式
  1421. h5Url += `&guestMode=true`;
  1422. }
  1423. // 添加方案ID(使用planId参数名)
  1424. h5Url += `&planId=${planId}`;
  1425. // 如果有分享用户ID,也添加到URL中
  1426. if (shareUserId) {
  1427. h5Url += `&shareUserId=${shareUserId}`;
  1428. }
  1429. // 如果有合作伙伴ID,也添加到URL中
  1430. if (partnerId) {
  1431. h5Url += `&partnerId=${partnerId}`;
  1432. }
  1433. console.log('🌐 H5 URL:', h5Url);
  1434. // 编码 URL
  1435. const encodedUrl = encodeURIComponent(h5Url);
  1436. // 构建 web-view 页面路径
  1437. let webViewPath = `/common-page/pages/web-view/index?path=${encodedUrl}&storeId=${storeId}`;
  1438. console.log('📄 web-view 页面路径:', webViewPath.substring(0, 100) + '...');
  1439. console.log('===========================================');
  1440. wx.redirectTo({
  1441. url: webViewPath,
  1442. success: () => {
  1443. console.log('✅ 跳转到门店首页展示方案成功');
  1444. },
  1445. fail: (err) => {
  1446. console.error('❌ 跳转失败:', err);
  1447. // 如果 redirectTo 失败,尝试 reLaunch
  1448. wx.reLaunch({
  1449. url: webViewPath,
  1450. fail: (err2) => {
  1451. console.error('❌ reLaunch 也失败:', err2);
  1452. }
  1453. });
  1454. }
  1455. });
  1456. } catch (error) {
  1457. console.error('❌ 跳转到门店首页展示方案失败:', error);
  1458. }
  1459. },
  1460. /**
  1461. * 记录扫码统计信息
  1462. * @param {Object} params - 扫码参数对象
  1463. * @param {string} params.storeId - 店铺 ID
  1464. * @param {string} params.sourceType - 来源类型 (employee/owner/partner/promoter/store)
  1465. * @param {string} params.sourceId - 来源 ID
  1466. * @param {string} params.ownerId - 老板 ID
  1467. * @param {string} params.employeeId - 员工 ID
  1468. * @param {string} params.partnerId - 合作伙伴 ID
  1469. * @param {string} params.userId - 推广员 ID
  1470. * @param {string} params.productId - 产品 ID
  1471. * @param {string} params.scanCount - 扫码次数
  1472. */
  1473. async recordScanStatistics(params) {
  1474. try {
  1475. console.log('📝 ===========================================');
  1476. console.log('📝 [扫码统计] 开始记录扫码统计');
  1477. console.log('📝 ===========================================');
  1478. const {
  1479. storeId,
  1480. sourceType,
  1481. sourceId,
  1482. ownerId,
  1483. employeeId,
  1484. partnerId,
  1485. userId,
  1486. productId,
  1487. scanCount
  1488. } = params;
  1489. console.log('📋 [统计参数] 接收到的参数:');
  1490. console.log(' - storeId:', storeId);
  1491. console.log(' - sourceType:', sourceType);
  1492. console.log(' - sourceId:', sourceId || '无');
  1493. console.log(' - ownerId:', ownerId || '无');
  1494. console.log(' - employeeId:', employeeId || '无');
  1495. console.log(' - partnerId:', partnerId || '无');
  1496. console.log(' - userId:', userId || '无');
  1497. console.log(' - productId:', productId || '无');
  1498. console.log(' - scanCount:', scanCount || '0');
  1499. // 检查用户是否已登录
  1500. const currentUser = Parse.User.current();
  1501. if (!currentUser) {
  1502. console.log('⚠️ [未登录] 用户未登录,不记录扫码统计');
  1503. return;
  1504. }
  1505. console.log('👤 [用户信息] 当前登录用户:');
  1506. console.log(' - 用户ID:', currentUser.id);
  1507. console.log(' - 手机号:', currentUser.get('mobile') || '无');
  1508. // 如果没有来源信息,不记录统计
  1509. if (!sourceId && !ownerId && !employeeId && !partnerId && !userId) {
  1510. console.log('ℹ️ [跳过] 无来源信息,跳过统计记录');
  1511. return;
  1512. }
  1513. console.log('✅ [开始记录] 用户已登录且有来源信息,开始记录');
  1514. // 创建扫码记录
  1515. const ScanRecord = Parse.Object.extend('ScanRecord');
  1516. const record = new ScanRecord();
  1517. record.set('company', {
  1518. __type: 'Pointer',
  1519. className: 'Company',
  1520. objectId: getApp().globalData.company
  1521. });
  1522. record.set('storeId', storeId);
  1523. record.set('sourceType', sourceType || 'unknown');
  1524. record.set('scanCount', parseInt(scanCount) || 0);
  1525. record.set('scanTime', new Date());
  1526. console.log('📦 [记录内容] 准备保存到 ScanRecord 表:');
  1527. console.log(' - company:', getApp().globalData.company);
  1528. console.log(' - storeId:', storeId);
  1529. console.log(' - sourceType:', sourceType || 'unknown');
  1530. console.log(' - scanCount:', parseInt(scanCount) || 0);
  1531. console.log(' - scanTime:', new Date().toISOString());
  1532. // 根据来源类型设置对应的ID
  1533. if (ownerId) {
  1534. record.set('ownerId', ownerId);
  1535. console.log(' - ownerId:', ownerId);
  1536. }
  1537. if (employeeId) {
  1538. record.set('employeeId', employeeId);
  1539. console.log(' - employeeId:', employeeId);
  1540. }
  1541. if (partnerId) {
  1542. record.set('partnerId', partnerId);
  1543. console.log(' - partnerId:', partnerId);
  1544. }
  1545. if (userId) {
  1546. record.set('userId', userId);
  1547. console.log(' - userId:', userId);
  1548. }
  1549. if (productId) {
  1550. record.set('productId', productId);
  1551. console.log(' - productId:', productId);
  1552. }
  1553. // 记录扫码用户
  1554. record.set('user', {
  1555. __type: 'Pointer',
  1556. className: '_User',
  1557. objectId: currentUser.id
  1558. });
  1559. console.log(' - user:', currentUser.id);
  1560. await record.save();
  1561. console.log('✅ [保存成功] 扫码记录已保存到数据库');
  1562. console.log(' - 记录ID:', record.id);
  1563. // 如果存在异业合作伙伴ID,处理异业绑定和扫码次数
  1564. if (partnerId) {
  1565. try {
  1566. console.log('🤝 ===========================================');
  1567. console.log('🤝 [异业处理] 开始处理异业扫码逻辑');
  1568. console.log('🤝 用户ID:', currentUser.id);
  1569. console.log('🤝 异业ID:', partnerId);
  1570. console.log('🤝 门店ID:', storeId);
  1571. console.log('🤝 ===========================================');
  1572. // 检查用户是否已经绑定了异业合作伙伴
  1573. const userBoundPartner = currentUser.get('Partner');
  1574. const userBoundPartnerId = userBoundPartner ? userBoundPartner.id : null;
  1575. console.log('🔍 [绑定检查] 用户已绑定的异业ID:', userBoundPartnerId || '无');
  1576. // 规则1:如果用户已绑定其他异业,不处理当前异业的扫码
  1577. if (userBoundPartnerId && userBoundPartnerId !== partnerId) {
  1578. console.log('⚠️ [规则1] 用户已绑定其他异业,不处理当前异业的扫码');
  1579. console.log(' - 已绑定异业:', userBoundPartnerId);
  1580. console.log(' - 当前扫码异业:', partnerId);
  1581. console.log('🤝 ===========================================');
  1582. return;
  1583. }
  1584. // 规则2:检查该用户在该门店是否已经扫过该异业的码
  1585. console.log('🔍 [重复检查] 检查是否已扫过该异业的码...');
  1586. const ScanRecord = Parse.Object.extend('ScanRecord');
  1587. const scanQuery = new Parse.Query(ScanRecord);
  1588. scanQuery.equalTo('user', {
  1589. __type: 'Pointer',
  1590. className: '_User',
  1591. objectId: currentUser.id
  1592. });
  1593. scanQuery.equalTo('partnerId', partnerId);
  1594. scanQuery.equalTo('storeId', storeId);
  1595. const existingScan = await scanQuery.first();
  1596. if (existingScan) {
  1597. console.log('⚠️ [规则2] 该用户在该门店已扫过该异业的码,不重复计数');
  1598. console.log(' - 首次扫码时间:', existingScan.get('scanTime'));
  1599. console.log(' - 记录ID:', existingScan.id);
  1600. console.log('🤝 ===========================================');
  1601. return;
  1602. }
  1603. console.log('✅ [首次扫码] 该用户在该门店首次扫该异业码');
  1604. // 规则3:如果用户还没有绑定异业,绑定当前异业
  1605. if (!userBoundPartnerId) {
  1606. console.log('🔗 [规则3] 用户首次扫异业码,开始绑定异业合作伙伴');
  1607. console.log(' - 异业ID:', partnerId);
  1608. currentUser.set('Partner', {
  1609. __type: 'Pointer',
  1610. className: 'Partner',
  1611. objectId: partnerId
  1612. });
  1613. await currentUser.save();
  1614. console.log('✅ [绑定成功] 异业绑定成功');
  1615. } else {
  1616. console.log('ℹ️ [已绑定] 用户已绑定该异业,无需重复绑定');
  1617. }
  1618. // 规则4:为该异业在该门店增加扫码次数
  1619. console.log('📊 [规则4] 开始更新异业扫码次数...');
  1620. const Partner = Parse.Object.extend('Partner');
  1621. const partnerQuery = new Parse.Query(Partner);
  1622. const partnerObj = await partnerQuery.get(partnerId);
  1623. if (partnerObj) {
  1624. console.log('📋 [异业信息] 找到异业合作伙伴记录');
  1625. console.log(' - 异业ID:', partnerObj.id);
  1626. console.log(' - 异业名称:', partnerObj.get('name') || '未设置');
  1627. // 获取或初始化门店扫码次数映射
  1628. let storeScans = partnerObj.get('storeScans') || {};
  1629. const oldCount = storeScans[storeId] || 0;
  1630. // 为该门店的扫码次数 +1
  1631. storeScans[storeId] = oldCount + 1;
  1632. partnerObj.set('storeScans', storeScans);
  1633. // 同时更新总扫码次数
  1634. const oldTotalCount = partnerObj.get('scanCount') || 0;
  1635. partnerObj.increment('scanCount', 1);
  1636. await partnerObj.save();
  1637. console.log('✅ [更新成功] 异业扫码次数已更新');
  1638. console.log(' - 门店ID:', storeId);
  1639. console.log(' - 该门店扫码次数: %d → %d', oldCount, storeScans[storeId]);
  1640. console.log(' - 总扫码次数: %d → %d', oldTotalCount, oldTotalCount + 1);
  1641. } else {
  1642. console.error('❌ [错误] 未找到异业合作伙伴记录');
  1643. }
  1644. console.log('🤝 ===========================================');
  1645. } catch (e) {
  1646. console.error('❌ [异业处理失败] 处理异业合作伙伴绑定失败:', e);
  1647. console.error(' - 错误信息:', e.message);
  1648. console.error(' - 错误堆栈:', e.stack);
  1649. console.log('🤝 ===========================================');
  1650. }
  1651. }
  1652. } catch (error) {
  1653. console.warn('⚠️ 保存扫码记录失败:', error);
  1654. // 不影响主流程
  1655. }
  1656. }
  1657. });
  1658. function normalizeCnMobile(value) {
  1659. if (value === null || value === undefined) return '';
  1660. const digits = String(value).replace(/\D/g, '');
  1661. if (!digits) return '';
  1662. const last11 = digits.length >= 11 ? digits.slice(-11) : digits;
  1663. if (/^1\d{10}$/.test(last11)) return last11;
  1664. if (/^1\d{10}$/.test(digits)) return digits;
  1665. return '';
  1666. }
  1667. function addCnMobileToSet(set, value) {
  1668. const m = normalizeCnMobile(value);
  1669. if (m) set.add(m);
  1670. }
  1671. function addCnMobilesFromValue(set, value) {
  1672. if (!value) return;
  1673. if (Array.isArray(value)) {
  1674. value.forEach((v) => addCnMobileToSet(set, v));
  1675. return;
  1676. }
  1677. addCnMobileToSet(set, value);
  1678. }
  1679. function addCnMobilesFromParseObject(set, parseObj, keys) {
  1680. if (!parseObj || typeof parseObj.get !== 'function') return;
  1681. keys.forEach((k) => {
  1682. try {
  1683. const v = parseObj.get(k);
  1684. addCnMobilesFromValue(set, v);
  1685. } catch (e) {}
  1686. });
  1687. }
  1688. async function addCnMobileFromUserPointer(set, Parse, pointer) {
  1689. if (!pointer) return;
  1690. if (typeof pointer.get === 'function') {
  1691. addCnMobileToSet(set, pointer.get('mobile'));
  1692. addCnMobileToSet(set, pointer.get('phone'));
  1693. }
  1694. const pointerId = pointer && pointer.id ? pointer.id : null;
  1695. if (!pointerId) return;
  1696. try {
  1697. const q = new Parse.Query('_User');
  1698. q.select('mobile');
  1699. const u = await q.get(pointerId);
  1700. if (u) addCnMobileToSet(set, u.get('mobile'));
  1701. } catch (e) {}
  1702. }
  1703. function collectUserMobiles(currentUser) {
  1704. const set = new Set();
  1705. addCnMobileToSet(set, currentUser && typeof currentUser.get === 'function' ? currentUser.get('mobile') : null);
  1706. addCnMobileToSet(set, currentUser && typeof currentUser.get === 'function' ? currentUser.get('phone') : null);
  1707. try {
  1708. const storageMobile = wx.getStorageSync('user_mobile');
  1709. addCnMobileToSet(set, storageMobile);
  1710. } catch (e) {}
  1711. return Array.from(set);
  1712. }
  1713. function setSample(set, limit) {
  1714. const arr = [];
  1715. let i = 0;
  1716. for (const v of set) {
  1717. arr.push(v);
  1718. i++;
  1719. if (i >= (limit || 5)) break;
  1720. }
  1721. return arr;
  1722. }
  1723. function intersectArrSet(arr, set) {
  1724. return arr.filter((v) => set.has(v));
  1725. }
  1726. async function collectPartnerPhonesForStore(Parse, storeId) {
  1727. const phones = new Set();
  1728. const mobiles = new Set();
  1729. try {
  1730. if (!storeId) return { phones, mobiles };
  1731. const currentUser = Parse.User.current();
  1732. const sessionToken = currentUser && typeof currentUser.getSessionToken === 'function'
  1733. ? currentUser.getSessionToken()
  1734. : null;
  1735. const requestOptions = sessionToken ? { sessionToken } : undefined;
  1736. const storePointer = new Parse.Object('ShopStore'); storePointer.id = storeId;
  1737. const q = new Parse.Query('Partner');
  1738. q.equalTo('store', storePointer);
  1739. q.limit(200);
  1740. q.select('phone', 'mobile');
  1741. const list = await q.find(requestOptions);
  1742. if (list && list.length) {
  1743. list.forEach((p) => {
  1744. addCnMobileToSet(phones, p.get('phone'));
  1745. addCnMobileToSet(mobiles, p.get('mobile'));
  1746. });
  1747. }
  1748. } catch (e) {
  1749. console.warn('⚠️ [traffic] Partner 查询失败:', e?.code, e?.message || e);
  1750. }
  1751. return { phones, mobiles };
  1752. }
  1753. async function collectTrafficWhitelistMobiles(Parse, storeId) {
  1754. const set = new Set();
  1755. const mobileKeys = [
  1756. 'mobile',
  1757. 'phone'
  1758. ];
  1759. const userPointerKeys = [
  1760. 'owner',
  1761. 'boss',
  1762. 'admin',
  1763. 'manager',
  1764. 'user',
  1765. 'ownerUser',
  1766. 'bossUser',
  1767. 'adminUser',
  1768. 'managerUser',
  1769. 'createdBy'
  1770. ];
  1771. try {
  1772. if (storeId) {
  1773. const q = new Parse.Query('ShopStore');
  1774. q.equalTo('objectId', storeId);
  1775. q.select(...mobileKeys, ...userPointerKeys);
  1776. userPointerKeys.forEach((k) => q.include(k));
  1777. const storeObj = await q.first();
  1778. if (storeObj) {
  1779. addCnMobilesFromParseObject(set, storeObj, mobileKeys);
  1780. for (const k of userPointerKeys) {
  1781. await addCnMobileFromUserPointer(set, Parse, storeObj.get(k));
  1782. }
  1783. }
  1784. }
  1785. } catch (e) {}
  1786. const userStaffPhoneKeys = ['mobile', 'phone'];
  1787. try {
  1788. if (storeId) {
  1789. const flags = getApp().globalData || {};
  1790. const disabled = !!flags.disableUserStaffQuery || wx.getStorageSync('disableUserStaffQuery') === true;
  1791. if (disabled) {
  1792. console.log('⚠️ [traffic] userStaff 查询已禁用(检测到类不存在/无权限)');
  1793. } else {
  1794. const currentUser = Parse.User.current();
  1795. const sessionToken = currentUser && typeof currentUser.getSessionToken === 'function'
  1796. ? currentUser.getSessionToken()
  1797. : null;
  1798. const requestOptions = sessionToken ? { sessionToken } : undefined;
  1799. const companyId = getApp().globalData.company;
  1800. const companyPointer = new Parse.Object('Company'); companyPointer.id = companyId;
  1801. const q = new Parse.Query('UserStaff');
  1802. q.equalTo('company', companyPointer);
  1803. q.limit(200);
  1804. q.select(...userStaffPhoneKeys);
  1805. const list = await q.find(requestOptions);
  1806. if (list && list.length) {
  1807. list.forEach((staff) => addCnMobilesFromParseObject(set, staff, userStaffPhoneKeys));
  1808. }
  1809. }
  1810. }
  1811. } catch (e) {
  1812. console.warn('⚠️ [traffic] userStaff 查询失败:', e?.code, e?.message || e);
  1813. try {
  1814. const msg = (e?.message || '').toLowerCase();
  1815. if (
  1816. e?.code === 119 ||
  1817. msg.includes('non-existent class') ||
  1818. msg.includes('class: userstaff') ||
  1819. msg.includes('class: userstaff') ||
  1820. msg.includes('class: userstaff')
  1821. ) {
  1822. getApp().globalData.disableUserStaffQuery = true;
  1823. wx.setStorageSync('disableUserStaffQuery', true);
  1824. console.warn('⚠️ [traffic] 已禁用后续 userStaff 查询(类不存在或无权限)');
  1825. }
  1826. } catch (_) {}
  1827. }
  1828. const partnerPhoneKeys = ['mobile', 'phone'];
  1829. try {
  1830. if (storeId) {
  1831. const currentUser = Parse.User.current();
  1832. const sessionToken = currentUser && typeof currentUser.getSessionToken === 'function'
  1833. ? currentUser.getSessionToken()
  1834. : null;
  1835. const requestOptions = sessionToken ? { sessionToken } : undefined;
  1836. const storePointer = new Parse.Object('ShopStore'); storePointer.id = storeId;
  1837. const q = new Parse.Query('Partner');
  1838. q.equalTo('store', storePointer);
  1839. q.limit(200);
  1840. q.select(...partnerPhoneKeys);
  1841. const list = await q.find(requestOptions);
  1842. if (list && list.length) {
  1843. list.forEach((p) => addCnMobilesFromParseObject(set, p, partnerPhoneKeys));
  1844. }
  1845. }
  1846. } catch (e) {
  1847. console.warn('⚠️ [traffic] Partner 查询失败:', e?.code, e?.message || e);
  1848. }
  1849. return set;
  1850. }
  1851. async function isTrafficExemptForStore(Parse, storeId, currentUser) {
  1852. try {
  1853. if (!currentUser) return false;
  1854. if (currentUser.get('trafficExempt') === true) return true;
  1855. const userNumbers = collectUserMobiles(currentUser);
  1856. const whitelist = await collectTrafficWhitelistMobiles(Parse, storeId);
  1857. const partnerSets = await collectPartnerPhonesForStore(Parse, storeId);
  1858. console.log('🧾 [traffic] 白名单判定开始:', {
  1859. storeId,
  1860. userId: currentUser.id,
  1861. userNumbers
  1862. });
  1863. console.log('🧾 [traffic] Partner 集合:', {
  1864. phoneCount: partnerSets.phones.size,
  1865. phoneSample: setSample(partnerSets.phones, 5),
  1866. mobileCount: partnerSets.mobiles.size,
  1867. mobileSample: setSample(partnerSets.mobiles, 5)
  1868. });
  1869. const partnerPhoneMatches = intersectArrSet(userNumbers, partnerSets.phones);
  1870. if (partnerPhoneMatches.length) {
  1871. console.log('🧾 [traffic] 命中 Partner.phone:', {
  1872. matches: partnerPhoneMatches
  1873. });
  1874. return true;
  1875. }
  1876. const partnerMobileMatches = intersectArrSet(userNumbers, partnerSets.mobiles);
  1877. if (partnerMobileMatches.length) {
  1878. console.log('🧾 [traffic] 命中 Partner.mobile:', {
  1879. matches: partnerMobileMatches
  1880. });
  1881. return true;
  1882. }
  1883. console.log('🧾 [traffic] 聚合白名单:', {
  1884. size: whitelist.size,
  1885. sample: setSample(whitelist, 10)
  1886. });
  1887. const whitelistMatches = intersectArrSet(userNumbers, whitelist);
  1888. if (whitelistMatches.length) {
  1889. console.log('🧾 [traffic] 命中聚合白名单:', {
  1890. matches: whitelistMatches
  1891. });
  1892. return true;
  1893. }
  1894. console.log('🧾 [traffic] 白名单未命中');
  1895. return false;
  1896. } catch (e) {
  1897. console.warn('⚠️ [traffic] 白名单判定失败,按非白名单处理:', e?.message || e);
  1898. return false;
  1899. }
  1900. }
  1901. async function decrementStoreTrafficImpl(Parse, storeId) {
  1902. const currentUser = Parse.User.current();
  1903. const sessionToken = currentUser && typeof currentUser.getSessionToken === 'function'
  1904. ? currentUser.getSessionToken()
  1905. : null;
  1906. const requestOptions = sessionToken ? { sessionToken } : undefined;
  1907. console.log('🧾 [traffic] 开始扣减 ShopStore.traffic');
  1908. console.log('🧾 [traffic] storeId:', storeId);
  1909. console.log('🧾 [traffic] userId:', currentUser ? currentUser.id : '无用户');
  1910. console.log('🧾 [traffic] hasSessionToken:', sessionToken ? 'true' : 'false');
  1911. const storeQuery = new Parse.Query('ShopStore');
  1912. const store = await storeQuery.get(storeId, requestOptions);
  1913. if (!store) {
  1914. console.error('🧾 [traffic] 未找到门店记录, storeId:', storeId);
  1915. throw new Error('未找到门店记录');
  1916. }
  1917. const trafficValue = store.get('traffic');
  1918. const trafficNumber = Number(trafficValue);
  1919. const traffic = Number.isFinite(trafficNumber) ? trafficNumber : 0;
  1920. const next = Math.max(traffic - 1, 0);
  1921. console.log('🧾 [traffic] before trafficValue:', trafficValue);
  1922. console.log('🧾 [traffic] parsed traffic:', traffic);
  1923. console.log('🧾 [traffic] next:', next);
  1924. if (next !== traffic || trafficValue === undefined) {
  1925. store.set('traffic', next);
  1926. const saved = await store.save(null, requestOptions);
  1927. console.log('🧾 [traffic] 保存成功 storeObjectId:', saved && saved.id ? saved.id : store.id);
  1928. }
  1929. console.log('🧾 [traffic] 结束扣减 ShopStore.traffic');
  1930. }
  1931. async function isInternalVisit(Parse, storeId, currentUser, ids) {
  1932. try {
  1933. if (!currentUser) return false;
  1934. const { ownerId, employeeId, partnerId } = ids || {};
  1935. const roles = Array.isArray(currentUser.get('roles')) ? currentUser.get('roles') : [];
  1936. const roleSet = new Set(roles.map(r => String(r).toLowerCase()));
  1937. const isOwnerRole = roleSet.has('owner') || roleSet.has('boss');
  1938. const isEmployeeRole = roleSet.has('employee') || roleSet.has('staff') || roleSet.has('sales');
  1939. const isAdminStaffRole = roleSet.has('userstaff') || roleSet.has('admin_staff') || roleSet.has('manager_staff');
  1940. const isPartnerRole = roleSet.has('partner') || roleSet.has('partner_admin') || roleSet.has('partner_staff');
  1941. if (partnerId && isPartnerRole) return true;
  1942. if (employeeId && isEmployeeRole) return true;
  1943. if (isAdminStaffRole) return true;
  1944. const storeQuery = new Parse.Query('ShopStore');
  1945. const store = await storeQuery.get(storeId);
  1946. const storeOwner = store ? store.get('user') : null;
  1947. if (storeOwner && storeOwner.id === currentUser.id) return true;
  1948. if (ownerId && isOwnerRole) return true;
  1949. if (employeeId && currentUser.id === employeeId) return true;
  1950. if (ownerId && currentUser.id === ownerId) return true;
  1951. return false;
  1952. } catch (e) {
  1953. return false;
  1954. }
  1955. }