map-sense.js 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  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 = 3.3
  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. // 筛选器,筛选出配对的一组信标
  134. let signSelect = [0,0,0]; //记录各组的信标出现数量,投票选择最多的,以此是prologue,annexHall,mainHall
  135. let prologue = ['10001','10002'] //序厅
  136. let annexHall = ['10003','10004'] //附厅
  137. let mainHall = ['10005','10006'] //主厅
  138. const R = 6 //设定范围半径
  139. Object.keys(accuracyList).forEach(key=>{
  140. let aveAccuracy = BeaconUtils.arrayAverage(accuracyList[key])
  141. // console.log(key,aveAccuracy)
  142. // let record = {'id':'10001','distance':aveAccuracy}
  143. // json.push(record)
  144. // if(json.length >= 350){
  145. // console.log(JSON.stringify(json))
  146. // }
  147. if(aveAccuracy < R){
  148. if(prologue.includes(key)){
  149. signSelect[0] += 1;
  150. }else if(annexHall.includes(key)){
  151. signSelect[1] += 1;
  152. }else if(mainHall.includes(key)){
  153. signSelect[2] += 1;
  154. }
  155. }
  156. })
  157. //对小于预设半径的信号进行数量投票
  158. let result = BeaconUtils.maxIndex(signSelect);
  159. console.log('result',result);
  160. this.setData({
  161. classfiy,
  162. status:'2'
  163. })
  164. })
  165. }).catch(()=>{
  166. wx.showToast({
  167. title: '连接失败',
  168. icon: 'error',
  169. duration: 500
  170. })
  171. })
  172. })
  173. },
  174. stopBeaconDiscovery() {
  175. var that = this;
  176. console.log('这是取消连接')
  177. wx.showToast({
  178. title: '取消连接成功',
  179. icon: 'error',
  180. duration: 1000
  181. })
  182. wx.stopBeaconDiscovery({})
  183. // 取消连接 停止播放音乐 目标对象置为{} 设置为第一次进入
  184. innerAudioContext.stop();
  185. that.setData({
  186. status: '0',
  187. targetObj: {}
  188. })
  189. },
  190. // 扫一扫
  191. toScanCode() {
  192. var that = this;
  193. that.setData({
  194. isScanPlay: true
  195. })
  196. wx.scanCode({
  197. success(res) {
  198. console.log('success', res)
  199. console.log('result', res['result'])
  200. console.log('innerAudioContext', innerAudioContext)
  201. that.setData({ targetObj: {} })
  202. innerAudioContext.autoplay = true;
  203. innerAudioContext.src = res['result']
  204. innerAudioContext.loop = true;
  205. innerAudioContext.play();
  206. },
  207. fail: function (res) {
  208. console.log('fail innerAudioContext', innerAudioContext)
  209. that.setData({
  210. isScanPlay: false
  211. })
  212. innerAudioContext.play();
  213. return;
  214. },
  215. complete(){
  216. // console.log()
  217. that.setData({
  218. isScanPlay: false
  219. })
  220. }
  221. })
  222. },
  223. getAudios() {
  224. http.get('/api/web/getAudioIndex')
  225. .then(res => {
  226. let { data } = res,target = {};
  227. data.forEach(item => {
  228. switch (item.type) {
  229. case 1:
  230. target['10001'] = `${this.data.api_base_url}/data/${item.audio}`;
  231. break;
  232. case 2:
  233. target['10002'] = `${this.data.api_base_url}/data/${item.audio}`;
  234. break;
  235. case 3:
  236. target['10003'] = `${this.data.api_base_url}/data/${item.audio}`;
  237. break;
  238. default:
  239. break
  240. }
  241. })
  242. this.setData({
  243. audio_address: target
  244. })
  245. })
  246. }
  247. },
  248. lifetimes:{
  249. attached: function () {
  250. innerAudioContext.stop();
  251. //获取语音
  252. this.getAudios();
  253. var that = this;
  254. // wx.onAccelerometerChange(function (e) {
  255. // console.log('手机咚咚咚给')
  256. // if (Math.abs(e.x) > 1.1 && Math.abs(e.y) > 1.1) {
  257. // //         wx.showToast({ title: "摇一摇" })
  258. // } else if (Math.abs(e.x) > 0.07 && Math.abs(e.y) > 0.02 && that.data.status === '2') {
  259. // // 扫一扫播放的话 移动无效
  260. // if (that.data.isScanPlay) return;
  261. // that.startBeaconDiscovery()
  262. // } else {
  263. // //         wx.showToast({ title: "静止" })
  264. // }
  265. // }),
  266. innerAudioContext.onEnded(() => {
  267. console.log('播放结束了')
  268. if (this.data.isScanPlay) {
  269. innerAudioContext.stop()
  270. this.setData({ isScanPlay: false })
  271. this.targetObj = {}
  272. console.log('innerAudioContext', innerAudioContext)
  273. if (this.data.status == 2) {
  274. console.log(2222222222222222)
  275. this.startBeaconDiscovery()
  276. }
  277. }
  278. })
  279. },
  280. detached: function () {
  281. innerAudioContext.stop();
  282. innerAudioContext.destroy();
  283. wx.stopBeaconDiscovery({})
  284. this.setData({status:"0"})
  285. }
  286. }
  287. })