create_stream_data.dart 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  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. /// CreateStreamData Example
  11. class CreateStreamData extends StatefulWidget {
  12. @override
  13. State<StatefulWidget> createState() => _State();
  14. }
  15. class _State extends State<CreateStreamData> {
  16. late final RtcEngine _engine;
  17. bool isJoined = false;
  18. int? remoteUid;
  19. @override
  20. void initState() {
  21. super.initState();
  22. }
  23. @override
  24. void dispose() {
  25. super.dispose();
  26. _engine.destroy();
  27. }
  28. _initEngine() async {
  29. if (defaultTargetPlatform == TargetPlatform.android) {
  30. await Permission.microphone.request();
  31. }
  32. _engine = await RtcEngine.createWithContext(RtcEngineContext(config.appId));
  33. this._addListener();
  34. // enable video module and set up video encoding configs
  35. await _engine.enableVideo();
  36. // make this room live broadcasting room
  37. await _engine.setChannelProfile(ChannelProfile.LiveBroadcasting);
  38. await _engine.setClientRole(ClientRole.Broadcaster);
  39. // Set audio route to speaker
  40. await _engine.setDefaultAudioRoutetoSpeakerphone(true);
  41. // start joining channel
  42. // 1. Users can only see each other after they join the
  43. // same channel successfully using the same app id.
  44. // 2. If app certificate is turned on at dashboard, token is needed
  45. // when joining channel. The channel name and uid used to calculate
  46. // the token has to match the ones used for channel join
  47. await _engine.joinChannel(config.token, config.channelId, null, 0, null);
  48. }
  49. Future<void> _showMyDialog(int uid, int streamId, String data) async {
  50. return showDialog(
  51. context: context,
  52. barrierDismissible: false, // user must tap button!
  53. builder: (BuildContext context) {
  54. return AlertDialog(
  55. title: Text('Receive from uid:${uid}'),
  56. content: SingleChildScrollView(
  57. child: ListBody(
  58. children: <Widget>[Text('StreamId ${streamId}:${data}')],
  59. ),
  60. ),
  61. actions: <Widget>[
  62. TextButton(
  63. child: Text('Ok'),
  64. onPressed: () {
  65. Navigator.of(context).pop();
  66. },
  67. ),
  68. ],
  69. );
  70. },
  71. );
  72. }
  73. _addListener() {
  74. _engine.setEventHandler(RtcEngineEventHandler(warning: (warningCode) {
  75. log('Warning ${warningCode}');
  76. }, error: (errorCode) {
  77. log('Warning ${errorCode}');
  78. }, joinChannelSuccess: (channel, uid, elapsed) {
  79. log('joinChannelSuccess ${channel} ${uid} ${elapsed}');
  80. ;
  81. setState(() {
  82. isJoined = true;
  83. });
  84. }, userJoined: (uid, elapsed) {
  85. log('userJoined $uid $elapsed');
  86. this.setState(() {
  87. remoteUid = uid;
  88. });
  89. }, userOffline: (uid, reason) {
  90. log('userOffline $uid $reason');
  91. this.setState(() {
  92. remoteUid = null;
  93. });
  94. }, streamMessage: (int uid, int streamId, String data) {
  95. _showMyDialog(uid, streamId, data);
  96. log('streamMessage $uid $streamId $data');
  97. }, streamMessageError:
  98. (int uid, int streamId, ErrorCode error, int missed, int cached) {
  99. log('streamMessage $uid $streamId $error $missed $cached');
  100. }));
  101. }
  102. _onPressSend() async {
  103. if (_controller.text.length == 0) {
  104. return;
  105. }
  106. var streamId = await _engine
  107. .createDataStreamWithConfig(DataStreamConfig(false, false));
  108. if (streamId != null) {
  109. _engine.sendStreamMessage(streamId, _controller.text);
  110. }
  111. _controller.clear();
  112. }
  113. final TextEditingController _controller = TextEditingController();
  114. @override
  115. Widget build(BuildContext context) {
  116. return Stack(
  117. children: [
  118. Column(
  119. children: [
  120. !isJoined
  121. ? Row(
  122. children: [
  123. Expanded(
  124. flex: 1,
  125. child: ElevatedButton(
  126. onPressed: _initEngine,
  127. child: Text('Join channel'),
  128. ),
  129. )
  130. ],
  131. )
  132. : _renderVideo(),
  133. if (isJoined)
  134. Row(
  135. mainAxisSize: MainAxisSize.max,
  136. children: [
  137. Expanded(
  138. child: TextField(
  139. controller: _controller,
  140. decoration: InputDecoration(
  141. hintText: 'Input Message',
  142. ))),
  143. ElevatedButton(
  144. onPressed: _onPressSend,
  145. child: Text('Send'),
  146. ),
  147. ],
  148. )
  149. ],
  150. ),
  151. ],
  152. );
  153. }
  154. _renderVideo() {
  155. return Row(children: [
  156. Expanded(
  157. child: AspectRatio(
  158. aspectRatio: 1,
  159. child: RtcLocalView.SurfaceView(),
  160. )),
  161. Expanded(
  162. child: AspectRatio(
  163. aspectRatio: 1,
  164. child: remoteUid != null
  165. ? RtcRemoteView.SurfaceView(
  166. uid: remoteUid!,
  167. )
  168. : Container(
  169. color: Colors.grey[200],
  170. ),
  171. ),
  172. )
  173. ]);
  174. }
  175. }