map-sense.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. // components/map-sense.js
  2. const innerAudioContext = wx.createInnerAudioContext();
  3. import http from '../../utils/http';
  4. import {promisify,BeaconUtils} from '../../utils/util';
  5. let openBluetoothAdapter = promisify(wx.openBluetoothAdapter)
  6. let startBeaconDiscovery = promisify(wx.startBeaconDiscovery)
  7. import { CDN_URL, CONNECT_STATUS, STATUS_TEXT, API_BASE_URL } from '../../config/index';
  8. const STATUS_PIC = {
  9. '0': 'default',
  10. '1': 'loading',
  11. '2': 'success',
  12. '3': 'fail'
  13. };
  14. // 选择6档
  15. const TXPOWER = -55
  16. // 距离经验值(调试所得)
  17. const N = 4
  18. let AveLength = 10
  19. Component({
  20. /**
  21. * 组件的属性列表
  22. */
  23. properties: {
  24. },
  25. /**
  26. * 组件的初始数据
  27. */
  28. data: {
  29. status: "0",
  30. cdn_url: CDN_URL,
  31. connect_status: CONNECT_STATUS,
  32. status_text: STATUS_TEXT,
  33. status_pic: STATUS_PIC,
  34. targetObj: {},
  35. audio_address: {},
  36. api_base_url: API_BASE_URL,
  37. // 是否是扫码播放
  38. isScanPlay: false,
  39. cdn_url:CDN_URL,
  40. classfiy:{}
  41. },
  42. /**
  43. * 组件的方法列表
  44. */
  45. methods: {
  46. openBluetooth(cb) {
  47. openBluetoothAdapter().then(res=>{
  48. cb(res)
  49. },()=>{
  50. wx.showToast({
  51. title: '请打开蓝牙',
  52. icon: 'error',
  53. duration: 2000
  54. })
  55. })
  56. },
  57. toHandle() {
  58. let aveArr = []
  59. let json = []
  60. this.openBluetooth(()=>{
  61. startBeaconDiscovery({uuids: ['FDA50693-A4E2-4FB1-AFCF-C6EB07647825']}).then((res)=>{
  62. wx.onBeaconUpdate(data=>{
  63. //打印最新收集到的信号
  64. // console.log('beacon',data.beacons)
  65. // 需要收集十组数据,索引号最大的组是最旧的
  66. if (aveArr.length > AveLength) {
  67. // aveArr[0] = data.beacons
  68. //当超过十组时,应该将索引号大的组淘汰掉
  69. aveArr.pop()
  70. }
  71. aveArr.unshift(data.beacons) //在队列前面插入
  72. // console.log('aveArr',aveArr);
  73. let all = []
  74. aveArr.forEach(item => {
  75. // console.log(aveArr.length,item)
  76. all = all.concat(item)
  77. });
  78. // classfiy = {10003:[{},{},···],10002:[{},{},···],10001:[{},{},···]}
  79. // console.log('all',all)
  80. let classfiy = BeaconUtils.classification(all,'major')
  81. let accuracyList = {} // 存放处理后的accuary
  82. // let dataArr = [] //记录各信标的信号收集数目
  83. Object.keys(classfiy).forEach(key=>{
  84. //每个major的AveLength个元素数组,元素为rssi
  85. let arr = classfiy[key].map(item=>{
  86. return item.rssi
  87. })
  88. // console.log(key,'rssi',arr)
  89. //每个major的rssi平均值
  90. let ave = BeaconUtils.arrayAverage(arr)
  91. // console.log('ave',ave)
  92. //计算平均差
  93. let meanDeviation = BeaconUtils.getVariance(arr,ave); //改成方差
  94. // console.log('meanDeviation',meanDeviation)
  95. //计算各rssi的高斯模糊权重
  96. let guassionArr = [];
  97. for(let i = 0; i < arr.length; ++i){
  98. // console.log('guassionArr ele',i,BeaconUtils.getOneGuassionArray(arr.length,i,meanDeviation))
  99. // 返回的可能是空数组
  100. guassionArr.push(BeaconUtils.getOneGuassionArray(arr.length,i,meanDeviation));
  101. }
  102. // console.log('guassionArr',guassionArr)
  103. //计算高斯模糊后的rssi值
  104. let rssiArr = []; //模糊后的rssi数组
  105. for(let i = 0; i < arr.length; ++i){
  106. let sum = 0;
  107. for(let j = 0; j <arr.length; ++j){
  108. if(guassionArr[i].length == 0){
  109. sum = arr[j]
  110. break;
  111. }
  112. sum += guassionArr[i][j] * arr[j]
  113. }
  114. rssiArr.push(sum);
  115. // console.log('sum',sum)
  116. }
  117. // console.log('rssiArr',rssiArr);
  118. //时间加权后求rssi平均值
  119. let aveOnTime = BeaconUtils.arrayAverage(rssiArr.slice(0,Math.floor(rssiArr.length / 3)).concat(rssiArr));
  120. // console.log('aveOnTime',aveOnTime)
  121. //测距
  122. classfiy[key].forEach(item=>{
  123. item.accuracy = BeaconUtils.calculateAccuracy(TXPOWER,aveOnTime,N)
  124. if(!accuracyList[key]){
  125. //如果还没有对应的“信标号”
  126. accuracyList[key] = [item.accuracy]
  127. }
  128. accuracyList[key].push(item.accuracy)
  129. // console.log('item',item)
  130. })
  131. // dataArr.push(classfiy[key].length)
  132. })
  133. // console.log('10001',classfiy['10001'])
  134. // console.log('10002',classfiy['10002'])
  135. // console.log('10003',classfiy['10003'])
  136. // console.log(classfiy)
  137. // 筛选器,筛选出配对的一组信标
  138. let aveAccuracys = new Map() //信标的平均距离,用于判断是否在范围内
  139. let matchSigns = [] //匹配的信号
  140. let prologue = ['1000110002'] //序厅
  141. let annexHall = ['1000310004'] //附厅
  142. let mainHall = ['1000510006'] //主厅
  143. let area = [prologue,annexHall,mainHall]
  144. const R = 3 //设定范围半径
  145. Object.keys(accuracyList).forEach(key=>{
  146. let aveAccuracy = BeaconUtils.arrayAverage(accuracyList[key])
  147. // console.log(key,aveAccuracy)
  148. let record = {'id':'10001','distance':aveAccuracy}
  149. json.push(record)
  150. if(json.length == 500){
  151. console.log(JSON.stringify(json))
  152. }
  153. if(aveAccuracy < R){
  154. aveAccuracys.set(key,aveAccuracy)
  155. }
  156. })
  157. //理想情况下只有两个信标的信号,刚好匹配为一组
  158. //判断编号是奇数偶数
  159. let result = ''
  160. for(let key of aveAccuracys.keys()){
  161. let num = parseInt(key)
  162. if(num % 2 == 0){
  163. if(aveAccuracys.has(String(num - 1))){
  164. num--
  165. result = key + num
  166. break
  167. }
  168. }else if(aveAccuracys.has(String(num + 1))){
  169. num++
  170. result = key + num
  171. break
  172. }
  173. }
  174. console.log('result',result);
  175. for(let i = 0; i < area.length; ++i){
  176. if(area[i].includes(result)){
  177. console.log(i)
  178. }
  179. }
  180. this.setData({
  181. classfiy,
  182. status:'2'
  183. })
  184. })
  185. }).catch(()=>{
  186. wx.showToast({
  187. title: '连接失败',
  188. icon: 'error',
  189. duration: 500
  190. })
  191. })
  192. })
  193. },
  194. stopBeaconDiscovery() {
  195. var that = this;
  196. console.log('这是取消连接')
  197. wx.showToast({
  198. title: '取消连接成功',
  199. icon: 'error',
  200. duration: 1000
  201. })
  202. wx.stopBeaconDiscovery({})
  203. // 取消连接 停止播放音乐 目标对象置为{} 设置为第一次进入
  204. innerAudioContext.stop();
  205. that.setData({
  206. status: '0',
  207. targetObj: {}
  208. })
  209. },
  210. // 扫一扫
  211. toScanCode() {
  212. var that = this;
  213. that.setData({
  214. isScanPlay: true
  215. })
  216. wx.scanCode({
  217. success(res) {
  218. console.log('success', res)
  219. console.log('result', res['result'])
  220. console.log('innerAudioContext', innerAudioContext)
  221. that.setData({ targetObj: {} })
  222. innerAudioContext.autoplay = true;
  223. innerAudioContext.src = res['result']
  224. innerAudioContext.loop = true;
  225. innerAudioContext.play();
  226. },
  227. fail: function (res) {
  228. console.log('fail innerAudioContext', innerAudioContext)
  229. that.setData({
  230. isScanPlay: false
  231. })
  232. innerAudioContext.play();
  233. return;
  234. },
  235. complete(){
  236. // console.log()
  237. that.setData({
  238. isScanPlay: false
  239. })
  240. }
  241. })
  242. },
  243. getAudios() {
  244. http.get('/api/web/getAudioIndex')
  245. .then(res => {
  246. let { data } = res,target = {};
  247. data.forEach(item => {
  248. switch (item.type) {
  249. case 1:
  250. target['10001'] = `${this.data.api_base_url}/data/${item.audio}`;
  251. break;
  252. case 2:
  253. target['10002'] = `${this.data.api_base_url}/data/${item.audio}`;
  254. break;
  255. case 3:
  256. target['10003'] = `${this.data.api_base_url}/data/${item.audio}`;
  257. break;
  258. default:
  259. break
  260. }
  261. })
  262. this.setData({
  263. audio_address: target
  264. })
  265. })
  266. }
  267. },
  268. lifetimes:{
  269. attached: function () {
  270. innerAudioContext.stop();
  271. //获取语音
  272. this.getAudios();
  273. var that = this;
  274. // wx.onAccelerometerChange(function (e) {
  275. // console.log('手机咚咚咚给')
  276. // if (Math.abs(e.x) > 1.1 && Math.abs(e.y) > 1.1) {
  277. // //         wx.showToast({ title: "摇一摇" })
  278. // } else if (Math.abs(e.x) > 0.07 && Math.abs(e.y) > 0.02 && that.data.status === '2') {
  279. // // 扫一扫播放的话 移动无效
  280. // if (that.data.isScanPlay) return;
  281. // that.startBeaconDiscovery()
  282. // } else {
  283. // //         wx.showToast({ title: "静止" })
  284. // }
  285. // }),
  286. innerAudioContext.onEnded(() => {
  287. console.log('播放结束了')
  288. if (this.data.isScanPlay) {
  289. innerAudioContext.stop()
  290. this.setData({ isScanPlay: false })
  291. this.targetObj = {}
  292. console.log('innerAudioContext', innerAudioContext)
  293. if (this.data.status == 2) {
  294. console.log(2222222222222222)
  295. this.startBeaconDiscovery()
  296. }
  297. }
  298. })
  299. },
  300. detached: function () {
  301. innerAudioContext.stop();
  302. innerAudioContext.destroy();
  303. wx.stopBeaconDiscovery({})
  304. this.setData({status:"0"})
  305. }
  306. }
  307. })