live_streaming.dart 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  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/foundation.dart';
  7. import 'package:flutter/material.dart';
  8. import 'package:flutter/scheduler.dart';
  9. import 'package:permission_handler/permission_handler.dart';
  10. /// LiveStreaming Example
  11. class LiveStreaming extends StatefulWidget {
  12. @override
  13. State<StatefulWidget> createState() => _State();
  14. }
  15. class _State extends State<LiveStreaming> {
  16. late final RtcEngine _engine;
  17. bool isJoined = false;
  18. ClientRole role = ClientRole.Audience;
  19. int? remoteUid;
  20. bool isLowAudio = true;
  21. @override
  22. void initState() {
  23. super.initState();
  24. SchedulerBinding.instance?.addPostFrameCallback((timeStamp) {
  25. _showMyDialog();
  26. });
  27. }
  28. @override
  29. void dispose() {
  30. super.dispose();
  31. _engine.destroy();
  32. }
  33. Future<void> _showMyDialog() async {
  34. return showDialog(
  35. context: context,
  36. barrierDismissible: false, // user must tap button!
  37. builder: (BuildContext context) {
  38. return AlertDialog(
  39. title: Text('AlertDialog Title'),
  40. content: SingleChildScrollView(
  41. child: ListBody(
  42. children: <Widget>[Text('Please choose role')],
  43. ),
  44. ),
  45. actions: <Widget>[
  46. TextButton(
  47. child: Text('Broadcaster'),
  48. onPressed: () {
  49. this.setState(() {
  50. role = ClientRole.Broadcaster;
  51. Navigator.of(context).pop();
  52. });
  53. },
  54. ),
  55. TextButton(
  56. child: Text('Audience'),
  57. onPressed: () {
  58. this.setState(() {
  59. role = ClientRole.Audience;
  60. Navigator.of(context).pop();
  61. });
  62. },
  63. ),
  64. ],
  65. );
  66. },
  67. );
  68. }
  69. _initEngine() async {
  70. if (defaultTargetPlatform == TargetPlatform.android) {
  71. await Permission.microphone.request();
  72. }
  73. _engine = await RtcEngine.createWithContext(RtcEngineContext(config.appId));
  74. this._addListener();
  75. // enable video module and set up video encoding configs
  76. await _engine.enableVideo();
  77. // make this room live broadcasting room
  78. await _engine.setChannelProfile(ChannelProfile.LiveBroadcasting);
  79. await this._updateClientRole(role);
  80. // Set audio route to speaker
  81. await _engine.setDefaultAudioRoutetoSpeakerphone(true);
  82. // start joining channel
  83. // 1. Users can only see each other after they join the
  84. // same channel successfully using the same app id.
  85. // 2. If app certificate is turned on at dashboard, token is needed
  86. // when joining channel. The channel name and uid used to calculate
  87. // the token has to match the ones used for channel join
  88. await _engine.joinChannel(config.token, config.channelId, null, 0, null);
  89. }
  90. _addListener() {
  91. _engine.setEventHandler(RtcEngineEventHandler(warning: (warningCode) {
  92. log('Warning ${warningCode}');
  93. }, error: (errorCode) {
  94. log('Warning ${errorCode}');
  95. }, joinChannelSuccess: (channel, uid, elapsed) {
  96. log('joinChannelSuccess ${channel} ${uid} ${elapsed}');
  97. setState(() {
  98. isJoined = true;
  99. });
  100. }, userJoined: (uid, elapsed) {
  101. log('userJoined $uid $elapsed');
  102. this.setState(() {
  103. remoteUid = uid;
  104. });
  105. }, userOffline: (uid, reason) {
  106. log('userOffline $uid $reason');
  107. this.setState(() {
  108. remoteUid = null;
  109. });
  110. }));
  111. }
  112. _updateClientRole(ClientRole role) async {
  113. var option;
  114. if (role == ClientRole.Broadcaster) {
  115. await _engine.setVideoEncoderConfiguration(VideoEncoderConfiguration(
  116. dimensions: VideoDimensions(width: 640, height: 360),
  117. frameRate: VideoFrameRate.Fps30,
  118. orientationMode: VideoOutputOrientationMode.Adaptative));
  119. // enable camera/mic, this will bring up permission dialog for first time
  120. await _engine.enableLocalAudio(true);
  121. await _engine.enableLocalVideo(true);
  122. } else {
  123. // You have to provide client role options if set to audience
  124. option = ClientRoleOptions(
  125. audienceLatencyLevel: isLowAudio
  126. ? AudienceLatencyLevelType.LowLatency
  127. : AudienceLatencyLevelType.UltraLowLatency);
  128. }
  129. await _engine.setClientRole(role, option);
  130. }
  131. _onPressToggleRole() {
  132. this.setState(() {
  133. role = role == ClientRole.Audience
  134. ? ClientRole.Broadcaster
  135. : ClientRole.Audience;
  136. _updateClientRole(role);
  137. });
  138. }
  139. _onPressToggleLatencyLevel(value) {
  140. this.setState(() {
  141. isLowAudio = !isLowAudio;
  142. _engine.setClientRole(
  143. ClientRole.Audience,
  144. ClientRoleOptions(
  145. audienceLatencyLevel: isLowAudio
  146. ? AudienceLatencyLevelType.LowLatency
  147. : AudienceLatencyLevelType.UltraLowLatency));
  148. });
  149. }
  150. _renderToolBar() {
  151. return Positioned(
  152. child: Column(
  153. crossAxisAlignment: CrossAxisAlignment.start,
  154. children: [
  155. ElevatedButton(
  156. child: Text('Toggle Role'),
  157. onPressed: _onPressToggleRole,
  158. ),
  159. Container(
  160. color: Colors.white,
  161. child: Row(mainAxisSize: MainAxisSize.min, children: [
  162. Text('Toggle Audience Latency Level'),
  163. Switch(
  164. value: isLowAudio,
  165. onChanged: _onPressToggleLatencyLevel,
  166. activeTrackColor: Colors.grey[350],
  167. activeColor: Colors.white,
  168. ),
  169. ]),
  170. )
  171. ],
  172. ),
  173. left: 10,
  174. bottom: 10,
  175. );
  176. }
  177. @override
  178. Widget build(BuildContext context) {
  179. return Stack(
  180. children: [
  181. Column(
  182. children: [
  183. Row(
  184. children: [
  185. if (!isJoined)
  186. Expanded(
  187. flex: 1,
  188. child: ElevatedButton(
  189. onPressed: _initEngine,
  190. child: Text('Join channel'),
  191. ),
  192. )
  193. ],
  194. ),
  195. if (isJoined) _renderVideo(),
  196. ],
  197. ),
  198. if (isJoined) _renderToolBar(),
  199. ],
  200. );
  201. }
  202. _renderVideo() {
  203. return Expanded(
  204. child: Stack(
  205. children: [
  206. role == ClientRole.Broadcaster
  207. ? RtcLocalView.SurfaceView()
  208. : remoteUid != null
  209. ? RtcRemoteView.SurfaceView(
  210. uid: remoteUid!,
  211. )
  212. : Container()
  213. ],
  214. ),
  215. );
  216. }
  217. }