api.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. /**
  2. * 请任何自动化工具每次更新本文档,要再下面的版本号自动加1,并更新日期
  3. *
  4. * 版本号:1
  5. * 日期:2025-12-4 16:14:00
  6. */
  7. (function (window) {
  8. 'use strict';
  9. // Logger Utility
  10. const Logger = {
  11. _isDev: false,
  12. init: function(isDev) {
  13. this._isDev = isDev;
  14. },
  15. log: function() {
  16. if (this._isDev) {
  17. console.log.apply(console, arguments);
  18. }
  19. },
  20. warn: function() {
  21. if (this._isDev) {
  22. console.warn.apply(console, arguments);
  23. }
  24. },
  25. error: function() {
  26. console.error.apply(console, arguments); // Always log errors
  27. }
  28. };
  29. // Determine _isDev status from main window's URL query params
  30. function getQueryParam(name) {
  31. const params = new URLSearchParams(window.location.search);
  32. return params.get(name);
  33. }
  34. const env = (getQueryParam('env') || '').toLowerCase();
  35. Logger.init(env === 'mock'); // Initialize Logger
  36. /**
  37. * ColorMapRun API SDK (Full Version with Mock)
  38. * 封装了与后端服务器的所有交互
  39. * 依赖: bridge.js (用于 401 跳转登录)
  40. */
  41. // 基础配置
  42. var Config = {
  43. baseUrl: 'https://colormaprun.com/api/card/',
  44. ossUrl: 'http://oss-card.colormaprun.com/card/',
  45. token: '',
  46. useMock: false
  47. };
  48. // 模拟数据定义 (涵盖所有接口)
  49. var MOCK_DB = {
  50. 'CardBaseQuery': {
  51. ecName: '[Mock]跑向大明湖卡片',
  52. ecDesc: '欢迎参加彩图奔跑活动!',
  53. beginSecond: Date.now() / 1000 - 86400 * 5,
  54. endSecond: Date.now() / 1000 + 86400 * 10,
  55. secondCardName: '地图导航'
  56. },
  57. 'CardDetailQuery': {
  58. mcId: 101,
  59. mcName: '[Mock]线上马拉松',
  60. nickName: '[Mock]用户昵称', // Added as per user request
  61. mcType: 1,
  62. beginSecond: Date.now() / 1000 - 86400 * 2,
  63. endSecond: Date.now() / 1000 + 86400 * 5,
  64. teamNum: 0,
  65. coiId: 1, // Mocked coiId
  66. coiName: '个人组',
  67. ocaId: 201 // Mocked ocaId
  68. },
  69. 'MatchRsDetailQuery': {
  70. id: 1,
  71. name: '[Mock]活动1',
  72. status: 1,
  73. totalSysPoint: 100,
  74. mcType: 1,
  75. mcId: 101,
  76. mcName: '[Mock]线上马拉松',
  77. beginSecond: Date.now() / 1000 - 86400 * 2,
  78. endSecond: Date.now() / 1000 + 86400 * 5,
  79. nickName: '[Mock]用户昵称',
  80. totalNum: 10,
  81. totalDistanct: 5000,
  82. totalDistanctRankNum: 5,
  83. totalCp: 20,
  84. totalCpRankNum: 3,
  85. totalSysPointRankNum: 10,
  86. fastPace: 300,
  87. fastPaceRankNum: 8
  88. },
  89. 'CardRankDetailQuery': {
  90. totalRankRs: [
  91. { userName: 'Mock张三', score: 10000, headUrl: 'https://picsum.photos/40/40?random=1', rankNum: 1, isSelf: 0, inRankNum: 1, isInGame: 0, additionalName: '计算机学院', isDispAdditionalName: 1, inGameUserNum: 0, isDispInGameUserNum: 0, isTodayFinishGame: 0 },
  92. { userName: 'Mock李四', score: 9500, headUrl: 'https://picsum.photos/40/40?random=2', rankNum: 2, isSelf: 0, inRankNum: 2, isInGame: 0, additionalName: '土木建筑', isDispAdditionalName: 1, inGameUserNum: 0, isDispInGameUserNum: 0, isTodayFinishGame: 0 },
  93. { userName: 'Mock王五', score: 8800, headUrl: 'https://picsum.photos/40/40?random=3', rankNum: 3, isSelf: 0, inRankNum: 3, isInGame: 0, additionalName: '艺术设计', isDispAdditionalName: 1, inGameUserNum: 0, isDispInGameUserNum: 0, isTodayFinishGame: 0 },
  94. { userName: 'Mock赵六', score: 7200, headUrl: 'https://picsum.photos/40/40?random=4', rankNum: 4, isSelf: 0, inRankNum: 4, isInGame: 0, additionalName: '交通运输', isDispAdditionalName: 1, inGameUserNum: 0, isDispInGameUserNum: 0, isTodayFinishGame: 0 },
  95. { userName: 'Mock小明', score: 6500, headUrl: 'https://picsum.photos/40/40?random=5', rankNum: 5, isSelf: 1, inRankNum: 5, isInGame: 0, additionalName: '信息工程', isDispAdditionalName: 1, inGameUserNum: 0, isDispInGameUserNum: 0, isTodayFinishGame: 0 }
  96. ],
  97. teamRankRs: [],
  98. inTeamRs: [],
  99. otherRs: [],
  100. regionPaceRs: null,
  101. regionTodayPaceRs: null,
  102. regionSpeedRs: null
  103. },
  104. 'UserCurrentRankNumQuery': { rankNum: 5 },
  105. 'UserJoinCardQuery': { isJoin: false }, // 默认未报名
  106. 'IsNewUserInCardComp': { isNew: true },
  107. 'OnlineMcSignUpDetail': {
  108. teamList: [
  109. { teamId: 1, teamName: '[Mock]个人组' },
  110. { teamId: 2, teamName: '[Mock]团队组' }
  111. ],
  112. signupFields: [
  113. { name: 'realName', label: '真实姓名', type: 'text', required: true, value: '' },
  114. { name: 'phone', label: '手机号码', type: 'tel', required: true, value: '' }
  115. ]
  116. },
  117. 'OnlineMcSignUp': {}, // 报名成功返回空 data
  118. 'IsAllowMcSignUp': { allowSignUp: true },
  119. 'CurrentMonthlyChallengeQuery': {
  120. year: '2023',
  121. monthRs: [{ month: 11, realNum: 10, targetNum: 20 }]
  122. },
  123. 'CardConfigQuery': {
  124. configJson: JSON.stringify({
  125. css: ".custom-header { background-color: #f0f8ff; }",
  126. tabActiveColor: "#007bff",
  127. teamType: 0, // Mocked teamType
  128. popupRuleConfig: { height: "60%", theme: "light" },
  129. popupMessageConfig: {}, // Mocked popupMessageConfig
  130. popupRuleList: [
  131. { type: 1, data: { title: '规则1', content: '这是<b>Mock</b>的活动规则内容。' } },
  132. { type: 1, data: { title: '规则2', content: '第二条规则。', logo: { src: 'https://picsum.photos/100/50', width: '100px', height: '50px' } } }
  133. ],
  134. popupDataList: [ // Mocked popupDataList
  135. { type: 1, data: { title: '通用弹窗1', content: '通用弹窗内容1。' } }
  136. ]
  137. })
  138. },
  139. 'UserConfigQuery': {
  140. configJson: JSON.stringify({
  141. tplInfo: { tplTypeId: 1, ssctId: 1 },
  142. mapInfo: [
  143. { activityList: [{ showName: '迷你路线', pathImg: '', ocaId: 1, matchType: 1, point: { longitude: 117.0, latitude: 36.6, name: '起点' } }] }
  144. ],
  145. popupRuleList: [{ type: 1, data: { title: '用户个性化', content: '这是用户专属内容' } }]
  146. })
  147. },
  148. 'MonthlyChallengeQuery': [
  149. { year: '2023', monthRs: [{ month: 10, realNum: 15, targetNum: 20 }] },
  150. { year: '2022', monthRs: [{ month: 12, realNum: 25, targetNum: 20 }] }
  151. ],
  152. 'MonthRankDetailQuery': [
  153. { nickName: '月榜冠军', score: 500, headUrl: 'https://picsum.photos/40/40?random=6' }
  154. ],
  155. 'AchievementQuery': [
  156. {
  157. year: '2023',
  158. aiRs: [
  159. { aiName: '初次登场', aiTime: Date.now() / 1000 - 86400 * 30, iconUrl: 'https://picsum.photos/60/60?random=7' },
  160. { aiName: '跑步达人', aiTime: Date.now() / 1000 - 86400 * 10, iconUrl: 'https://picsum.photos/60/60?random=8' }
  161. ]
  162. }
  163. ],
  164. 'ExchangeListQuery': [
  165. { exchangeId: 1, goodsName: '[Mock]运动手环', createTime: Date.now() / 1000 - 86400 * 7, status: 1 },
  166. { exchangeId: 2, goodsName: '[Mock]定制水杯', createTime: Date.now() / 1000 - 86400 * 15, status: 0 }
  167. ],
  168. 'ExchangeDetailQuery': {
  169. exchangeId: 1, goodsName: '[Mock]运动手环', createTime: Date.now() / 1000 - 86400 * 7, status: 1,
  170. address: 'Mock地址', receiver: 'Mock收件人', phone: '138****8888'
  171. },
  172. 'UnReadMessageQuery': [
  173. { mqId: 1, mqType: 1, mqTitle: '[Mock]新成就', mqMessage: '恭喜您解锁新成就!', iconUrl: 'https://picsum.photos/50/50?random=9' }
  174. ],
  175. 'ReadMessage': {},
  176. 'MapListQuery': [
  177. { mapId: 1, mapName: '[Mock]公园地图', latitude: 36.6, longitude: 117.0, activityList: [] }
  178. ],
  179. 'CompStatisticQuery': { totalDistance: 5024, totalPeople: 1000, totalAnswerNum: 1500, totalCp: 8900 },
  180. 'WarnMessageQuery': [
  181. { warnType: 1, warnTitle: '[Mock]黄牌警告', warnMessage: '您的成绩异常', iconUrl: 'https://picsum.photos/50/50?random=10' }
  182. ],
  183. 'CertStyleQuery': {
  184. styleId: 1, styleName: '简约风', templateUrl: 'https://picsum.photos/600/400?random=11',
  185. elements: [
  186. { type: 'text', field: 'userName', x: 100, y: 150, fontSize: 24, color: '#333' }
  187. ]
  188. },
  189. 'UserBaseQueryInCertificate': { userName: '[Mock]证书用户', activityName: '[Mock]活动名', completionTime: Date.now() / 1000 },
  190. 'CertificateCreateByUserAi': { certUrl: 'https://picsum.photos/600/400?random=12' },
  191. 'OnlineScoreQuery': { score: 880, extTime: Date.now() / 1000 + 86400 * 30 },
  192. 'CanExchangeGoodsList': [
  193. { goodsId: 1, goodsName: '[Mock]运动手环', goodsPic: 'https://picsum.photos/100/100?random=13', goodsLeftNum: 50, corrScore: 500 },
  194. { goodsId: 2, goodsName: '[Mock]定制水杯', goodsPic: 'https://picsum.photos/100/100?random=14', goodsLeftNum: 0, corrScore: 300 }
  195. ],
  196. 'CanExchangeGoodsDetail': {
  197. goodsId: 1, goodsName: '[Mock]运动手环', goodsPic: 'https://picsum.photos/200/200?random=15', corrScore: 500,
  198. desc: '这是<b>Mock</b>的运动手环详情,功能强大,是您运动的好伙伴!'
  199. },
  200. 'ScoreExchangeGoods': {},
  201. 'UserBasicInformationQuery': { nickName: 'Mock用户', headUrl: 'https://picsum.photos/50/50?random=16' },
  202. 'GridsQuery': {
  203. compId: 201, compName: '[Mock]网格挑战', widthNum: 3, heightNum: 3,
  204. maskImgPic: 'https://picsum.photos/300/300?random=17',
  205. actualImgPic: 'https://picsum.photos/300/300?random=18',
  206. state: 2, // 1:未开始 2:进行中 3:已结束
  207. detailRs: [
  208. { orderNum: 1, isComplete: 1, showName: '完成1', relationType: 1, ocaId: 10, longitude: 117.1, latitude: 36.6, popupImg: 'https://picsum.photos/100/100?random=19' },
  209. { orderNum: 2, isComplete: 0, showName: '未完2', relationType: 1, ocaId: 11, longitude: 117.2, latitude: 36.7, popupImg: 'https://picsum.photos/100/100?random=20' }
  210. ]
  211. },
  212. 'CardUrlQuery': { url: 'https://mock-uri.colormaprun.com/card/nanning1/index.html'}, // 暂无 Mock 结构
  213. 'MatchFininshInfoQuery': {}, // 暂无 Mock 结构
  214. 'RedisRebulid': {} // 暂无 Mock 结构
  215. };
  216. var API = {
  217. init: function(options) {
  218. if (options.baseUrl) Config.baseUrl = options.baseUrl;
  219. if (options.ossUrl) Config.ossUrl = options.ossUrl;
  220. if (options.token) Config.token = options.token;
  221. if (typeof options.useMock === 'boolean') Config.useMock = options.useMock;
  222. if (Config.useMock) Logger.warn('%c [API] Mock 模式已开启 ', 'background: orange; color: white;');
  223. },
  224. getOssUrl: function() {
  225. return Config.ossUrl;
  226. },
  227. setToken: function(token) {
  228. Config.token = token;
  229. },
  230. request: function(endpoint, data, method) {
  231. var reqMethod = (method || 'POST').toUpperCase();
  232. if (Config.useMock) {
  233. return new Promise(function(resolve, reject) {
  234. Logger.log('[API-Mock] Request (' + reqMethod + '):', endpoint, data);
  235. setTimeout(function() {
  236. var mockData = MOCK_DB[endpoint];
  237. if (endpoint === 'OnlineMcSignUp') MOCK_DB['UserJoinCardQuery'].isJoin = true;
  238. if (endpoint === 'ScoreExchangeGoods') MOCK_DB['OnlineScoreQuery'].score -= 100;
  239. Logger.log('[API-Mock] Response:', endpoint, mockData || {});
  240. resolve(mockData || {});
  241. }, 300);
  242. });
  243. }
  244. var url = Config.baseUrl + endpoint;
  245. var body = null;
  246. if (reqMethod === 'GET') {
  247. var headers = { 'Content-Type': 'application/x-www-form-urlencoded','Authorization': 'Bearer ' + Config.token };
  248. var params = new URLSearchParams();
  249. for (var key in data) { if (data.hasOwnProperty(key)) params.append(key, data[key]); }
  250. var queryString = params.toString();
  251. if (queryString) url += '?' + queryString;
  252. } else {
  253. var headers = { 'Content-Type': 'application/x-www-form-urlencoded', 'token': Config.token};
  254. var formData = new URLSearchParams();
  255. for (var key in data) { if (data.hasOwnProperty(key)) formData.append(key, data[key]); }
  256. body = formData;
  257. }
  258. Logger.log('[API] Request (' + reqMethod + '):', endpoint, data);
  259. return fetch(url, { method: reqMethod, headers: headers, body: body, mode: 'cors', credentials: 'omit' })
  260. .then(function(response) { return response.json(); })
  261. .then(function(res) {
  262. Logger.log('[API] Response:', endpoint, res);
  263. if (res.code === 0) return res.data;
  264. // if (res.code === 401 || res.statusCode === 401) {
  265. // Logger.warn('[API] Token invalid');
  266. // if (window.Bridge && window.Bridge._post) window.Bridge._post('toLogin');
  267. // else alert('登录已过期');
  268. // throw new Error('Unauthorized');
  269. // }
  270. // var msg = res.message || '请求失败';
  271. // if (window.Bridge && window.Bridge.showToast) {
  272. // window.Bridge.showToast(msg, 'none');
  273. // } else {
  274. // alert(msg);
  275. // }
  276. // throw new Error(msg);
  277. })
  278. .catch(function(err) { Logger.error('[API] Error:', err); throw err; });
  279. },
  280. // ==============================
  281. // 完整业务接口封装 (按原始 api.js 顺序)
  282. // ==============================
  283. // 1. 卡片基本信息查询
  284. getCardBase: function(ecId) { return this.request('CardBaseQuery', { ecId: ecId }); },
  285. // 2. 卡片对应活动或赛事详情查询
  286. getCardDetail: function(ecId) { return this.request('CardDetailQuery', { ecId: ecId }); },
  287. // 3. 卡片对应线上赛多个活动查询
  288. getMatchRsDetail: function(ecId, ocaId) { return this.request('MatchRsDetailQuery', { ecId: ecId, ocaId: ocaId }); },
  289. // 4. 排名查询
  290. getRankDetail: function(mcIdListStr, mcType, ocaId, dispArrStr) { return this.request('CardRankDetailQuery', { mcIdListStr: mcIdListStr, mcType: mcType, ocaId: ocaId, dispArrStr: dispArrStr }); },
  291. // 5. 卡片用户当前排名查询
  292. getUserCurrentRank: function(ecId) { return this.request('UserCurrentRankNumQuery', { ecId: ecId }); },
  293. // 6. 用户是否已经报名卡片对应赛事查询
  294. getUserJoinStatus: function(ecId) { return this.request('UserJoinCardQuery', { ecId: ecId }); },
  295. // 7. 用户在卡片对应赛事是否新用户
  296. isNewUserInCardComp: function(ecId) { return this.request('IsNewUserInCardComp', { ecId: ecId }); },
  297. // 8. 线上赛报名页面信息详情
  298. getOnlineMcSignUpDetail: function(ecId, mcId) { return this.request('OnlineMcSignUpDetail', { ecId: ecId, mcId: mcId }); },
  299. // 9. 线上赛报名(重新分组)
  300. signUpOnline: function(mcId, coiId, selectTeam, nickName) {
  301. return this.request('OnlineMcSignUp', { mcId: mcId, coiId: coiId, selectTeam: selectTeam, nickName: nickName });
  302. },
  303. // 10. 是否允许重新分组(报名)
  304. isAllowMcSignUp: function(ecId) { return this.request('IsAllowMcSignUp', { ecId: ecId }); },
  305. // 11. 玩家当前月挑战记录查询
  306. getCurrentMonthlyChallenge: function(year, month) { return this.request('CurrentMonthlyChallengeQuery', {year: year, month: month}); },
  307. // 12. 卡片配置信息查询
  308. getCardConfig: function(ecId, pageName) { return this.request('CardConfigQuery', { ecId: ecId, pageName: pageName }); },
  309. // 13. 用户自定义配置信息查询
  310. getUserConfig: function(ecId, pageName) { return this.request('UserConfigQuery', { ecId: ecId, pageName: pageName }); },
  311. // 14. 玩家所有月挑战记录查询
  312. getMonthlyChallenge: function() { return this.request('MonthlyChallengeQuery', {}); },
  313. // 15. 月挑战排名查询
  314. getMonthRankDetail: function(year, month, dispArrStr) { return this.request('MonthRankDetailQuery', {year: year, month: month, dispArrStr: dispArrStr}); },
  315. // 16. 玩家活动成就查询
  316. getAchievement: function() { return this.request('AchievementQuery', {}); },
  317. // 17. 玩家兑换记录查询
  318. getExchangeList: function() { return this.request('ExchangeListQuery', {}); },
  319. // 18. 玩家兑换详情查询
  320. getExchangeDetail: function(oarId) { return this.request('ExchangeDetailQuery', { oarId: oarId}); },
  321. // 19. 未读消息列表查询
  322. getUnReadMessages: function(relationId, relationType) { return this.request('UnReadMessageQuery', { relationId: relationId, relationType: relationType || 2 }); },
  323. // 20. 标记消息已读
  324. readMessage: function(mqIdListStr) { return this.request('ReadMessage', { mqIdListStr: mqIdListStr }); },
  325. // 21. 卡片对应地图列表详情查询
  326. getMapList: function(mcId) { return this.request('MapListQuery', { mcId: mcId }); },
  327. // 22. 赛事总成绩统计查询
  328. getCompStatistic: function(mcId) { return this.request('CompStatisticQuery', { mcId: mcId }); },
  329. // 23. 警告列表查询
  330. getWarnMessage: function(ecId) { return this.request('WarnMessageQuery', { ecId: ecId }); },
  331. // 24. 查询电子证书样式
  332. getCertStyle: function(certStyleType) { return this.request('CertStyleQuery', { certStyleType: certStyleType }); },
  333. // 25. 查询电子证书成就对应用户基本信息
  334. getUserBaseInCertificate: function(oarId) { return this.request('UserBaseQueryInCertificate', { oarId: oarId }); },
  335. // 26. 根据成就信息确认生成电子证书
  336. createCertificate: function(nickName,oarId) { return this.request('CertificateCreateByUserAi', {nickName: nickName, oarId: oarId}); },
  337. // 27. 卡片内可用积分查询
  338. getScore: function(ecId) { return this.request('OnlineScoreQuery', { ecId: ecId }); },
  339. // 28. 积分可兑换商品列表查询
  340. getGoodsList: function(ecId) { return this.request('CanExchangeGoodsList', { ecId: ecId }); },
  341. // 29. 积分可兑换商品详情
  342. getGoodsDetail: function(ecId, goodsId) { return this.request('CanExchangeGoodsDetail', {ecid: ecId, goodsId: goodsId }); },
  343. // 30. 积分兑换商品
  344. exchangeGoods: function(ecId, goodsId, exchNum) { return this.request('ScoreExchangeGoods', { ecId: ecId, goodsId: goodsId, exchNum: exchNum }); },
  345. // 31. 用户基本信息查询
  346. getUserInfo: function() { return this.request('UserBasicInformationQuery', {}, 'GET'); },
  347. // 32. 网格卡片信息查询
  348. getGrids: function(ecId) { return this.request('GridsQuery', { ecId: ecId }); },
  349. // 33. 卡片URI查询
  350. getCardUrl: function(actId, matchType) { return this.request('CardUrlQuery', { actId: actId, matchType: matchType }); },
  351. // 34. 赛事完赛信息查询
  352. getMatchFininshInfo: function(actId, matchType) { return this.request('MatchFininshInfoQuery', { actId, matchType }); },
  353. // 35. Redis 重建 (管理接口)
  354. redisRebulid: function(compId) { return this.request('RedisRebulid', { compId: compId }); }
  355. };
  356. window.API = API;
  357. })(window);