audio.vue 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. <template>
  2. <view
  3. v-if="recordStatus != RecordStatus.HIDE"
  4. class="modal modal-record"
  5. @tap="toggleRecordModal"
  6. >
  7. <view class="modal-body" @tap.stop="toggleWithoutAction">
  8. <view class="sound-waves">
  9. <view
  10. v-for="(item, index) in radomheight"
  11. :key="index"
  12. :style="'height:' + item + 'rpx;margin-top:-' + item / 2 + 'rpx'"
  13. ></view>
  14. <view style="clear: both; width: 0; height: 0"></view>
  15. </view>
  16. <text class="desc">{{ RecordDesc[recordStatus] }}</text>
  17. <view
  18. class="dot"
  19. @touchstart="handleRecording"
  20. @touchmove="handleRecordingMove"
  21. @touchend="handleRecordingCancel"
  22. >
  23. <image class="icon-mic" src="/static/images/send.png"></image>
  24. </view>
  25. </view>
  26. </view>
  27. </template>
  28. <script>
  29. let WebIM = require("../../../../../utils/WebIM")["default"];
  30. let msgType = require("../../../msgtype");
  31. let RECORD_CONST = require("./record_status");
  32. let RecordStatus = RECORD_CONST.RecordStatus;
  33. let RecordDesc = RECORD_CONST.RecordDesc;
  34. let disp = require("../../../../../utils/broadcast");
  35. let msgStorage = require("../../../msgstorage");
  36. let RunAnimation = false;
  37. let recordTimeInterval = null;
  38. const InitHeight = [50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50];
  39. export default {
  40. data() {
  41. return {
  42. changedTouches: null,
  43. recordStatus: RecordStatus.HIDE,
  44. RecordStatus,
  45. RecordDesc,
  46. // 模板中有引用
  47. radomheight: InitHeight,
  48. recorderManager: uni.getRecorderManager(),
  49. recordClicked: false,
  50. isLongPress: false,
  51. recordTime: 0
  52. };
  53. },
  54. components: {},
  55. props: {
  56. username: {
  57. type: Object,
  58. default: () => ({})
  59. },
  60. chatType: {
  61. type: String,
  62. default: msgType.chatType.SINGLE_CHAT
  63. }
  64. },
  65. // lifetimes
  66. created() {},
  67. beforeMount() {},
  68. moved() {},
  69. destroyed() {
  70. clearInterval(recordTimeInterval)
  71. this.recordTime = 0
  72. },
  73. mounted() {},
  74. methods: {
  75. toggleWithoutAction(e) {// 阻止 tap 冒泡
  76. },
  77. toggleRecordModal() {
  78. this.recordStatus = this.recordStatus == RecordStatus.HIDE ? RecordStatus.SHOW : RecordStatus.HIDE
  79. this.radomheight = InitHeight
  80. },
  81. handleRecordingMove(e) {
  82. var touches = e.touches[0];
  83. var changedTouches = this.changedTouches;
  84. if (!changedTouches) {
  85. return;
  86. }
  87. if (this.recordStatus == RecordStatus.SWIPE) {
  88. if (changedTouches.pageY - touches.pageY < 20) {
  89. this.recordStatus = RecordStatus.HOLD
  90. }
  91. }
  92. if (this.recordStatus == RecordStatus.HOLD) {
  93. if (changedTouches.pageY - touches.pageY > 20) {
  94. this.recordStatus = RecordStatus.SWIPE
  95. }
  96. }
  97. },
  98. handleRecording(e) {
  99. console.log('开始点击',uni.getSystemInfoSync())
  100. if(uni.getSystemInfoSync().app === "alipay"){
  101. // https://forum.alipay.com/mini-app/post/7301031?ant_source=opendoc_recommend
  102. uni.showModal({
  103. content: '支付宝小程序不支持语音消息,请查看支付宝相关api了解详情'
  104. })
  105. return
  106. }
  107. let me = this;
  108. me.recordClicked = true
  109. setTimeout(() => {
  110. if (me.recordClicked == true) {
  111. executeRecord();
  112. }
  113. }, 350);
  114. function executeRecord() {
  115. if (uni.getSetting) {
  116. uni.getSetting({
  117. success: res => {
  118. clearInterval(recordTimeInterval);
  119. me.recordTime = 0
  120. let recordAuth = res.authSetting['scope.record'];
  121. if (recordAuth == false) {
  122. //已申请过授权,但是用户拒绝
  123. uni.openSetting({
  124. success: function (res) {
  125. let recordAuth = res.authSetting['scope.record'];
  126. if (recordAuth == true) {
  127. uni.showToast({
  128. title: "授权成功",
  129. icon: "success"
  130. });
  131. } else {
  132. uni.showToast({
  133. title: "请授权录音",
  134. icon: "none"
  135. });
  136. }
  137. me.isLongPress = false
  138. }
  139. });
  140. } else if (recordAuth == true) {
  141. // 用户已经同意授权
  142. startRecord();
  143. } else {
  144. // 第一次进来,未发起授权
  145. uni.authorize({
  146. scope: 'scope.record',
  147. success: () => {
  148. //授权成功
  149. uni.showToast({
  150. title: "授权成功",
  151. icon: "success"
  152. });
  153. }
  154. });
  155. }
  156. },
  157. fail: function () {
  158. uni.showToast({
  159. title: "鉴权失败,请重试",
  160. icon: "none"
  161. });
  162. }
  163. })
  164. return
  165. }else{
  166. startRecord()
  167. return
  168. }
  169. }
  170. function startRecord() {
  171. clearInterval(recordTimeInterval);
  172. me.recordTime = 0
  173. me.changedTouches = e.touches[0];
  174. me.recordStatus = RecordStatus.HOLD
  175. RunAnimation = true;
  176. me.myradom();
  177. let recorderManager = me.recorderManager || uni.getRecorderManager();
  178. recorderManager.onStart(() => {
  179. // console.log("开始录音...");
  180. recordTimeInterval = setInterval(()=>{
  181. me.recordTime ++
  182. },1000)
  183. });
  184. recorderManager.start({
  185. format: "mp3"
  186. }); // 超时
  187. setTimeout(function () {
  188. me.handleRecordingCancel();
  189. RunAnimation = false;
  190. }, 100000);
  191. }
  192. },
  193. handleRecordingCancel() {
  194. RunAnimation = false;
  195. let recorderManager = this.recorderManager; // 向上滑动状态停止:取消录音发放
  196. if (this.recordStatus == RecordStatus.SWIPE) {
  197. this.recordStatus = RecordStatus.RELEASE
  198. } else {
  199. this.recordStatus = RecordStatus.HIDE
  200. this.recordClicked = false
  201. }
  202. recorderManager.onStop(res => {
  203. // console.log("结束录音...", res);
  204. clearInterval(recordTimeInterval);
  205. let duration = this.recordTime * 1000;
  206. if (this.recordStatus == RecordStatus.RELEASE) {
  207. console.log("user canceled");
  208. this.recordStatus = RecordStatus.HIDE
  209. return;
  210. }
  211. if (duration <= 1000) {
  212. uni.showToast({
  213. title: "录音时间太短",
  214. icon: "none"
  215. });
  216. } else {
  217. // 上传
  218. this.uploadRecord(res.tempFilePath, duration);
  219. }
  220. clearInterval(recordTimeInterval);
  221. this.recordStatus = RecordStatus.HIDE
  222. this.recordTime = 0;
  223. }); // 停止录音
  224. recorderManager.stop();
  225. },
  226. isGroupChat() {
  227. return this.chatType == msgType.chatType.CHAT_ROOM;
  228. },
  229. getSendToParam() {
  230. return this.isGroupChat() ? this.username.groupId : this.username.your;
  231. },
  232. uploadRecord(tempFilePath, dur) {
  233. var str = WebIM.config.appkey.split("#");
  234. var me = this;
  235. var token = WebIM.conn.context.accessToken;
  236. uni.uploadFile({
  237. url: "https://a1.easemob.com/" + str[0] + "/" + str[1] + "/chatfiles",
  238. filePath: tempFilePath,
  239. fileType: 'audio',
  240. name: "file",
  241. header: {
  242. "Content-Type": "multipart/form-data",
  243. Authorization: "Bearer " + token
  244. },
  245. success(res) {
  246. var id = WebIM.conn.getUniqueId();
  247. var msg = new WebIM.message(msgType.AUDIO, id);
  248. var dataObj = JSON.parse(res.data); // 接收消息对象
  249. msg.set({
  250. apiUrl: WebIM.config.apiURL,
  251. accessToken: token,
  252. body: {
  253. type: msgType.AUDIO,
  254. url: dataObj.uri + "/" + dataObj.entities[0].uuid,
  255. filetype: "",
  256. filename: tempFilePath,
  257. accessToken: token,
  258. length: Math.ceil(dur / 1000)
  259. },
  260. from: me.username.myName,
  261. to: me.getSendToParam(),
  262. roomType: false,
  263. chatType: me.chatType,
  264. success: function (argument) {
  265. disp.fire('em.chat.sendSuccess', id);
  266. }
  267. });
  268. if (me.isGroupChat()) {
  269. msg.setGroup("groupchat");
  270. }
  271. msg.body.length = Math.ceil(dur / 1000); //console.log('发送的语音消息', msg.body)
  272. WebIM.conn.send(msg.body);
  273. let obj = {
  274. msg: msg,
  275. type: msgType.AUDIO
  276. }
  277. me.saveSendMsg(obj);
  278. }
  279. });
  280. },
  281. saveSendMsg(evt) {
  282. msgStorage.saveMsg(evt.msg, evt.type);
  283. },
  284. myradom() {
  285. const that = this;
  286. var _radomheight = that.radomheight;
  287. for (var i = 0; i < that.radomheight.length; i++) {
  288. //+1是为了避免为0
  289. _radomheight[i] = 100 * Math.random().toFixed(2) + 10;
  290. }
  291. that.radomheight = _radomheight
  292. if (RunAnimation) {
  293. setTimeout(function () {
  294. that.myradom();
  295. }, 500);
  296. } else {
  297. return;
  298. }
  299. }
  300. }
  301. };
  302. </script>
  303. <style>
  304. @import "./audio.css";
  305. </style>