index.nvue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. <template>
  2. <view class="wraper">
  3. <view class="infoConnecting">
  4. 11多人会议
  5. </view>
  6. <live-pusher class="live-pusher" :url="pubUrl" @netstatus="netstatusChange" :muted="false" :enable-camera="true" id="livePusher"
  7. ref="livePusher" @statechange="statechange" mode="HD" @error="error"></live-pusher>
  8. </live-pusher>
  9. <view
  10. v-if="subUrls.length > 0"
  11. v-for="item in subUrls">
  12. <video
  13. :id="item.streamId"
  14. :src="item.subUrl"
  15. object-fit="contain"
  16. autoplay
  17. >
  18. <view class="userName">{{item.memName}}</view>
  19. </video>
  20. </view>
  21. <!-- </scroll-view> -->
  22. <!-- <view class="controlContent">
  23. <view class="emediaContrContent">
  24. <view class="controlItem" @tap="toggleCamera" style="{color: devicePositionColor}">
  25. <image
  26. class="icon-record"
  27. :src="'../../../static/images/'+devicePositionIcon+'2x.png'" style="{width:'22px'; height: '24px'}"/>
  28. 切换摄像头
  29. </view>
  30. <view class="controlItem" @tap="toggleMuted" style="{color: micphoneColor}">
  31. <image
  32. class="icon-record"
  33. :src="'../../../static/images/'+micphoneIcon+'2x.png'" style="{width:'22px'; height: '24px'}"/>
  34. 麦克风</view>
  35. <view class="controlItem" @tap="togglePlay" style="{color: videoColor}">
  36. <image
  37. class="icon-record"
  38. :src="'../../../static/images/'+videoIcon+'2x.png'" style="{width:'22px'; height: '24px'}"/>
  39. 视频</view>
  40. <view class="controlItem" @tap="toggleBeauty" style="{color: beautyColor}">
  41. <image
  42. class="icon-record"
  43. :src="'../../../static/images/'+beautyIcon+'.png'" style="{width:'16px'; height: '24px'}"/>
  44. 美颜</view>
  45. <view class="controlItem" @tap="inviteMember">
  46. <image
  47. class="icon-record"
  48. src='../../../static/images/invite_white2x.png' style="{width:'22px'; height: '24px'"/>
  49. 邀请</view>
  50. </view>
  51. <view class="hangup" @tap="hangup">
  52. <image
  53. class="icon-record"
  54. src='../../../static/images/hangup2x.png'/>
  55. </view>
  56. </view>
  57. -->
  58. </view>
  59. </template>
  60. <script>
  61. export default {
  62. data() {
  63. return {
  64. pubUrl: "",
  65. subUrls: [],
  66. showInvite: true,
  67. devicePosition: "front",
  68. muted: false,
  69. playVideoMuted: false,
  70. devicePositionIcon: 'switchCamera_white',
  71. devicePositionColor: '#fff',
  72. micphoneIcon: 'micphone_white',
  73. beautyIcon: 'beauty',
  74. micphoneColor: '#fff',
  75. videoIcon: 'video_white',
  76. videoColor: '#fff',
  77. beauty: 9,
  78. beautyColor: '#fff',
  79. myName: '',
  80. confrId: '',
  81. enableCamera: true,
  82. time: '',
  83. context: {}
  84. }
  85. },
  86. props: {
  87. username: {
  88. type: Object,
  89. default: () => ({}),
  90. },
  91. action: {
  92. type: Object,
  93. default: {},
  94. },
  95. groupId:{
  96. type: String,
  97. default: '',
  98. },
  99. },
  100. methods: {
  101. joinRoom(data){
  102. let me = this;
  103. console.log('joinRoom', data)
  104. me.context = uni.createLivePusherContext("livePusher", me);
  105. console.log('我的播放组件', me.context)
  106. var context = this.context
  107. if (context && context.ctx && context.ctx.attr && context.ctx.attr.url) {
  108. context.stop()
  109. }
  110. let id = wx.WebIM.conn.getUniqueId();
  111. let roomName = 'wxConfr' + id //随机的房间名,防止和别人的房间名冲突
  112. let rec = wx.getStorageSync("rec") || false;
  113. let recMerge = wx.getStorageSync("recMerge") || false;
  114. let params = {
  115. roomName,
  116. password: '',
  117. role: 7,
  118. config: {
  119. rec,
  120. recMerge
  121. }
  122. }
  123. if (data) {
  124. params.roomName = data.roomName
  125. params.password = data.password
  126. }
  127. wx.emedia.mgr.joinRoom(params).then((res) => {
  128. console.log('res', res)
  129. let confrId = res.confrId
  130. me.confrId= confrId
  131. me.$emit('createConfrSuccess', {confrId: confrId, groupId: me.username.groupId, roomName: params.roomName, password: params.password})
  132. console.log('--------+--+----+---+--------')
  133. let rtcId = wx.emedia.util.getRtcId()
  134. console.log(11111111111111, confrId)
  135. wx.emedia.mgr.pubStream(rtcId).then(function(res){
  136. console.log('pubUrl-------------', res.data.rtmp)
  137. console.log(2222222222222)
  138. me.pubUrl= res.data.rtmp
  139. setTimeout(() => {
  140. console.log('55555555555', me)
  141. console.log(me.context)
  142. me.context.start()
  143. }, 500)
  144. }).catch(e => {
  145. console.log(4444444, e) //404
  146. return f(false); //即便返回了一个成功的promise,下面的finally也会执行,如果返回的是失败的promise,控制台最后一行会报错uncaught (in promise) 404
  147. })
  148. .finally( (e) => {
  149. console.log(100,e) //100
  150. })
  151. console.log(33333333333333, me.pubUrl)
  152. })
  153. },
  154. createConf(){
  155. console.log('>>> createConf');
  156. var me = this
  157. let rec = wx.getStorageSync("rec") || false;
  158. let recMerge = wx.getStorageSync("recMerge") || false;
  159. //参数:会议类型 密码 是否录制 是否合并
  160. wx.emedia.mgr.createConference(10, '', rec, recMerge).then(function(data){
  161. console.log('成功', data)
  162. let ticket = data.data.ticket
  163. let ticketJosn = JSON.parse(ticket)
  164. let confrId = ticketJosn.confrId
  165. wx.emedia.mgr.joinConferenceWithTicket(confrId, ticket).then(function(res){
  166. console.log('加入会议成功', res)
  167. })
  168. // wx.emedia.mgr.joinConference(confrId, '').then(function(res){
  169. // console.log('加入会议成功', res)
  170. // })
  171. me.confrId = confrId
  172. me.$emit('createConfrSuccess', {confrId: confrId, groupId: me.username.groupId})
  173. })
  174. },
  175. joinConf(data){
  176. console.log('加入会议 ————-------————')
  177. console.log(data)
  178. let me = this
  179. wx.emedia.mgr.getConferenceTkt(data.confrId, data.password).then(function(res){
  180. console.log('申请reqTkt成功', res.data)
  181. let ticket = res.data.ticket || ''
  182. let tktObj = JSON.parse(ticket)
  183. wx.emedia.mgr.joinConferenceWithTicket(data.confrId, ticket).then(function(res){
  184. console.log('加入会议成功', res)
  185. })
  186. me.confrId= tktObj.confrId
  187. me.$emit('createConfrSuccess', {confrId: tktObj.confrId, groupId: me.username.groupId})
  188. })
  189. },
  190. togglePlay(){
  191. let me = this
  192. console.log("%c togglePlay", "color:green")
  193. // this.LivePusherContext.stop()
  194. this.enableCamera= !me.enableCamera
  195. this.pubUrl= me.pubUrl
  196. this.videoIcon= this.videoIcon == 'video_white'?'video_gray': 'video_white'
  197. this.videoColor= this.videoColor == '#fff'? '#aaa': '#fff'
  198. this.LivePusherContext.start()
  199. /* this.setData({
  200. }, () => {
  201. }) */
  202. },
  203. toggleCamera(){
  204. console.log("%c toggleCamera", "color:green")
  205. let me = this
  206. // me.LivePusherContext.stop()
  207. me.LivePusherContext.switchCamera({
  208. success: function(){
  209. me.devicePosition= me.devicePosition == 'fron' ? 'back' : 'front',
  210. me.devicePositionIcon= me.devicePositionIcon =='switchCamera_white'?'switchCamera_gray': 'switchCamera_white',
  211. me.devicePositionColor= me.devicePositionColor == '#fff'? '#aaa':'#fff'
  212. }
  213. })
  214. },
  215. toggleMuted(){
  216. console.log("%c toggleMuted", "color:green")
  217. this.muted= !this.muted
  218. this.micphoneIcon= this.micphoneIcon == 'micphone_white'? 'micphone_gray': 'micphone_white'
  219. this.micphoneColor= this.micphoneColor == '#fff'? '#aaa': '#fff'
  220. },
  221. toggleBeauty(){
  222. this.beauty= this.beauty == 0 ? 9 : 0
  223. this.beautyColor= this.beautyColor == '#fff'? '#aaa': '#fff'
  224. this.beautyIcon= this.beautyIcon == 'beauty' ? 'beauty_gray': 'beauty'
  225. },
  226. hangup(){
  227. console.log('挂断', this.confrId)
  228. wx.emedia.mgr.exitConference(this.confrId)
  229. this.$emit('hangup')
  230. this.stopTimer()
  231. },
  232. inviteMember(){
  233. this.$emit('inviteMember', this.groupId)
  234. },
  235. // statechange(e){
  236. // console.log('>>>>>>>>>live-pusher code:', e.detail)
  237. // if (e.detail.code === 5001) {
  238. // // 部分安卓手机在接电话时会停止推拉流报错,状态码5001,此时退出会议。 https://developers.weixin.qq.com/community/develop/doc/0006ac6d7a4968fa675a49fef53c00
  239. // this.hangup()
  240. // console.error('由于有电话接入,已退出会议')
  241. // }
  242. // },
  243. netstatusChange(e){
  244. console.log('>>>>>>>>>>net status:', e.detail)
  245. },
  246. getTimer(){
  247. let count = 0;
  248. let time = '00:00:00'
  249. this.timer = setInterval(() => {
  250. count++;
  251. let s = showNum(count % 60);
  252. let m = showNum(parseInt((count / 60)) % 60)
  253. let h = showNum(parseInt(count / 60 / 60))
  254. time = `${h}:${m}:${s}`
  255. this.time = time
  256. }, 1000)
  257. function showNum(num) {
  258. if (num < 10) {
  259. return '0' + num
  260. }
  261. return num
  262. }
  263. },
  264. stopTimer(){
  265. clearInterval(this.timer)
  266. },
  267. error(err){
  268. console.log('EEEEEEEEEEEEE', err)
  269. },
  270. statechange(state){
  271. console.log("SSSSSSSSSSS", state)
  272. }
  273. },
  274. mounted() {
  275. console.log('进入')
  276. wx.setKeepScreenOn({
  277. keepScreenOn: true
  278. })
  279. console.log('进入111')
  280. this.myName= wx.WebIM.conn.context.userId
  281. console.log('进入222')
  282. this.getTimer()
  283. var me = this
  284. let subUrls = []
  285. let obj = {};
  286. console.log('进入333')
  287. this.LivePusherContext = uni.createLivePusherContext('livePusher',this)
  288. console.log('this.LivePusherContext ..', this.LivePusherContext)
  289. if(this.action&&this.action.action == 'join'){
  290. console.log('join')
  291. // 音视频sdk提供两种创建、加入会议的api 使用任意一种都可以:(1) 一种是通过会议id ticket 或者 会议id 密码加入会议 如使用下面 joinConf
  292. // (2)另一种是通过房间名 密码加入 如使用下面joinRoom
  293. //this.joinConf(this.action) // (1)
  294. //this.joinRoom(this.action) // (2)
  295. if (this.action.roomName) {
  296. console.log('使用joinroom')
  297. this.joinRoom(this.action)
  298. }else{
  299. console.log('使用joinConf')
  300. this.joinConf(this.action)
  301. }
  302. }else{
  303. // 创建会议同样是两种api (1)一种是使用createConf 单纯创建一个会议,需要再申请ticket 或者用密码加入会议 如使用下面的createConf
  304. // (2)也可以使用joinRoom,通过房间名、密码创建房间并直接加入 不需再进行加入会议的操作
  305. this.joinRoom() // (1)
  306. //this.createConf() // (2)
  307. }
  308. wx.emedia.mgr.onMediaChanaged = function(e){
  309. console.log('onMediaChanaged', e)
  310. }
  311. wx.emedia.mgr.onConferenceExit = function(e){
  312. console.log('onConferenceExit', e)
  313. }
  314. wx.emedia.mgr.onMemberExited = function(reason){
  315. console.log('onMemberExited', reason)
  316. };
  317. wx.emedia.mgr.onStreamControl = function(mem){
  318. console.log('onStreamControl', mem)
  319. }
  320. wx.emedia.mgr.onStreamControl.onSoundChanage = function(a, b, c, d){
  321. console.log('onSoundChanage')
  322. }
  323. wx.emedia.mgr.onReconnect = function (res, ent){
  324. // 发生断网重连,相当于重新加入会议
  325. // 清空live-player 否则在原来的后面追加,导致原来的黑屏显示
  326. subUrls = []
  327. // 重新加入恢复到初始状态,防止和控制按钮状态不符
  328. // 重连后摄像头方向不会改变
  329. me.subUrls= []
  330. me.showInvite= true
  331. me.devicePosition= "front"
  332. me.muted= false
  333. me.playVideoMuted= false
  334. me.micphoneIcon= 'micphone_white'
  335. me.micphoneColor= '#fff'
  336. me.videoIcon= 'video_white'
  337. me.beautyIcon= 'beauty'
  338. me.videoColor= '#fff'
  339. me.beauty= 9
  340. me.beautyColor= '#fff'
  341. me.enableCamera= true
  342. }
  343. wx.emedia.mgr.onMemberJoined = function(mem){
  344. console.log("++++++++++ member", mem)
  345. var jid = wx.WebIM.conn.context.jid
  346. let identityName = jid.appKey + '_' + jid.name+ '@' + jid.domain
  347. // let identityName = wx.WebIM.conn.context.jid.split("/")[0]
  348. // 如果是自己进入会议了,开始发布流
  349. if(mem.name == identityName){
  350. let rtcId = wx.emedia.util.getRtcId()
  351. wx.emedia.mgr.pubStream(rtcId).then(function(res){
  352. me.pubUrl= res.data.rtmp
  353. })
  354. var enableCamera = me.enableCamera;
  355. console.warn("begin enable camera", me.enableCamera);
  356. //默认enableCamera为false 关闭摄像头时声音不会有延迟,否则有延迟
  357. //所以最好别用autopush
  358. me.enableCamera= false
  359. me.pubUrl= me.url + 'record_type=audio' || 'https://domain/push_stream'
  360. // enableCamera && me.enableCamera= enableCamera
  361. /* me.setData({
  362. enableCamera: false,
  363. pubUrl: me.url + 'record_type=audio' || 'https://domain/push_stream',
  364. }, () => {
  365. // var enableCameraDefault = true
  366. //if(!enableCameraDefault){ //治疗不推流的毛病
  367. // console.log('关闭摄像头推流')
  368. // setTimeout(() => {
  369. // me.LivePusherContext.start({
  370. // success: function () {
  371. // console.log('关闭摄像头推流', enableCamera)
  372. // }
  373. // })
  374. // }, 1500)
  375. // }else{
  376. // me.LivePusherContext.start({
  377. // success: function () {
  378. // console.log('开始推流了', enableCamera)
  379. // enableCamera && me.setData({enableCamera: enableCamera})
  380. // }
  381. // })
  382. // }
  383. }) */
  384. }
  385. }
  386. wx.emedia.mgr.onStreamAdded = function(stream){
  387. console.log('%c onAddStream', 'color: green', stream)
  388. let streamId = stream.id
  389. // setTimeout(() => {
  390. if(subUrls.length > 8){
  391. return
  392. }
  393. wx.emedia.mgr.subStream(streamId).then(function(data){
  394. console.log('%c 订阅流成功', 'color:green', data)
  395. // let playContext = wx.createLivePlayerContext(streamId, me)
  396. let subUrl = {
  397. streamId: streamId,
  398. subUrl: data.data.rtmp,
  399. memName: stream.memName.split("_")[1].split("@")[0],
  400. // playContext: playContext
  401. }
  402. subUrls.push(subUrl)
  403. console.log('%c subUrls 11 ....', "background:yellow")
  404. console.log(subUrls)
  405. me.subUrls= subUrls
  406. me.showInvite= false
  407. })
  408. // }, 2000)
  409. }
  410. wx.emedia.mgr.onStreamRemoved = function(stream){
  411. console.log('%c onRemoveStream', 'color: red', stream)
  412. subUrls = subUrls.filter((item) => {
  413. if(item.streamId != stream.id){
  414. return item
  415. }else{
  416. console.log('%c ------', 'backgroukd:yellow')
  417. console.log(item)
  418. // item.playContext.stop({
  419. // success: function(){
  420. // console.log('关闭成功')
  421. // },
  422. // complete: function(){
  423. // console.log('关闭成功')
  424. // }
  425. // })
  426. }
  427. })
  428. obj[stream.id] = false
  429. me.subUrls= subUrls
  430. console.log('subUrls', subUrls)
  431. },
  432. wx.emedia.mgr.onConfrAttrsUpdated = function(e){
  433. console.log('onConfrAttrsUpdated: ', e)
  434. }
  435. },
  436. }
  437. </script>