multi_channel.dart 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. import 'dart:developer';
  2. import 'package:agora_rtc_engine/rtc_channel.dart';
  3. import 'package:agora_rtc_engine/rtc_engine.dart';
  4. import 'package:agora_rtc_engine/rtc_local_view.dart' as RtcLocalView;
  5. import 'package:agora_rtc_engine/rtc_remote_view.dart' as RtcRemoteView;
  6. import 'package:agora_rtc_engine_example/config/agora.config.dart' as config;
  7. import 'package:flutter/cupertino.dart';
  8. import 'package:flutter/foundation.dart';
  9. import 'package:flutter/material.dart';
  10. import 'package:permission_handler/permission_handler.dart';
  11. const _channelId0 = 'channel0';
  12. const _channelId1 = 'channel1';
  13. /// MultiChannel Example
  14. class MultiChannel extends StatefulWidget {
  15. @override
  16. State<StatefulWidget> createState() => _State();
  17. }
  18. class _State extends State<MultiChannel> {
  19. late final RtcEngine _engine;
  20. late final RtcChannel _channel0, _channel1;
  21. String? renderChannelId;
  22. bool isJoined0 = false, isJoined1 = false;
  23. List<int> remoteUid0 = [], remoteUid1 = [];
  24. @override
  25. void initState() {
  26. super.initState();
  27. this._initEngine();
  28. }
  29. @override
  30. void dispose() {
  31. super.dispose();
  32. _engine.destroy();
  33. }
  34. _initEngine() async {
  35. _engine = await RtcEngine.createWithContext(RtcEngineContext(config.appId));
  36. await _engine.enableVideo();
  37. await _engine.startPreview();
  38. await _engine.setChannelProfile(ChannelProfile.LiveBroadcasting);
  39. await _engine.setClientRole(ClientRole.Broadcaster);
  40. }
  41. _joinChannel0() async {
  42. if (defaultTargetPlatform == TargetPlatform.android) {
  43. await [Permission.microphone, Permission.camera].request();
  44. }
  45. _channel0 = await RtcChannel.create(_channelId0);
  46. this._addListener(_channel0);
  47. await _channel0.setClientRole(ClientRole.Broadcaster);
  48. await _channel0.joinChannel(
  49. null,
  50. null,
  51. 0,
  52. ChannelMediaOptions(
  53. publishLocalAudio: false,
  54. publishLocalVideo: false,
  55. ));
  56. }
  57. _joinChannel1() async {
  58. if (defaultTargetPlatform == TargetPlatform.android) {
  59. await [Permission.microphone, Permission.camera].request();
  60. }
  61. _channel1 = await RtcChannel.create(_channelId1);
  62. this._addListener(_channel1);
  63. await _channel1.setClientRole(ClientRole.Broadcaster);
  64. await _channel1.joinChannel(
  65. null,
  66. null,
  67. 0,
  68. ChannelMediaOptions(
  69. publishLocalAudio: false,
  70. publishLocalVideo: false,
  71. ));
  72. }
  73. _addListener(RtcChannel channel) {
  74. String channelId = channel.channelId;
  75. channel.setEventHandler(
  76. RtcChannelEventHandler(joinChannelSuccess: (channel, uid, elapsed) {
  77. log('joinChannelSuccess ${channel} ${uid} ${elapsed}');
  78. if (channelId == _channelId0) {
  79. setState(() {
  80. isJoined0 = true;
  81. });
  82. } else if (channelId == _channelId1) {
  83. setState(() {
  84. isJoined1 = true;
  85. });
  86. }
  87. }, userJoined: (uid, elapsed) {
  88. log('userJoined ${channel.channelId} $uid $elapsed');
  89. }, userOffline: (uid, reason) {
  90. log('userOffline ${channel.channelId} $uid $reason');
  91. }, leaveChannel: (stats) {
  92. log('leaveChannel ${channel.channelId} ${stats.toJson()}');
  93. if (channelId == _channelId0) {
  94. this.setState(() {
  95. isJoined0 = false;
  96. remoteUid0.clear();
  97. });
  98. } else if (channelId == _channelId1) {
  99. this.setState(() {
  100. isJoined1 = false;
  101. remoteUid1.clear();
  102. });
  103. }
  104. }, remoteVideoStateChanged: (uid, state, reason, elapsed) {
  105. log('remoteVideoStateChanged ${uid} ${state} ${reason} ${elapsed}');
  106. if (state == VideoRemoteState.Starting) {
  107. if (channelId == _channelId0) {
  108. this.setState(() {
  109. remoteUid0.add(uid);
  110. });
  111. } else if (channelId == _channelId1) {
  112. this.setState(() {
  113. remoteUid1.add(uid);
  114. });
  115. }
  116. } else if (state == VideoRemoteState.Stopped) {
  117. if (channelId == _channelId0) {
  118. this.setState(() {
  119. remoteUid0.removeWhere((element) => element == uid);
  120. });
  121. } else if (channelId == _channelId1) {
  122. this.setState(() {
  123. remoteUid1.removeWhere((element) => element == uid);
  124. });
  125. }
  126. }
  127. }));
  128. }
  129. _publishChannel0() async {
  130. await _channel1.unpublish();
  131. await _channel0.publish();
  132. }
  133. _publishChannel1() async {
  134. await _channel0.unpublish();
  135. await _channel1.publish();
  136. }
  137. _leaveChannel0() async {
  138. await _channel0.leaveChannel();
  139. }
  140. _leaveChannel1() async {
  141. await _channel1.leaveChannel();
  142. }
  143. @override
  144. Widget build(BuildContext context) {
  145. return Stack(
  146. children: [
  147. Column(
  148. children: [
  149. Row(
  150. children: [
  151. Expanded(
  152. flex: 1,
  153. child: ElevatedButton(
  154. onPressed: () {
  155. if (isJoined0) {
  156. this._leaveChannel0();
  157. } else {
  158. this._joinChannel0();
  159. }
  160. },
  161. child: Text('${isJoined0 ? 'Leave' : 'Join'} $_channelId0'),
  162. ),
  163. )
  164. ],
  165. ),
  166. Row(
  167. children: [
  168. Expanded(
  169. flex: 1,
  170. child: ElevatedButton(
  171. onPressed: () {
  172. if (isJoined1) {
  173. this._leaveChannel1();
  174. } else {
  175. this._joinChannel1();
  176. }
  177. },
  178. child: Text('${isJoined1 ? 'Leave' : 'Join'} $_channelId1'),
  179. ),
  180. )
  181. ],
  182. ),
  183. _renderVideo(),
  184. ],
  185. ),
  186. Align(
  187. alignment: Alignment.bottomRight,
  188. child: Column(
  189. mainAxisSize: MainAxisSize.min,
  190. children: [
  191. ElevatedButton(
  192. onPressed: this._publishChannel0,
  193. child: Text('Publish ${_channelId0}'),
  194. ),
  195. ElevatedButton(
  196. onPressed: () {
  197. setState(() {
  198. renderChannelId = _channelId0;
  199. });
  200. },
  201. child: Text('Render ${_channelId0}'),
  202. ),
  203. ElevatedButton(
  204. onPressed: this._publishChannel1,
  205. child: Text('Publish ${_channelId1}'),
  206. ),
  207. ElevatedButton(
  208. onPressed: () {
  209. setState(() {
  210. renderChannelId = _channelId1;
  211. });
  212. },
  213. child: Text('Render ${_channelId1}'),
  214. ),
  215. ],
  216. ),
  217. )
  218. ],
  219. );
  220. }
  221. _renderVideo() {
  222. List<int>? remoteUid = null;
  223. if (renderChannelId == _channelId0) {
  224. remoteUid = remoteUid0;
  225. } else if (renderChannelId == _channelId1) {
  226. remoteUid = remoteUid1;
  227. }
  228. return Expanded(
  229. child: Stack(
  230. children: [
  231. RtcLocalView.SurfaceView(
  232. channelId: renderChannelId,
  233. ),
  234. if (remoteUid != null)
  235. Align(
  236. alignment: Alignment.topLeft,
  237. child: SingleChildScrollView(
  238. scrollDirection: Axis.horizontal,
  239. child: Row(
  240. children: List.of(remoteUid.map(
  241. (e) => Container(
  242. width: 120,
  243. height: 120,
  244. child: RtcRemoteView.SurfaceView(
  245. uid: e,
  246. channelId: renderChannelId,
  247. ),
  248. ),
  249. )),
  250. ),
  251. ),
  252. )
  253. ],
  254. ),
  255. );
  256. }
  257. }