index.js 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831
  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. wx.login({
  25. success: function (res) {
  26. if (res.code) {
  27. console.log(res);
  28. // wx.request({
  29. // url: "https://server.fmode.cn/api/wxapp/auth_wxapp",
  30. // data: {
  31. // c: getApp().globalData.company,
  32. // code: res.code,
  33. // },
  34. // async success(res) {
  35. // wx.setStorageSync("userInfo", res.data);
  36. // resolve(res)
  37. // },
  38. // });
  39. }
  40. },
  41. fail: function (err) {
  42. wx.showToast({
  43. title: '服务器繁忙,请稍后重试',
  44. })
  45. }
  46. });
  47. wx.setStorageSync("invite", null);
  48. // 处理扫码链接(options.q 包含完整的URL或activityId)
  49. if (options.q) {
  50. let str = decodeURIComponent(options.q); // 扫描二维码获得的跳转链接
  51. // 兼容一些环境中把 & 编码成 & 的情况,防止参数解析错误(如 scanCount=0&storeId=...)
  52. if (str.indexOf('&') !== -1) {
  53. console.log('🔧 检测到 URL 中包含 &,自动还原为 &');
  54. str = str.replace(/&/g, '&');
  55. }
  56. // 检查是否是员工邀请链接(app.fmode.cn/dev/pobingfeng/manager/staff?invite=xxx)
  57. if (str.includes('app.fmode.cn/dev/pobingfeng/manager/staff')) {
  58. let obj = this.getParaName(str);
  59. if (obj && obj.invite) {
  60. wx.setStorageSync("invite", obj.invite);
  61. console.log('✅ 检测到员工邀请链接,invite:', obj.invite);
  62. }
  63. // 员工邀请链接不需要跳转到店铺,直接返回
  64. this.setData({ options: options });
  65. plugin.init(config, wx.getStorageSync('invite'));
  66. return;
  67. }
  68. // 检查是否是活动海报二维码(activityId作为qrCode参数值)
  69. // 如果str不包含http/https,且看起来像是一个ID,则可能是activityId
  70. if (!str.includes('http://') && !str.includes('https://') && !str.includes('?')) {
  71. // 可能是活动ID,尝试作为activityId处理
  72. if (str && str.length > 0 && !isNaN(str)) {
  73. options.activityId = str;
  74. console.log('✅ 检测到活动海报二维码,activityId:', str);
  75. // 活动二维码需要特殊处理,先保存activityId
  76. wx.setStorageSync("activityId", str);
  77. }
  78. }
  79. // 处理店铺相关的二维码链接(pwa.fmode.cn/gomini/pmd/)
  80. // 从完整URL中提取参数
  81. if (str.includes('pwa.fmode.cn/gomini/pmd') || str.includes('?')) {
  82. let obj = this.getParaName(str);
  83. if (obj && obj.invite) {
  84. wx.setStorageSync("invite", obj.invite);
  85. }
  86. // 将所有参数合并到 options 中
  87. obj && Object.keys(obj).forEach(key=> options[key] = obj[key]);
  88. }
  89. }
  90. // 兼容从编译参数或页面直达传入的 storeId
  91. if (options && options.scene && !options.storeId) {
  92. try {
  93. const sceneStr = decodeURIComponent(options.scene)
  94. if (sceneStr) {
  95. const pairs = sceneStr.split('&')
  96. for (const p of pairs) {
  97. const [k, v] = p.split('=')
  98. if (k === 'storeId' && v) {
  99. options.storeId = v
  100. break
  101. }
  102. }
  103. }
  104. } catch (e) {
  105. console.warn('解析 scene 失败:', e)
  106. }
  107. }
  108. this.setData({
  109. options: options
  110. })
  111. let {
  112. time,
  113. dramaId,
  114. roomId,
  115. orderId,
  116. shopId,
  117. invite,
  118. activityId,
  119. company,
  120. inviteHost,
  121. storeId
  122. } = options
  123. time && wx.setStorageSync("time", time);
  124. dramaId && wx.setStorageSync("dramaId", dramaId);
  125. roomId && wx.setStorageSync("roomId", roomId);
  126. orderId && wx.setStorageSync("orderId", orderId);
  127. shopId && wx.setStorageSync("shopId", shopId);
  128. invite && wx.setStorageSync("invite", invite);
  129. activityId && wx.setStorageSync("activityId", activityId);
  130. inviteHost && wx.setStorageSync("inviteHost", true);
  131. if (storeId) {
  132. wx.setStorageSync('storeId', storeId)
  133. getApp().globalData.storeId = storeId
  134. console.log('✅ 入口页已设置店铺 ID:', storeId)
  135. }
  136. if (company) getApp().globalData.toCompany = true;
  137. // 检查是否是扫码进入(需要统计扫码次数)
  138. this.checkAndHandleScan(options);
  139. plugin.init(config, wx.getStorageSync('invite'))
  140. },
  141. /**
  142. * 生命周期函数--监听页面初次渲染完成
  143. */
  144. onReady: async function () { },
  145. /**
  146. * 生命周期函数--监听页面显示
  147. */
  148. onShow: async function () {
  149. await this.review()
  150. },
  151. async review(force){
  152. try {
  153. let options = this.data.options
  154. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath
  155. if (options) {
  156. let objArr = Object.keys(options)
  157. if (objArr && objArr.length > 0) {
  158. let parms = '?'
  159. objArr.forEach((o, index) => {
  160. if (index > 0) {
  161. parms += '&' + o + '=' + options[o]
  162. } else {
  163. parms += o + '=' + options[o]
  164. }
  165. })
  166. url += parms
  167. }
  168. }
  169. let currentUser = Parse.User.current()
  170. console.log(Parse.User.current())
  171. // 修改:不强制登录,允许游客访问
  172. if (!currentUser || force) {
  173. // 尝试静默登录(不强制授权)
  174. let r = await checkAuth(false) // 改为 false,不强制授权
  175. console.log(r);
  176. // 即使登录失败,也允许继续访问
  177. if(!r) {
  178. console.log('⚠️ 用户未登录或拒绝授权,允许游客访问');
  179. // 不要 return,继续执行后面的跳转逻辑
  180. }
  181. } else {
  182. this.updateUser(currentUser.id)
  183. }
  184. getApp().Parse = Parse
  185. getApp().checkAuth = checkAuth
  186. if (!await this.getCompanyServerExpire(url)) {
  187. return
  188. }
  189. // 检查是否需要跳转到活动页面
  190. if (wx.getStorageSync('need_activity_redirect') === true) {
  191. await this.redirectToActivityPage();
  192. return;
  193. }
  194. // 检查是否需要跳转到扫码统计页面
  195. if (this.shouldRedirectToScanPage()) {
  196. await this.redirectToScanPage();
  197. return;
  198. }
  199. console.log('✅ 准备跳转到:', url);
  200. wx.redirectTo({
  201. url: url,
  202. success: () => {
  203. console.log('✅ redirectTo 跳转成功');
  204. },
  205. fail: (err) => {
  206. console.error('❌ redirectTo 失败:', err);
  207. console.log('⚠️ 尝试使用 reLaunch');
  208. // 降级:尝试使用 reLaunch
  209. wx.reLaunch({
  210. url: url,
  211. success: () => {
  212. console.log('✅ reLaunch 跳转成功');
  213. },
  214. fail: (err2) => {
  215. console.error('❌ reLaunch 也失败:', err2);
  216. // 显示错误提示
  217. this.setData({ loading: false });
  218. wx.showModal({
  219. title: '温馨提示',
  220. content: '页面加载失败,请检查网络后重试。',
  221. showCancel: true,
  222. cancelText: '退出',
  223. confirmText: '重试',
  224. success: (result) => {
  225. if (result.confirm) {
  226. this.review(true);
  227. } else {
  228. wx.exitMiniProgram();
  229. }
  230. }
  231. });
  232. }
  233. });
  234. }
  235. });
  236. }
  237. catch (err) {
  238. console.log(err);
  239. /* 登录身份信息到期,重新登陆 */
  240. if((err?.message.indexOf('Session token is expired') != -1 || err?.message.indexOf('Invalid session token') != -1) && !force){
  241. let invite = wx.getStorageSync('invite')
  242. wx.clearStorageSync()
  243. invite && wx.setStorageSync('invite', invite)
  244. /* 强制重新登录 */
  245. this.review(true)
  246. return
  247. }
  248. // 如果出错,尝试继续跳转到主页(游客模式)
  249. console.log('⚠️ 登录出错,尝试游客模式访问');
  250. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath;
  251. if (this.data.options) {
  252. let objArr = Object.keys(this.data.options)
  253. if (objArr && objArr.length > 0) {
  254. let parms = '?'
  255. objArr.forEach((o, index) => {
  256. if (index > 0) {
  257. parms += '&' + o + '=' + this.data.options[o]
  258. } else {
  259. parms += o + '=' + this.data.options[o]
  260. }
  261. })
  262. url += parms
  263. }
  264. }
  265. wx.redirectTo({
  266. url: url,
  267. fail: () => {
  268. // 如果跳转失败,显示错误提示
  269. this.setData({
  270. loading:false
  271. })
  272. wx.showModal({
  273. title: '温馨提示',
  274. content: '页面加载失败,请检查网络后重试。',
  275. showCancel: true,
  276. cancelText: '退出',
  277. confirmText: '重试',
  278. success: (result) => {
  279. if (result.confirm) {
  280. // 用户选择重试
  281. this.review(true)
  282. } else {
  283. // 用户选择退出
  284. wx.exitMiniProgram()
  285. }
  286. },
  287. });
  288. }
  289. });
  290. }
  291. },
  292. async updateUser(id) {
  293. let User = new Parse.Query('_User')
  294. let user = await User.get(id)
  295. let invite = wx.getStorageSync('invite')
  296. //查询邀请人user
  297. let query = new Parse.Query("_User")
  298. query.equalTo('objectId', invite)
  299. let result = await query.first()
  300. if (result && result.id && result.get("invite")?.id == user.id) {
  301. console.error('邀请人不能是自己的下级')
  302. return
  303. }
  304. if (invite && !user.get('invite') && user.id != invite && !user.get('agentLevel')) {
  305. console.log('上下级绑定成功');
  306. user.set('invite', {
  307. __type: "Pointer",
  308. className: "_User",
  309. objectId: invite
  310. })
  311. user.set('agent', {
  312. __type: "Pointer",
  313. className: "_User",
  314. objectId: invite
  315. })
  316. await Parse.Cloud.run('user_save', {
  317. userJson: user.toJSON()
  318. })
  319. }
  320. },
  321. async getCompanyServerExpire(url) {
  322. let query = new Parse.Query('Company')
  323. query.equalTo('objectId', getApp().globalData.company)
  324. query.select('expireDate', 'expireMap')
  325. let com = await query.first()
  326. if (com?.id && com?.get('expireDate')) {
  327. let now = + new Date()
  328. let expireTime = + new Date(com?.get('expireDate'))
  329. if (com?.get('expireMap') && com.get('expireMap')[getApp().globalData.appid]) {
  330. expireTime = + new Date(com.get('expireMap')[getApp().globalData.appid])
  331. }
  332. if (now >= expireTime) {
  333. console.log('服务器到期');
  334. wx.reLaunch({
  335. url: `common-page/pages/loading/index?url=${url}`,
  336. });
  337. return
  338. }
  339. }
  340. return true
  341. },
  342. onUnload: function () {
  343. wx.setStorageSync("active", 0);
  344. },
  345. getParaName(url) {
  346. if (!url || url.indexOf('?') == -1) {
  347. return
  348. }
  349. // 兼容 URL 中 & 的情况,先统一还原为 &
  350. if (url.indexOf('&') !== -1) {
  351. url = url.replace(/&/g, '&');
  352. }
  353. // 提取查询参数部分(去除可能的 hash 部分)
  354. let queryString = url.split('?')[1];
  355. // 如果包含 #,只取 # 之前的部分
  356. if (queryString.indexOf('#') !== -1) {
  357. queryString = queryString.split('#')[0];
  358. }
  359. return this.setObject(queryString) //封装成对象
  360. },
  361. setObject(paraArr) {
  362. let obj = {}
  363. let arr1 = paraArr.split('&')
  364. arr1.forEach(item => {
  365. let str = item.split('=')
  366. let key = str[0]
  367. let val = str[1]
  368. obj[key] = val
  369. })
  370. return obj
  371. },
  372. /**
  373. * 检查并处理扫码参数
  374. * 支持所有类型的二维码:
  375. * 1. 推广员二维码: ?scanCount=0&storeId=xxx&userId=xxx
  376. * 2. 产品二维码: ?scanCount=0&storeId=xxx&productId=xxx
  377. * 3. 活动海报二维码: activityId作为qrCode参数值
  378. * 4. 异业合作伙伴二维码(移动端 / PC端统一): ?scanCount=xxx&storeId=xxx&partnerId=xxx
  379. * 5. 员工邀请二维码(移动端): ?scanCount=0&storeId=xxx&employeeId=xxx
  380. * 6. 我的二维码(老板): ?scanCount=0&storeId=xxx&ownerId=xxx
  381. * 7. 我的二维码(员工): ?scanCount=0&storeId=xxx&employeeId=xxx
  382. */
  383. checkAndHandleScan(options) {
  384. const {
  385. scanCount,
  386. ownerId,
  387. employeeId,
  388. partnerId,
  389. storeId,
  390. productId,
  391. userId, // 推广员二维码使用userId
  392. activityId
  393. } = options;
  394. // 处理活动海报二维码(activityId作为qrCode参数值)
  395. if (activityId && !storeId) {
  396. console.log('===========================================');
  397. console.log('======= 检测到活动海报二维码 =======');
  398. console.log('活动ID (activityId):', activityId);
  399. console.log('===========================================');
  400. // 活动二维码需要跳转到活动页面,不需要跳转到店铺
  401. wx.setStorageSync('scan_activityId', activityId);
  402. wx.setStorageSync('need_activity_redirect', true);
  403. return;
  404. }
  405. // 如果存在 storeId,说明是扫码进入具体门店(包括异业分享)
  406. if (storeId) {
  407. console.log('===========================================');
  408. console.log('======= 检测到扫码参数 =======');
  409. console.log('店铺ID (storeId):', storeId);
  410. console.log('扫码次数 (scanCount):', scanCount || '0');
  411. // 确定来源类型
  412. let sourceType = 'unknown';
  413. let sourceId = null;
  414. if (employeeId) {
  415. sourceType = 'employee';
  416. sourceId = employeeId;
  417. console.log('来源类型: 员工展业');
  418. console.log('员工ID (employeeId):', employeeId);
  419. } else if (userId) {
  420. // 推广员二维码使用userId
  421. sourceType = 'promoter';
  422. sourceId = userId;
  423. console.log('来源类型: 推广员展业');
  424. console.log('推广员ID (userId):', userId);
  425. } else if (ownerId) {
  426. sourceType = 'owner';
  427. sourceId = ownerId;
  428. console.log('来源类型: 老板展业');
  429. console.log('老板ID (ownerId):', ownerId);
  430. } else if (partnerId) {
  431. sourceType = 'partner';
  432. sourceId = partnerId;
  433. console.log('来源类型: 异业合作伙伴');
  434. console.log('合作伙伴ID (partnerId):', partnerId);
  435. } else {
  436. sourceType = 'store';
  437. console.log('来源类型: 店铺分享');
  438. }
  439. if (productId) {
  440. console.log('产品ID (productId):', productId);
  441. }
  442. console.log('===========================================');
  443. // 保存到临时存储,用于后续跳转和统计
  444. wx.setStorageSync('scan_storeId', storeId);
  445. wx.setStorageSync('scan_scanCount', scanCount || '0');
  446. wx.setStorageSync('scan_sourceType', sourceType);
  447. wx.setStorageSync('scan_sourceId', sourceId || '');
  448. wx.setStorageSync('scan_ownerId', ownerId || '');
  449. wx.setStorageSync('scan_employeeId', employeeId || '');
  450. wx.setStorageSync('scan_partnerId', partnerId || '');
  451. wx.setStorageSync('scan_userId', userId || '');
  452. if (productId) {
  453. wx.setStorageSync('scan_productId', productId);
  454. }
  455. wx.setStorageSync('need_scan_redirect', true);
  456. } else if (partnerId) {
  457. // 理论上异业二维码现在也必须带 storeId,这里仅作为兜底保护
  458. console.warn('⚠️ 检测到异业合作伙伴二维码缺少 storeId,无法确定门店,请检查二维码生成逻辑');
  459. }
  460. },
  461. /**
  462. * 判断是否需要跳转到扫码统计页面
  463. */
  464. shouldRedirectToScanPage() {
  465. return wx.getStorageSync('need_scan_redirect') === true;
  466. },
  467. /**
  468. * 跳转到扫码对应的店铺页面
  469. * 根据 storeId 跳转到对应店铺,同时记录所有来源信息用于统计
  470. */
  471. async redirectToScanPage() {
  472. try {
  473. // 获取所有扫码相关参数
  474. const storeId = wx.getStorageSync('scan_storeId');
  475. const scanCount = wx.getStorageSync('scan_scanCount');
  476. const sourceType = wx.getStorageSync('scan_sourceType');
  477. const sourceId = wx.getStorageSync('scan_sourceId');
  478. const ownerId = wx.getStorageSync('scan_ownerId');
  479. const employeeId = wx.getStorageSync('scan_employeeId');
  480. const partnerId = wx.getStorageSync('scan_partnerId');
  481. const userId = wx.getStorageSync('scan_userId');
  482. const productId = wx.getStorageSync('scan_productId');
  483. // 清除临时存储
  484. wx.removeStorageSync('scan_storeId');
  485. wx.removeStorageSync('scan_scanCount');
  486. wx.removeStorageSync('scan_sourceType');
  487. wx.removeStorageSync('scan_sourceId');
  488. wx.removeStorageSync('scan_ownerId');
  489. wx.removeStorageSync('scan_employeeId');
  490. wx.removeStorageSync('scan_partnerId');
  491. wx.removeStorageSync('scan_userId');
  492. wx.removeStorageSync('scan_productId');
  493. wx.removeStorageSync('need_scan_redirect');
  494. if (!storeId) {
  495. console.error('❌ 缺少 storeId 参数,无法跳转');
  496. return;
  497. }
  498. console.log('===========================================');
  499. console.log('======= 扫码进入店铺 =======');
  500. console.log('店铺 ID (storeId):', storeId);
  501. console.log('来源类型 (sourceType):', sourceType);
  502. console.log('来源 ID (sourceId):', sourceId || '无');
  503. console.log('扫码次数 (scanCount):', scanCount);
  504. if (ownerId) console.log('老板 ID (ownerId):', ownerId);
  505. if (employeeId) console.log('员工 ID (employeeId):', employeeId);
  506. if (partnerId) console.log('合作伙伴 ID (partnerId):', partnerId);
  507. if (userId) console.log('推广员 ID (userId):', userId);
  508. if (productId) console.log('产品 ID (productId):', productId);
  509. console.log('===========================================');
  510. // 设置店铺 ID 到全局和本地存储
  511. wx.setStorageSync('storeId', storeId);
  512. getApp().globalData.storeId = storeId;
  513. // 保存来源信息到本地存储(用于后续统计或绑定关系)
  514. if (sourceId) {
  515. wx.setStorageSync('scan_from_sourceType', sourceType);
  516. wx.setStorageSync('scan_from_sourceId', sourceId);
  517. console.log('✅ 已记录来源信息:', sourceType, sourceId);
  518. }
  519. // 记录扫码统计(调用后端接口记录扫码次数)
  520. await this.recordScanStatistics({
  521. storeId,
  522. sourceType,
  523. sourceId,
  524. ownerId,
  525. employeeId,
  526. partnerId,
  527. userId,
  528. productId,
  529. scanCount
  530. });
  531. // 如果有产品ID,直接跳转到产品详情的 H5 页面
  532. if (productId) {
  533. console.log('🎯 检测到产品ID,跳转到产品详情页');
  534. await this.redirectToProductDetail(storeId, productId, partnerId);
  535. return;
  536. }
  537. // 获取默认首页路径并跳转
  538. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath;
  539. url += `?storeId=${storeId}`;
  540. // 如果有产品ID,添加到URL参数中
  541. if (productId) {
  542. url += `&productId=${productId}`;
  543. }
  544. // 如果有异业合作伙伴ID,添加到URL参数中传回web-view
  545. if (partnerId) {
  546. url += `&partnerId=${partnerId}`;
  547. }
  548. console.log('✅ 跳转到店铺页面:', url);
  549. wx.redirectTo({
  550. url: url,
  551. fail: (err) => {
  552. console.error('❌ 跳转失败:', err);
  553. // 如果 redirectTo 失败,尝试 reLaunch
  554. wx.reLaunch({
  555. url: url,
  556. fail: (err2) => {
  557. console.error('❌ reLaunch 也失败:', err2);
  558. }
  559. });
  560. }
  561. });
  562. } catch (error) {
  563. console.error('❌ 跳转到店铺页面失败:', error);
  564. }
  565. },
  566. /**
  567. * 跳转到活动页面
  568. */
  569. async redirectToActivityPage() {
  570. try {
  571. const activityId = wx.getStorageSync('scan_activityId');
  572. // 清除临时存储
  573. wx.removeStorageSync('scan_activityId');
  574. wx.removeStorageSync('need_activity_redirect');
  575. if (!activityId) {
  576. console.error('❌ 缺少 activityId 参数,无法跳转');
  577. return;
  578. }
  579. console.log('===========================================');
  580. console.log('======= 扫码进入活动页面 =======');
  581. console.log('活动 ID (activityId):', activityId);
  582. console.log('===========================================');
  583. // 保存活动ID
  584. wx.setStorageSync('activityId', activityId);
  585. // 获取默认首页路径并跳转(活动页面可能需要根据实际路由调整)
  586. let url = getApp().globalData.rootPage || getApp().globalData.defaultTabBar.list[0].pagePath;
  587. url += `?activityId=${activityId}`;
  588. console.log('✅ 跳转到活动页面:', url);
  589. wx.redirectTo({
  590. url: url,
  591. fail: (err) => {
  592. console.error('❌ 跳转失败:', err);
  593. wx.reLaunch({
  594. url: url,
  595. fail: (err2) => {
  596. console.error('❌ reLaunch 也失败:', err2);
  597. }
  598. });
  599. }
  600. });
  601. } catch (error) {
  602. console.error('❌ 跳转到活动页面失败:', error);
  603. }
  604. },
  605. /**
  606. * 跳转到产品详情的 H5 页面
  607. * @param {string} storeId - 店铺 ID
  608. * @param {string} productId - 产品 ID
  609. * @param {string} partnerId - 可选的合作伙伴 ID
  610. */
  611. async redirectToProductDetail(storeId, productId, partnerId = null) {
  612. try {
  613. console.log('===========================================');
  614. console.log('======= 跳转到产品详情页 =======');
  615. console.log('店铺 ID:', storeId);
  616. console.log('产品 ID:', productId);
  617. if (partnerId) console.log('合作伙伴 ID:', partnerId);
  618. console.log('===========================================');
  619. const currentUser = Parse.User.current();
  620. if (!currentUser) {
  621. console.error('❌ 用户未登录,无法跳转');
  622. return;
  623. }
  624. const token = currentUser.getSessionToken();
  625. if (!token) {
  626. console.error('❌ 无法获取 Session Token');
  627. return;
  628. }
  629. // 构建产品详情的 H5 URL
  630. let h5Url = `https://app.fmode.cn/dev/pobingfeng/owner/nav/products?storeId=${storeId}&token=${token}&productId=${encodeURIComponent(productId)}`;
  631. // 如果有合作伙伴ID,也添加到URL中
  632. if (partnerId) {
  633. h5Url += `&partnerId=${partnerId}`;
  634. }
  635. console.log('🌐 H5 URL:', h5Url);
  636. // 编码 URL
  637. const encodedUrl = encodeURIComponent(h5Url);
  638. // 构建 web-view 页面路径
  639. let webViewPath = `/common-page/pages/web-view/index?path=${encodedUrl}&storeId=${storeId}`;
  640. console.log('📄 web-view 页面路径:', webViewPath.substring(0, 100) + '...');
  641. console.log('===========================================');
  642. wx.redirectTo({
  643. url: webViewPath,
  644. success: () => {
  645. console.log('✅ 跳转到产品详情页成功');
  646. },
  647. fail: (err) => {
  648. console.error('❌ 跳转失败:', err);
  649. // 如果 redirectTo 失败,尝试 reLaunch
  650. wx.reLaunch({
  651. url: webViewPath,
  652. fail: (err2) => {
  653. console.error('❌ reLaunch 也失败:', err2);
  654. }
  655. });
  656. }
  657. });
  658. } catch (error) {
  659. console.error('❌ 跳转到产品详情页失败:', error);
  660. }
  661. },
  662. /**
  663. * 记录扫码统计信息
  664. * @param {Object} params - 扫码参数对象
  665. * @param {string} params.storeId - 店铺 ID
  666. * @param {string} params.sourceType - 来源类型 (employee/owner/partner/promoter/store)
  667. * @param {string} params.sourceId - 来源 ID
  668. * @param {string} params.ownerId - 老板 ID
  669. * @param {string} params.employeeId - 员工 ID
  670. * @param {string} params.partnerId - 合作伙伴 ID
  671. * @param {string} params.userId - 推广员 ID
  672. * @param {string} params.productId - 产品 ID
  673. * @param {string} params.scanCount - 扫码次数
  674. */
  675. async recordScanStatistics(params) {
  676. try {
  677. const {
  678. storeId,
  679. sourceType,
  680. sourceId,
  681. ownerId,
  682. employeeId,
  683. partnerId,
  684. userId,
  685. productId,
  686. scanCount
  687. } = params;
  688. // 如果没有来源信息,不记录统计
  689. if (!sourceId && !ownerId && !employeeId && !partnerId && !userId) {
  690. console.log('ℹ️ 无来源信息,跳过统计记录');
  691. return;
  692. }
  693. // 创建扫码记录
  694. const ScanRecord = Parse.Object.extend('ScanRecord');
  695. const record = new ScanRecord();
  696. record.set('company', {
  697. __type: 'Pointer',
  698. className: 'Company',
  699. objectId: getApp().globalData.company
  700. });
  701. record.set('storeId', storeId);
  702. record.set('sourceType', sourceType || 'unknown');
  703. record.set('scanCount', parseInt(scanCount) || 0);
  704. record.set('scanTime', new Date());
  705. // 根据来源类型设置对应的ID
  706. if (ownerId) {
  707. record.set('ownerId', ownerId);
  708. }
  709. if (employeeId) {
  710. record.set('employeeId', employeeId);
  711. }
  712. if (partnerId) {
  713. record.set('partnerId', partnerId);
  714. }
  715. if (userId) {
  716. record.set('userId', userId);
  717. }
  718. if (productId) {
  719. record.set('productId', productId);
  720. }
  721. // 如果用户已登录,记录扫码用户
  722. const currentUser = Parse.User.current();
  723. if (currentUser) {
  724. record.set('user', {
  725. __type: 'Pointer',
  726. className: '_User',
  727. objectId: currentUser.id
  728. });
  729. }
  730. await record.save();
  731. console.log('✅ 扫码记录已保存');
  732. // 如果存在异业合作伙伴ID,则在其 scanCount 字段上自增 1(数据库中的 Partner 表)
  733. if (partnerId) {
  734. try {
  735. const Partner = Parse.Object.extend('Partner');
  736. const partnerQuery = new Parse.Query(Partner);
  737. const partnerObj = await partnerQuery.get(partnerId);
  738. if (partnerObj) {
  739. partnerObj.increment('scanCount', 1);
  740. await partnerObj.save();
  741. console.log('✅ 已为异业合作伙伴累加一次扫码次数,partnerId:', partnerId);
  742. }
  743. } catch (e) {
  744. console.warn('⚠️ 更新异业合作伙伴扫码次数失败:', e);
  745. }
  746. }
  747. } catch (error) {
  748. console.warn('⚠️ 保存扫码记录失败:', error);
  749. // 不影响主流程
  750. }
  751. }
  752. });