live_streaming.dart 6.8 KB

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