media_channel_relay.dart 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. import 'dart:developer';
  2. import 'package:agora_rtc_engine/rtc_engine.dart';
  3. import 'package:agora_rtc_engine/rtc_local_view.dart' as RtcLocalView;
  4. import 'package:agora_rtc_engine/rtc_remote_view.dart' as RtcRemoteView;
  5. import 'package:agora_rtc_engine_example/config/agora.config.dart' as config;
  6. import 'package:flutter/cupertino.dart';
  7. import 'package:flutter/foundation.dart';
  8. import 'package:flutter/material.dart';
  9. import 'package:permission_handler/permission_handler.dart';
  10. /// MediaChannelRelay Example
  11. class MediaChannelRelay extends StatefulWidget {
  12. @override
  13. State<StatefulWidget> createState() => _State();
  14. }
  15. class _State extends State<MediaChannelRelay> {
  16. late final RtcEngine _engine;
  17. bool isJoined = false;
  18. int? remoteUid;
  19. bool isRelaying = false;
  20. @override
  21. void initState() {
  22. super.initState();
  23. }
  24. @override
  25. void dispose() {
  26. super.dispose();
  27. _engine.destroy();
  28. }
  29. _initEngine() async {
  30. if (defaultTargetPlatform == TargetPlatform.android) {
  31. await Permission.microphone.request();
  32. }
  33. _engine = await RtcEngine.createWithContext(RtcEngineContext(config.appId));
  34. this._addListener();
  35. // enable video module and set up video encoding configs
  36. await _engine.enableVideo();
  37. // make this room live broadcasting room
  38. await _engine.setChannelProfile(ChannelProfile.LiveBroadcasting);
  39. await _engine.setClientRole(ClientRole.Broadcaster);
  40. // Set audio route to speaker
  41. await _engine.setDefaultAudioRoutetoSpeakerphone(true);
  42. // start joining channel
  43. // 1. Users can only see each other after they join the
  44. // same channel successfully using the same app id.
  45. // 2. If app certificate is turned on at dashboard, token is needed
  46. // when joining channel. The channel name and uid used to calculate
  47. // the token has to match the ones used for channel join
  48. await _engine.joinChannel(config.token, config.channelId, null, 0, null);
  49. }
  50. _addListener() {
  51. _engine.setEventHandler(RtcEngineEventHandler(warning: (warningCode) {
  52. log('Warning ${warningCode}');
  53. }, error: (errorCode) {
  54. log('Warning ${errorCode}');
  55. }, joinChannelSuccess: (channel, uid, elapsed) {
  56. log('joinChannelSuccess ${channel} ${uid} ${elapsed}');
  57. ;
  58. setState(() {
  59. isJoined = true;
  60. });
  61. }, userJoined: (uid, elapsed) {
  62. log('userJoined $uid $elapsed');
  63. this.setState(() {
  64. remoteUid = uid;
  65. });
  66. }, userOffline: (uid, reason) {
  67. log('userOffline $uid $reason');
  68. this.setState(() {
  69. remoteUid = null;
  70. });
  71. }, channelMediaRelayStateChanged:
  72. (ChannelMediaRelayState state, ChannelMediaRelayError code) {
  73. switch (state) {
  74. case ChannelMediaRelayState.Idle:
  75. log('ChannelMediaRelayState.Idle $code');
  76. this.setState(() {
  77. isRelaying = false;
  78. });
  79. break;
  80. case ChannelMediaRelayState.Connecting:
  81. log('ChannelMediaRelayState.Connecting $code)');
  82. break;
  83. case ChannelMediaRelayState.Running:
  84. log('ChannelMediaRelayState.Running $code)');
  85. this.setState(() {
  86. isRelaying = true;
  87. });
  88. break;
  89. case ChannelMediaRelayState.Failure:
  90. log('ChannelMediaRelayState.Failure $code)');
  91. this.setState(() {
  92. isRelaying = false;
  93. });
  94. break;
  95. default:
  96. log('default $code)');
  97. break;
  98. }
  99. }));
  100. }
  101. _onPressRelayOrStop() async {
  102. if (isRelaying) {
  103. await _engine.stopChannelMediaRelay();
  104. return;
  105. }
  106. if (_controller.text.length == 0) {
  107. return;
  108. }
  109. await _engine.startChannelMediaRelay(ChannelMediaRelayConfiguration(
  110. ChannelMediaInfo(config.channelId, 0, token: config.token),
  111. [ChannelMediaInfo('', 0, token: '')]));
  112. _controller.clear();
  113. }
  114. final TextEditingController _controller = TextEditingController();
  115. @override
  116. Widget build(BuildContext context) {
  117. return Stack(
  118. children: [
  119. Column(
  120. children: [
  121. !isJoined
  122. ? Row(
  123. children: [
  124. Expanded(
  125. flex: 1,
  126. child: ElevatedButton(
  127. onPressed: _initEngine,
  128. child: Text('Join channel'),
  129. ),
  130. )
  131. ],
  132. )
  133. : _renderVideo(),
  134. if (isJoined)
  135. Row(
  136. mainAxisSize: MainAxisSize.max,
  137. children: [
  138. Expanded(
  139. child: TextField(
  140. controller: _controller,
  141. decoration: InputDecoration(
  142. hintText: 'Enter target relay channel name',
  143. ))),
  144. ElevatedButton(
  145. onPressed: _onPressRelayOrStop,
  146. child: Text(!isRelaying ? 'Relay' : 'Stop'),
  147. ),
  148. ],
  149. )
  150. ],
  151. ),
  152. ],
  153. );
  154. }
  155. _renderVideo() {
  156. return Row(children: [
  157. Expanded(
  158. child: AspectRatio(
  159. aspectRatio: 1,
  160. child: RtcLocalView.SurfaceView(),
  161. )),
  162. Expanded(
  163. child: AspectRatio(
  164. aspectRatio: 1,
  165. child: remoteUid != null
  166. ? RtcRemoteView.SurfaceView(
  167. uid: remoteUid!,
  168. )
  169. : Container(
  170. color: Colors.grey[200],
  171. ),
  172. ),
  173. )
  174. ]);
  175. }
  176. }