player.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506
  1. // import L from 'leaflet'
  2. import config from '@/utils/map/sub/config'
  3. import global from '@/utils/map/sub/global'
  4. // var interval_creatCircleMarker = null
  5. // var interval_showTrail = null
  6. var togglePlayerFlag = true
  7. var toggleTooltipFlag = true
  8. var toggleTrailFlag = true
  9. export default {
  10. playerLayerGroup: null, // 玩家标记图层组
  11. trailLayerGroup: null, // 玩家轨迹图层组
  12. init() {
  13. this.playerLayerGroup = new L.layerGroup()
  14. this.trailLayerGroup = new L.layerGroup()
  15. },
  16. // 即将开始地图缩放时触发本方法
  17. onZoomStart(e) {},
  18. // 地图发生缩放时触发本方法
  19. onZoom(e) {},
  20. // 地图缩放结束时触发本方法
  21. onZoomEnd(e) {},
  22. togglePlayer(flag) {
  23. if (flag != null) {
  24. togglePlayerFlag = !flag
  25. }
  26. if (togglePlayerFlag) {
  27. this.playerLayerGroup.removeFrom(global.map)
  28. togglePlayerFlag = false
  29. } else {
  30. this.playerLayerGroup.addTo(global.map)
  31. togglePlayerFlag = true
  32. }
  33. },
  34. toggleTooltip(flag) {
  35. if (flag != null) {
  36. toggleTooltipFlag = !flag
  37. }
  38. if (!togglePlayerFlag) {
  39. this.togglePlayer(true)
  40. toggleTooltipFlag = false
  41. }
  42. for (let i = 0; i < global.storePlayers.length; i++) {
  43. var player = global.storePlayers[i]
  44. if (player != null && player.marker != null) {
  45. // player.marker.toggleTooltip()
  46. // let isopen = player.marker.isTooltipOpen()
  47. if (toggleTooltipFlag) {
  48. player.marker.closeTooltip()
  49. } else {
  50. player.marker.openTooltip()
  51. }
  52. }
  53. }
  54. toggleTooltipFlag = !toggleTooltipFlag
  55. },
  56. toggleTrail(flag) {
  57. if (flag != null) {
  58. toggleTrailFlag = !flag
  59. }
  60. if (toggleTrailFlag) {
  61. this.trailLayerGroup.removeFrom(global.map)
  62. toggleTrailFlag = false
  63. } else {
  64. this.trailLayerGroup.addTo(global.map)
  65. toggleTrailFlag = true
  66. }
  67. },
  68. drawAllPlayers() {
  69. // console.log('[drawAllPlayers]', global.players)
  70. // if (this.playerLayerGroup != null)
  71. // this.playerLayerGroup.clearLayers()
  72. for (let i = 0; i < global.players.length; i++) {
  73. this.drawOnePlayer(global.players[i].id)
  74. // this.drawOnePlayer2(global.players[i].id)
  75. }
  76. },
  77. drawOnePlayer2(playerId, animate = true) {
  78. var that = this
  79. var player = global.getPlayerById(playerId)
  80. var player_position = global.getPlayerPositionById(playerId)
  81. var storePlayer = global.getStorePlayersById(playerId)
  82. // console.log(player, player_position)
  83. if (player == null || player_position == null) {
  84. console.warn('[drawOnePlayer2] 关键数据为空', player, player_position)
  85. return
  86. }
  87. // 首次创建
  88. if (storePlayer == null || storePlayer.marker == null) {
  89. // 在地图上创建 marker 并存储用户信息
  90. var playerIcon = L.icon({
  91. iconUrl: 'static/image/marker-icon.png',
  92. iconSize: [20, 32.8],
  93. iconAnchor: [10, 30]
  94. });
  95. var marker = L.marker([player_position.latitude, player_position.longitude], {
  96. icon: playerIcon
  97. })
  98. .addTo(this.playerLayerGroup)
  99. // .addTo(global.map)
  100. // .bindPopup("Hello, I'm a Marker!<br><img src='my-image.png' width='100'>").openPopup();
  101. .bindTooltip(`${player.name}`, {
  102. permanent: true,
  103. offset: [0, -30],
  104. direction: 'top',
  105. interactive: true
  106. })
  107. marker.type = 'marker'
  108. marker.playerId = player.id // 存储用户信息
  109. marker.animate = animate
  110. marker.on('click', function(e) {
  111. var playerId = e.target.playerId; // 获取点击的 marker 的用户信息
  112. console.log("Selected player ID: " + playerId);
  113. that.handleFocusPlayer(playerId)
  114. });
  115. var tooltip = marker.getTooltip()
  116. tooltip.playerId = player.id // 存储用户信息
  117. tooltip.animate = animate
  118. tooltip.on('click', function(e) {
  119. // console.log("[Marker.Tooltip]", e)
  120. var playerId = e.target.playerId; // 获取点击的 marker 的用户信息
  121. console.log("[Marker.Tooltip] Selected player ID: " + playerId)
  122. that.handleFocusPlayer(playerId)
  123. })
  124. // this.playerLayerGroup.addTo(global.map)
  125. if (storePlayer == null) {
  126. storePlayer = {
  127. id: playerId,
  128. marker: marker, // 玩家标识
  129. markerLayerGroup: this.playerLayerGroup,
  130. trail: null, // 玩家轨迹
  131. trailData: [], // 玩家轨迹信息
  132. interval_creatCircleMarker: null,
  133. interval_showTrail: null,
  134. }
  135. global.storePlayers.push(storePlayer)
  136. } else {
  137. storePlayer.marker = marker
  138. storePlayer.markerLayerGroup = this.playerLayerGroup
  139. }
  140. }
  141. // 非首次创建,直接移动位置
  142. else {
  143. if (!toggleTrailFlag) {
  144. var marker = storePlayer.marker
  145. var curPoint = [player_position.latitude, player_position.longitude]
  146. // console.warn('[drawOnePlayer] 更新玩家的标记位置', storePlayer.marker, curPoint)
  147. marker.setLatLng(curPoint)
  148. }
  149. }
  150. },
  151. // 创建动画效果函数,用于实现circleMarker的大小和透明度随时间变化
  152. animateCircle(circle, minRadius, maxRadius, step = 1) {
  153. var radius = circle.getRadius();
  154. var opacity = circle.options.fillOpacity;
  155. var zoomType = 'zoomIn' // zoomIn: 放大 zoomOut: 缩小
  156. var interval_creatCircleMarker = setInterval(function() {
  157. if (!circle.animate) {
  158. clearInterval(interval_creatCircleMarker)
  159. return
  160. }
  161. // 改变圆形半径和透明度
  162. if (zoomType == 'zoomIn') {
  163. radius = radius + step
  164. // opacity = opacity - 0.1
  165. } else if (zoomType == 'zoomOut') {
  166. radius = radius - step
  167. // opacity = opacity - 0.1
  168. }
  169. circle.setRadius(radius);
  170. circle.setStyle({
  171. fillOpacity: opacity
  172. });
  173. if (radius >= maxRadius) {
  174. zoomType = 'zoomOut'
  175. } else if (radius <= minRadius) {
  176. zoomType = 'zoomIn'
  177. }
  178. }, 50);
  179. return interval_creatCircleMarker
  180. },
  181. getPopupContent(player) {
  182. var popupContent = '<div style="line-height: 20px;">'
  183. popupContent += '<div style="color: blue">' + ' <span onClick="uni.makePhoneCall({phoneNumber: \'' + player
  184. .phone + '\'});">' + player.name + '</span></div>'
  185. popupContent += '<div>打点 ' + player.effectivenum + '/' + player.totalcontrolnum +
  186. ' 距离 ' + player.distance + '米 配速 ' + player.pace + '</div>'
  187. popupContent += '<div>' + '心率 ' + player.lasthr + ' 平均 ' + player.avghr + ' 最大 ' + player.maxhr + '</div>'
  188. popupContent += '<div>CAL ' + Math.round(player.calorie / 1000) + ' Ck ' + Math.round(player.ck / 1000) +
  189. ' Ei ' + Math.round(player.ei * 100) / 100 + '</div>'
  190. popupContent += '</div>'
  191. return popupContent
  192. },
  193. drawOnePlayer(playerId, animate = true) {
  194. var that = this
  195. var player = global.getPlayerById(playerId)
  196. var player_position = global.getPlayerPositionById(playerId)
  197. var storePlayer = global.getStorePlayersById(playerId)
  198. // console.log('[drawOnePlayer]', player, player_position)
  199. if (player == null) {
  200. console.warn('[drawOnePlayer] 玩家数据为空', player)
  201. return
  202. }
  203. if (player_position == null) {
  204. // console.log('[drawOnePlayer] 玩家位置数据为空', player_position)
  205. return
  206. }
  207. var popupContent = this.getPopupContent(player)
  208. // 首次创建
  209. if (storePlayer == null || storePlayer.marker == null) {
  210. // console.log('[drawOnePlayer] storePlayer == null', playerId)
  211. var playerColor = this.getPlayerColor(playerId)
  212. var options = config.gStyle.marker.default
  213. options.fillColor = playerColor
  214. options.stroke = config.gStyle.common.stroke
  215. options.color = config.gStyle.common.color
  216. // console.log('[drawOnePlayer] options', options)
  217. var marker = L.circleMarker([player_position.latitude, player_position.longitude], options)
  218. .addTo(this.playerLayerGroup)
  219. .bindPopup(popupContent)
  220. .bindTooltip(`${player.name}`, config.gStyle.marker.tooltip)
  221. marker.type = 'circleMarker'
  222. marker.playerId = player.id // 存储用户信息
  223. marker.animate = animate
  224. marker.on('click', function(e) {
  225. // console.log("[Marker]", e)
  226. var playerId = e.target.playerId; // 获取点击的 marker 的用户信息
  227. console.log("[Marker] Selected player ID: " + playerId)
  228. that.handleFocusPlayer(playerId)
  229. })
  230. var tooltip = marker.getTooltip()
  231. tooltip.playerId = player.id // 存储用户信息
  232. tooltip.animate = animate
  233. tooltip.on('click', function(e) {
  234. // console.log("[Marker.Tooltip]", e)
  235. var playerId = e.target.playerId; // 获取点击的 marker 的用户信息
  236. console.log("[Marker.Tooltip] Selected player ID: " + playerId)
  237. that.handleFocusPlayer(playerId)
  238. })
  239. var interval_creatCircleMarker = null
  240. if (animate) {
  241. interval_creatCircleMarker = this.animateCircle(marker, 5, 10, 0.5)
  242. }
  243. // this.playerLayerGroup.addTo(global.map)
  244. if (storePlayer == null) {
  245. storePlayer = {
  246. id: playerId,
  247. marker: marker, // 玩家标识
  248. markerLayerGroup: this.playerLayerGroup,
  249. trail: null, // 玩家轨迹 外层描边轨迹
  250. trail2: null, // 玩家轨迹 内层填充轨迹
  251. trailLayerGroup: null,
  252. trailData: [], // 玩家轨迹信息
  253. interval_creatCircleMarker: interval_creatCircleMarker,
  254. interval_showTrail: null,
  255. }
  256. global.storePlayers.push(storePlayer)
  257. } else {
  258. storePlayer.marker = marker
  259. storePlayer.markerLayerGroup = this.playerLayerGroup
  260. storePlayer.interval_creatCircleMarker = interval_creatCircleMarker
  261. }
  262. }
  263. // 非首次创建,直接移动位置
  264. else {
  265. if (!toggleTrailFlag) {
  266. var marker = storePlayer.marker
  267. var curPoint = [player_position.latitude, player_position.longitude]
  268. // console.log('[drawOnePlayer] 更新玩家的标记位置', storePlayer.marker, curPoint)
  269. marker.setLatLng(curPoint)
  270. marker.bringToFront()
  271. marker.setPopupContent(popupContent)
  272. }
  273. }
  274. },
  275. handleFocusPlayer(playerId) {
  276. console.log("[handleFocusPlayer] 当前选中玩家ID: " + playerId)
  277. if (global.focusPlayerId > 0) {
  278. // 先把之前选中的目标恢复成默认状态
  279. var unfocusPlayer = global.getStorePlayersById(global.focusPlayerId)
  280. // console.log("unfocusPlayer", unfocusPlayer)
  281. var playerColor = this.getPlayerColor(global.focusPlayerId)
  282. if (unfocusPlayer != null && unfocusPlayer.marker != null && unfocusPlayer.marker.type == 'circleMarker') {
  283. var options = config.gStyle.marker.default
  284. options.fillColor = playerColor
  285. unfocusPlayer.marker.setStyle(options)
  286. }
  287. if (unfocusPlayer != null && unfocusPlayer.trail != null) {
  288. var options = config.gStyle.trail.default
  289. options.color = playerColor
  290. unfocusPlayer.trail2.setStyle(options)
  291. }
  292. }
  293. global.focusPlayerId = playerId
  294. global._caller.focusPlayerId = playerId
  295. var focusPlayer = global.getStorePlayersById(global.focusPlayerId)
  296. // console.log("focusPlayer", focusPlayer)
  297. if (focusPlayer != null && focusPlayer.trail != null) {
  298. var options = config.gStyle.trail.focus
  299. focusPlayer.trail2.setStyle(options)
  300. focusPlayer.trail.bringToFront()
  301. focusPlayer.trail2.bringToFront()
  302. }
  303. if (focusPlayer != null && focusPlayer.marker != null && focusPlayer.marker.type == 'circleMarker') {
  304. var options = config.gStyle.marker.focus
  305. options.stroke = config.gStyle.common.stroke
  306. options.color = config.gStyle.common.color
  307. focusPlayer.marker.setStyle(options)
  308. focusPlayer.marker.bringToFront()
  309. }
  310. },
  311. drawAllTrails(duration) {
  312. // if (this.trailLayerGroup != null)
  313. // this.trailLayerGroup.clearLayers()
  314. for (let i = 0; i < global.players.length; i++) {
  315. this.drawOneTrail(global.players[i].id, duration)
  316. }
  317. },
  318. // 显示运动轨迹 duration:毫秒
  319. drawOneTrail(playerId, duration) {
  320. var player = global.getPlayerById(playerId)
  321. var player_position = global.getPlayerPositionById(playerId)
  322. var storePlayer = global.getStorePlayersById(playerId)
  323. // console.log('[drawOneTrail] param', player, player_position, storePlayer)
  324. if (player == null) {
  325. console.warn('[drawOneTrail] 玩家数据为空', player)
  326. return
  327. }
  328. if (player_position == null) {
  329. // console.log('[drawOneTrail] 玩家位置数据为空', player_position)
  330. return
  331. }
  332. if (storePlayer == null || storePlayer.trail == null) {
  333. // console.warn('[drawOneTrail] 轨迹数据为空', player, storePlayer)
  334. var trail = null
  335. var trail2 = null
  336. var curPoint = [player_position.latitude, player_position.longitude, new Date()]
  337. // console.log('[drawOneTrail] curPoint', curPoint)
  338. var trailData = [curPoint]
  339. var options = config.gStyle.trail.default
  340. // 外层描边轨迹
  341. if (config.gStyle.common.stroke) {
  342. options.weight = 6.9
  343. } else {
  344. options.weight = 0
  345. }
  346. options.color = config.gStyle.common.color
  347. trail = L.polyline(trailData, options)
  348. .addTo(this.trailLayerGroup)
  349. // console.log('[drawOneTrail] trail', trail)
  350. // 内层填充轨迹
  351. var playerColor = this.getPlayerColor(playerId)
  352. options.weight = 3.9
  353. options.color = playerColor
  354. // options.fillColor = playerColor
  355. trail2 = L.polyline(trailData, options)
  356. .addTo(this.trailLayerGroup)
  357. if (storePlayer == null) {
  358. storePlayer = {
  359. id: playerId,
  360. marker: null, // 玩家标识
  361. markerLayerGroup: null,
  362. trail: trail, // 玩家轨迹 外层描边轨迹
  363. trail2: trail2, // 玩家轨迹 内层填充轨迹
  364. trailLayerGroup: this.trailLayerGroup,
  365. trailData: trailData, // 玩家轨迹信息
  366. interval_creatCircleMarker: null,
  367. interval_showTrail: null,
  368. }
  369. global.storePlayers.push(storePlayer)
  370. } else {
  371. storePlayer.trail = trail
  372. storePlayer.trail2 = trail2
  373. storePlayer.trailLayerGroup = this.trailLayerGroup
  374. storePlayer.trailData = trailData
  375. }
  376. var that = this
  377. var interval_showTrail = setInterval(function() {
  378. // 去除过期的历史轨迹
  379. if (duration > 0) {
  380. var now = +new Date();
  381. storePlayer.trailData = storePlayer.trailData.filter(function(point) {
  382. return now - point[2] < duration;
  383. });
  384. }
  385. player_position = global.getPlayerPositionById(playerId)
  386. curPoint = [player_position.latitude, player_position.longitude, new Date()]
  387. storePlayer.trailData.push(curPoint)
  388. // storePlayer.trailData.push(curPoint.slice(0, 2))
  389. if (toggleTrailFlag) {
  390. // console.log('interval_showTrail', storePlayer.trailData)
  391. trail.setLatLngs(storePlayer.trailData);
  392. trail2.setLatLngs(storePlayer.trailData);
  393. if (storePlayer.marker != null) {
  394. if (toggleTrailFlag) {
  395. // console.warn('[drawOneTrail] 更新玩家的标记位置', storePlayer.marker, curPoint)
  396. storePlayer.marker.setLatLng(curPoint.slice(0, 2)) // 更新玩家的标记位置
  397. storePlayer.marker.bringToFront()
  398. var popupContent = that.getPopupContent(player)
  399. storePlayer.marker.setPopupContent(popupContent)
  400. }
  401. }
  402. // global.map.setView(point.slice(0, 2), 18); //地图中心跟踪最新位置
  403. }
  404. }, 500);
  405. storePlayer.interval_showTrail = interval_showTrail
  406. } else {
  407. // console.log('[drawOneTrail] 更新玩家轨迹', storePlayer, storePlayer.trail)
  408. }
  409. },
  410. getPlayerColor(playerId) {
  411. var colorIndex = playerId % 10
  412. return config.playerColorList[colorIndex]
  413. },
  414. // // duration: 轨迹保存时间 为0表示保存全部轨迹
  415. // getCurPos(lastpos, duration, player_trail) {
  416. // // console.log(lastpos, pointsT)
  417. // var now = +new Date();
  418. // if (duration > 0) {
  419. // player_trail.points = player_trail.points.filter(function(point) {
  420. // return now - point[2] < duration;
  421. // });
  422. // }
  423. // var step = 0.00001
  424. // var t = 10
  425. // var point = null
  426. // if (player_trail.pointsT < 20 * t)
  427. // point = [lastpos[0] + Math.random() * step, lastpos[1] + Math.random() * step, now];
  428. // else if (player_trail.pointsT >= 20 * t && player_trail.pointsT < 40 * t)
  429. // point = [lastpos[0] - Math.random() * step, lastpos[1] + Math.random() * step, now];
  430. // else if (player_trail.pointsT >= 40 * t && player_trail.pointsT < 60 * t)
  431. // point = [lastpos[0] - Math.random() * step, lastpos[1] - Math.random() * step, now];
  432. // else if (player_trail.pointsT >= 60 * t && player_trail.pointsT < 80 * t)
  433. // point = [lastpos[0] + Math.random() * step, lastpos[1] - Math.random() * step, now];
  434. // // point = [point[0] - Math.random() * 0.00001, point[1] - Math.random() * 0.00001, now];
  435. // player_trail.pointsT++
  436. // if (player_trail.pointsT >= 80 * t)
  437. // player_trail.pointsT = 0
  438. // player_trail.points.push(point);
  439. // return point
  440. // },
  441. free() {
  442. for (let i = 0; i < global.storePlayers.length; i++) {
  443. var storePlayer = global.storePlayers[i]
  444. if (storePlayer.interval_creatCircleMarker != null) {
  445. clearInterval(storePlayer.interval_creatCircleMarker)
  446. }
  447. if (storePlayer.interval_creatCircleMarker != null) {
  448. clearInterval(storePlayer.interval_showTrail)
  449. }
  450. }
  451. }
  452. }