chat.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. <template>
  2. <!-- 聊天记录 -->
  3. <view class="chat">
  4. <view class="chat-list">
  5. <view class="more-btn" @click="tomore">
  6. {{showmore ? '点击查看更多' : '没有更多了'}}
  7. </view>
  8. <view v-for="(item,index) in list" :key="index">
  9. <view class="u-flex u-row-between" style="margin-bottom: 10rpx;">
  10. <text :style="{opacity:item.type == 2 ? 1 : 0}"
  11. style="margin-bottom: -20rpx;">{{item.create_at}}</text>
  12. <text :style="{opacity:item.type == 2 ? 0: 1}">{{item.create_at}}</text>
  13. </view>
  14. <view class="chat-item u-flex u-col-top u-row-between">
  15. <image :src="item.headimg" :style="{opacity:item.type == 2 ? 1 : 0}" class="user-head" mode="">
  16. </image>
  17. <view class="item-box u-flex-1 u-flex-col"
  18. :style="{alignItems: item.type == 2 ? 'flex-start' : 'flex-end'}">
  19. <text>{{item.name}}</text>
  20. <text v-if="item.is_url == 0"
  21. :class="item.type == 2 ? 'text1' : 'text2'">{{item.content}}</text>
  22. <view class="" v-if="item.is_url == '1'" @tap="onPlay(item,index)"
  23. :class="item.type == 2 ? 'back1' : 'back2'">
  24. <view class="record" v-if="item.type==1">
  25. <text style="#fff">{{item.time}}</text> "
  26. <u-icon custom-style="transform: rotate(-90deg) !important;" name="wifi"
  27. size="28"></u-icon>
  28. </view>
  29. <view class="record1" v-if="item.type==2">
  30. <u-icon custom-style="transform: rotate(90deg) !important;" name="wifi"
  31. size="28"></u-icon>
  32. "
  33. <text>{{item.time}}</text>
  34. </view>
  35. </view>
  36. </view>
  37. <image :src="item.headimg" :style="{opacity:item.type == 2 ? 0 : 1}" class="user-head" mode="">
  38. </image>
  39. </view>
  40. </view>
  41. <view class="" v-for="(item,index) in chatlist" :key="index">
  42. <view class="u-flex u-row-between" style="margin-bottom: 10rpx;">
  43. <text :style="{opacity:item.type == 2 ? 1 : 0}"
  44. style="margin-bottom: -20rpx;">{{item.create_at}}</text>
  45. <text :style="{opacity:item.type == 2 ? 0: 1}">{{item.create_at}}</text>
  46. </view>
  47. <view class="chat-item u-flex u-col-top u-row-between">
  48. <image :src="item.headimg" :style="{opacity:item.type == 2 ? 1 : 0}" class="user-head" mode="">
  49. </image>
  50. <view class="item-box u-flex-1 u-flex-col"
  51. :style="{alignItems: item.type == 2 ? 'flex-start' : 'flex-end'}">
  52. <text>{{item.name}}</text>
  53. <text v-if="item.type1 != 'record'"
  54. :class="item.type == 2 ? 'text1' : 'text2'">{{item.content}}</text>
  55. <view class="record" v-if="item.type1 == 'record'" :class="item.type == 2 ? 'text1' : 'text2'"
  56. @tap="onPlay1(item,index)">
  57. <view class="record" v-if="item.type==1">
  58. <text>{{item.message}}</text>"<u-icon name="wifi" size="28"></u-icon>
  59. </view>
  60. <view class="record1" v-if="item.type==2">
  61. <u-icon name="wifi" size="28"></u-icon>
  62. "
  63. <text>{{item.message}}</text>
  64. </view>
  65. </view>
  66. </view>
  67. <image :src="item.headimg" :style="{opacity:item.type == 2 ? 0 : 1}" class="user-head" mode="">
  68. </image>
  69. </view>
  70. </view>
  71. </view>
  72. <view class="" style="height: 170rpx;"></view>
  73. <view class="chat-btn u-flex u-row-between">
  74. <u-icon v-if="show == false" @click="startRecord" name="mic" size="44"></u-icon>
  75. <u-icon v-if="show == true" @click="startRecord1" name="more-circle-fill" size="44"></u-icon>
  76. <nb-voice-record v-if="show == true" :vibrate="false" @startRecord="start" @endRecord="end"
  77. @cancelRecord="cancel"></nb-voice-record>
  78. <input v-if="show == false" type="text" placeholder="请输入" v-model="text">
  79. <text v-if="show == false" class="post" @click="send(text)">发送</text>
  80. </view>
  81. </view>
  82. </template>
  83. <script>
  84. export default {
  85. data() {
  86. return {
  87. bgAudioManager: '', // 全局音频播放
  88. show: false,
  89. text: '',
  90. hx_username: '',
  91. worker_id: '',
  92. page: 1,
  93. list: [],
  94. chatlist: [],
  95. myname: '',
  96. myheadimg: '',
  97. workername: '',
  98. workerheadimg: '',
  99. voicePath: '',
  100. userinfoheadimg: "",
  101. ossdata: {},
  102. duration: 0,
  103. time1: 0,
  104. url: "",
  105. url1: ""
  106. }
  107. },
  108. onLoad(option) {
  109. this.hx_username = option.hx_username
  110. this.worker_id = option.worker_id
  111. console.log(this.worker_id);
  112. console.log(this.hx_username);
  113. this.myname = uni.getStorageSync("name")
  114. this.myheadimg = uni.getStorageSync("headimg")
  115. this.getlist()
  116. this.getworker()
  117. this.getmsg()
  118. let self = this;
  119. this.getoss()
  120. // this.getaudio()
  121. },
  122. computed: {
  123. showmore() {
  124. if (this.list.length % 20 == 0 && this.list.length > 0) {
  125. return true
  126. } else {
  127. return false
  128. }
  129. }
  130. },
  131. methods: {
  132. onPlay(e, index) {
  133. let that = this
  134. if (this.url && this.url == e.content) return
  135. if (this.url && this.url !== e.content) {
  136. this.bgAudioManager.stop()
  137. let obj = this.list.find(item => item.content == this.url)
  138. clearTimeout(obj.timeOut)
  139. obj.time = obj.defaultTime
  140. }
  141. this.url = e.content
  142. // 我这里没有写暂停播放 需要的可以 设置变量 根据变量状态判断 pause 或 play
  143. this.bgAudioManager = uni.createInnerAudioContext()
  144. this.bgAudioManager.src = e.content
  145. this.bgAudioManager.play()
  146. this.time1 = e.time
  147. var music = wx.setInnerAudioOption({
  148. obeyMuteSwitch: false,
  149. success: function(res) {
  150. console.log("开启静音模式下播放音乐的功能");
  151. },
  152. fail: function(err) {
  153. console.log(err);
  154. console.log("静音设置失败");
  155. },
  156. });
  157. this.bgAudioManager.onPlay(() => {
  158. function time_a(obj) {
  159. obj.timeOut = setTimeout(() => {
  160. obj.time--
  161. obj.time < 0 ? obj.time = that.time1 : time_a(obj)
  162. }, 1000)
  163. }
  164. time_a(this.list[index])
  165. })
  166. this.bgAudioManager.onEnded(() => {
  167. this.url = ''
  168. // this.list[index].time = this.time1
  169. this.bgAudioManager.offPlay()
  170. this.bgAudioManager.offEnded()
  171. this.bgAudioManager = null
  172. })
  173. },
  174. onPlay1(e, index) {
  175. let that = this
  176. if (this.url1 && this.url1 == e.content) return
  177. if (this.url1 && this.url1 !== e.content) {
  178. this.bgAudioManager.stop()
  179. let obj = this.chatlist.find(item => item.content == this.url1)
  180. clearTimeout(obj.timeOut)
  181. obj.message = obj.defaultTime
  182. }
  183. this.url1 = e.content
  184. this.bgAudioManager = uni.createInnerAudioContext()
  185. this.bgAudioManager.src = e.content
  186. this.bgAudioManager.play()
  187. // this.bgAudioManager.obeyMuteSwitch = false
  188. this.time1 = e.message
  189. console.log(that.time1);
  190. var music = wx.setInnerAudioOption({
  191. obeyMuteSwitch: false,
  192. success: function(res) {},
  193. fail: function(err) {},
  194. });
  195. this.bgAudioManager.onPlay(() => {
  196. function time_a(obj) {
  197. obj.timeOut = setTimeout(() => {
  198. obj.message--
  199. // obj.message < 0 ? obj.message = that.time1 : time_a(obj)
  200. if (obj.message < 0) {
  201. obj.message = that.time1
  202. clearTimeout(obj.timeOut)
  203. that.bgAudioManager.offPlay()
  204. that.bgAudioManager.offEnded()
  205. that.bgAudioManager = null
  206. that.url1 = ''
  207. } else {
  208. // console.log(time_a(obj));
  209. // time_a(obj)
  210. obj.message = obj.message
  211. time_a(obj)
  212. }
  213. }, 1000)
  214. }
  215. time_a(this.chatlist[index])
  216. })
  217. },
  218. start() {
  219. // 开始录音
  220. },
  221. getoss() {
  222. this.$u.post('/api/Upload/getSignedUrl').then(res => {
  223. this.ossdata = res.data
  224. })
  225. },
  226. end(event) {
  227. // 结束录音并处理得到的录音文件
  228. // event中,app端仅有tempFilePath字段,微信小程序还有duration和fileSize两个字段
  229. // this.sendPrivateAudio(event.tempFilePath,event.duration)
  230. if (event.duration < 1000) {
  231. this.$u.toast("说话时间太短了")
  232. } else {
  233. this.duration = event.duration
  234. var key = this.ossdata.key + new Date().getTime() + Math.floor(Math.random() * 150) + '.mp3'
  235. uni.uploadFile({
  236. url: this.ossdata.host, //输入你的bucketname.endpoint
  237. filePath: event.tempFilePath,
  238. name: 'file',
  239. formData: {
  240. key: key,
  241. policy: this.ossdata.policy, // 输入你获取的的policy
  242. OSSAccessKeyId: this.ossdata.OSSAccessKeyId, // 输入你的AccessKeyId
  243. success_action_status: '200', // 让服务端返回200,不然,默认会返回204
  244. signature: this.ossdata.Signature, // 输入你获取的的signature
  245. },
  246. success: (res) => {
  247. if (res.statusCode == 200) {
  248. this.userinfoheadimg = this.ossdata.host + '/' + key
  249. this.send1(this.userinfoheadimg, this.duration)
  250. }
  251. },
  252. fail: (err) => {
  253. console.log(err);
  254. }
  255. })
  256. }
  257. },
  258. cancel() {
  259. // 用户取消录音
  260. console.log(3333);
  261. },
  262. startRecord() {
  263. this.show = true
  264. },
  265. startRecord1() {
  266. this.show = false
  267. },
  268. tomore() {
  269. if (this.showmore) {
  270. this.page++
  271. this.getlist()
  272. }
  273. },
  274. getmsg() {
  275. this.$WebIM.conn.addEventHandler('getmsg', {
  276. // 当前用户收到文本消息。
  277. onTextMessage: (message) => {
  278. console.log(message);
  279. this.chatlist.push({
  280. content: message.msg,
  281. create_at: this.$u.timeFormat(message.time, 'yyyy-mm-dd hh:MM:ss'),
  282. headimg: message.from == uni.getStorageSync("hx_username") ? this
  283. .myheadimg : this.workerheadimg,
  284. name: message.from == uni.getStorageSync("hx_username") ? this.myname :
  285. this.workername,
  286. type: message.from == uni.getStorageSync("hx_username") ? 1 : 2,
  287. })
  288. this.$nextTick(() => {
  289. uni.pageScrollTo({
  290. scrollTop: 99999
  291. })
  292. })
  293. },
  294. onAudioMessage: (message) => {
  295. this.chatlist.push({
  296. // defaultTime:message.length,
  297. content: message.url,
  298. create_at: this.$u.timeFormat(message.time, 'yyyy-mm-dd hh:MM:ss'),
  299. headimg: message.from == uni.getStorageSync("hx_username") ? this
  300. .myheadimg : this.workerheadimg,
  301. name: message.from == uni.getStorageSync("hx_username") ? this.myname :
  302. this.workername,
  303. type: message.from == uni.getStorageSync("hx_username") ? 1 : 2,
  304. type1: "record",
  305. message: message.length
  306. })
  307. this.$nextTick(() => {
  308. uni.pageScrollTo({
  309. scrollTop: 99999
  310. })
  311. })
  312. },
  313. })
  314. },
  315. getworker() {
  316. this.$u.post('/api/News/get_worker_info', {
  317. worker_id: this.worker_id
  318. }).then(res => {
  319. this.workername = res.data.name
  320. this.workerheadimg = res.data.headimg
  321. })
  322. },
  323. getlist() {
  324. uni.showLoading({
  325. mask: true,
  326. title: "请稍后"
  327. })
  328. this.$u.post('/api/News/news_list', {
  329. worker_id: this.worker_id,
  330. page_num: 20,
  331. page: this.page
  332. }).then(res => {
  333. var list = res.data
  334. list.forEach(val => {
  335. val.defaultTime = val.time
  336. this.list.unshift(val)
  337. })
  338. if (this.page == 1) {
  339. this.$nextTick(() => {
  340. uni.pageScrollTo({
  341. scrollTop: 99999
  342. })
  343. })
  344. }
  345. })
  346. },
  347. send1(tempFilePath, duration) {
  348. var option = {
  349. type: "audio",
  350. chatType: "singleChat",
  351. filename: "tempFilePath",
  352. from: uni.getStorageSync("hx_username"),
  353. // 消息接收方:单聊为对端用户 ID,群聊和聊天室分别为群组 ID 和聊天室 ID。
  354. to: this.hx_username,
  355. body: {
  356. //文件 URL。
  357. url: tempFilePath,
  358. //文件类型。
  359. type: "audio",
  360. //文件名。
  361. filename: "tempFilePath",
  362. // 音频文件时长,单位为秒。
  363. length: Math.ceil(duration / 1000),
  364. },
  365. };
  366. // let option = {
  367. // // 消息类型。
  368. // type: "audio",
  369. // file: tempFilePath,
  370. // // 语音文件长度,单位为秒。
  371. // length: Math.ceil(duration / 1000),
  372. // // 消息接收方:单聊为对方用户 ID,群聊和聊天室分别为群组 ID 和聊天室 ID。
  373. // to: this.hx_username,
  374. // // 会话类型:单聊、群聊和聊天室分别为 `singleChat`、`groupChat` 和 `chatRoom`。
  375. // chatType: "singleChat",
  376. // // 语音文件上传失败。
  377. // // onFileUploadError: function() {
  378. // // console.log("onFileUploadError");
  379. // // },
  380. // // // 语音文件上传进度。
  381. // // onFileUploadProgress: function(e) {
  382. // // console.log(e);
  383. // // },
  384. // // // 语音文件上传成功。
  385. // // onFileUploadComplete: function() {
  386. // // console.log("onFileUploadComplete");
  387. // // },
  388. // // ext: {},
  389. // };
  390. // let option = {
  391. // // 消息类型。
  392. // type: "txt",
  393. // // 消息内容。
  394. // msg: tempFilePath,
  395. // // 消息发送方:单聊为对方用户 ID,群聊和聊天室分别为群组 ID 和聊天室 ID。
  396. // from: uni.getStorageSync("hx_username"),
  397. // // 消息接收方:单聊为对方用户 ID,群聊和聊天室分别为群组 ID 和聊天室 ID。
  398. // to: this.hx_username,
  399. // // 会话类型:单聊、群聊和聊天室分别为 `singleChat`、`groupChat` 和 `chatRoom`,默认为单聊。
  400. // chatType: "singleChat",
  401. // };
  402. let msg = this.$WebIM.message.create(option);
  403. console.log(msg);
  404. this.$WebIM.conn.send(msg).then(res => {
  405. this.$u.post('/api/News/send_news', {
  406. worker_id: this.worker_id,
  407. content: tempFilePath,
  408. type1: "record",
  409. time: msg.body.length
  410. }).then(res => {
  411. this.chatlist.push({
  412. defaultTime: msg.body.length,
  413. content: tempFilePath,
  414. create_at: this.$u.timeFormat(msg.time, 'yyyy-mm-dd hh:MM:ss'),
  415. headimg: this.myheadimg,
  416. name: this.myname,
  417. type: 1,
  418. type1: "record",
  419. message: msg.body.length
  420. })
  421. this.$nextTick(() => {
  422. uni.pageScrollTo({
  423. scrollTop: 99999
  424. })
  425. })
  426. console.log(this.chatlist);
  427. })
  428. })
  429. },
  430. send(text) {
  431. if (!text) {
  432. this.$u.toast("请输入内容")
  433. return
  434. }
  435. this.text = ''
  436. let option = {
  437. // 消息类型。
  438. type: "txt",
  439. // 消息内容。
  440. msg: text,
  441. // 消息发送方:单聊为对方用户 ID,群聊和聊天室分别为群组 ID 和聊天室 ID。
  442. from: uni.getStorageSync("hx_username"),
  443. // 消息接收方:单聊为对方用户 ID,群聊和聊天室分别为群组 ID 和聊天室 ID。
  444. to: this.hx_username,
  445. // 会话类型:单聊、群聊和聊天室分别为 `singleChat`、`groupChat` 和 `chatRoom`,默认为单聊。
  446. chatType: "singleChat",
  447. };
  448. // 创建文本消息。
  449. let msg = this.$WebIM.message.create(option);
  450. // console.log(msg);
  451. this.$WebIM.conn.send(msg).then(res => {
  452. this.$u.post('/api/News/send_news', {
  453. worker_id: this.worker_id,
  454. content: text
  455. }).then(res => {
  456. this.chatlist.push({
  457. content: msg.msg,
  458. create_at: this.$u.timeFormat(msg.time, 'yyyy-mm-dd hh:MM:ss'),
  459. headimg: this.myheadimg,
  460. name: this.myname,
  461. type: 1,
  462. })
  463. this.$nextTick(() => {
  464. uni.pageScrollTo({
  465. scrollTop: 99999
  466. })
  467. })
  468. // console.log(this.chatlist);
  469. })
  470. })
  471. }
  472. }
  473. }
  474. </script>
  475. <style lang="scss" scoped>
  476. page {
  477. background-color: #F2F2F2;
  478. }
  479. .record {
  480. display: flex;
  481. align-items: center;
  482. background-color: #1F7EFF;
  483. color: #fff;
  484. border-radius: 20rpx;
  485. box-sizing: border-box;
  486. }
  487. .record1 {
  488. display: flex;
  489. align-items: center;
  490. background-color: #eee;
  491. color: #000;
  492. border-radius: 20rpx;
  493. box-sizing: border-box;
  494. }
  495. .quxiao {
  496. background: #1F7EFF;
  497. border-radius: 12rpx;
  498. font-size: 30rpx;
  499. font-family: PingFangSC-Medium, PingFang SC;
  500. font-weight: 500;
  501. color: #000;
  502. text-align: center;
  503. height: 84rpx;
  504. line-height: 84rpx;
  505. }
  506. .fasong {
  507. background: limegreen;
  508. border-radius: 12rpx;
  509. font-size: 30rpx;
  510. font-family: PingFangSC-Medium, PingFang SC;
  511. font-weight: 500;
  512. color: #fff;
  513. text-align: center;
  514. height: 84rpx;
  515. line-height: 84rpx;
  516. }
  517. .chat {
  518. .chat-list {
  519. padding: 24rpx;
  520. .more-btn {
  521. padding: 10rpx 0;
  522. text-align: center;
  523. }
  524. .chat-item {
  525. margin-bottom: 44rpx;
  526. .item-box {
  527. text:first-child {
  528. font-size: 28rpx;
  529. font-weight: 500;
  530. // color: #666666;
  531. padding: 8rpx 24rpx;
  532. }
  533. text:last-child {
  534. padding: 9rpx 24rpx;
  535. border-radius: 20rpx;
  536. }
  537. .text1 {
  538. background-color: #fff;
  539. border-radius: 20rpx;
  540. }
  541. .text2 {
  542. background-color: #1F7EFF;
  543. color: #fff;
  544. border-radius: 20rpx;
  545. }
  546. }
  547. .user-head {
  548. width: 70rpx;
  549. height: 70rpx;
  550. border-radius: 100rpx;
  551. }
  552. }
  553. }
  554. .chat-btn {
  555. position: fixed;
  556. bottom: 0;
  557. left: 0;
  558. width: 750rpx;
  559. height: 166rpx;
  560. background: #FFFFFF;
  561. z-index: 10;
  562. padding: 0 24rpx 60rpx 24rpx;
  563. input {
  564. // width: 540rpx;
  565. flex: 1;
  566. margin-right: 20rpx;
  567. height: 58rpx;
  568. background: #F2F2F2;
  569. border-radius: 30rpx;
  570. padding: 0 28rpx;
  571. box-sizing: border-box;
  572. }
  573. .post {
  574. width: 102rpx;
  575. line-height: 58rpx;
  576. background: #1F7EFF;
  577. border-radius: 30rpx;
  578. text-align: center;
  579. font-size: 24rpx;
  580. // font-family: PingFangSC-Regular, PingFang SC;
  581. font-weight: 400;
  582. color: #FFFFFF;
  583. }
  584. }
  585. }
  586. ::v-deep .record-btn {
  587. width: 80vw !important;
  588. text-align: center !important;
  589. flex: 1 !important;
  590. }
  591. ::v-deep .record .u-icon.data-v-6e20bb40 {
  592. transform: rotate(-90deg) !important;
  593. }
  594. ::v-deep .record1 .u-icon.data-v-6e20bb40 {
  595. transform: rotate(90deg) !important;
  596. }
  597. </style>