join_channel_video.dart 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  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. /// MultiChannel Example
  11. class JoinChannelVideo extends StatefulWidget {
  12. @override
  13. State<StatefulWidget> createState() => _State();
  14. }
  15. class _State extends State<JoinChannelVideo> {
  16. late final RtcEngine _engine;
  17. String channelId = config.channelId;
  18. bool isJoined = false, switchCamera = true, switchRender = true;
  19. List<int> remoteUid = [];
  20. TextEditingController? _controller;
  21. @override
  22. void initState() {
  23. super.initState();
  24. _controller = TextEditingController(text: channelId);
  25. this._initEngine();
  26. }
  27. @override
  28. void dispose() {
  29. super.dispose();
  30. _engine.destroy();
  31. }
  32. _initEngine() async {
  33. _engine = await RtcEngine.createWithContext(RtcEngineContext(config.appId));
  34. this._addListeners();
  35. await _engine.enableVideo();
  36. await _engine.startPreview();
  37. await _engine.setChannelProfile(ChannelProfile.LiveBroadcasting);
  38. await _engine.setClientRole(ClientRole.Broadcaster);
  39. }
  40. _addListeners() {
  41. _engine.setEventHandler(RtcEngineEventHandler(
  42. joinChannelSuccess: (channel, uid, elapsed) {
  43. log('joinChannelSuccess ${channel} ${uid} ${elapsed}');
  44. setState(() {
  45. isJoined = true;
  46. });
  47. },
  48. userJoined: (uid, elapsed) {
  49. log('userJoined ${uid} ${elapsed}');
  50. setState(() {
  51. remoteUid.add(uid);
  52. });
  53. },
  54. userOffline: (uid, reason) {
  55. log('userOffline ${uid} ${reason}');
  56. setState(() {
  57. remoteUid.removeWhere((element) => element == uid);
  58. });
  59. },
  60. leaveChannel: (stats) {
  61. log('leaveChannel ${stats.toJson()}');
  62. setState(() {
  63. isJoined = false;
  64. remoteUid.clear();
  65. });
  66. },
  67. ));
  68. }
  69. _joinChannel() async {
  70. if (defaultTargetPlatform == TargetPlatform.android) {
  71. await [Permission.microphone, Permission.camera].request();
  72. }
  73. await _engine.joinChannel(config.token, channelId, null, config.uid);
  74. }
  75. _leaveChannel() async {
  76. await _engine.leaveChannel();
  77. }
  78. _switchCamera() {
  79. _engine.switchCamera().then((value) {
  80. setState(() {
  81. switchCamera = !switchCamera;
  82. });
  83. }).catchError((err) {
  84. log('switchCamera $err');
  85. });
  86. }
  87. _switchRender() {
  88. setState(() {
  89. switchRender = !switchRender;
  90. remoteUid = List.of(remoteUid.reversed);
  91. });
  92. }
  93. @override
  94. Widget build(BuildContext context) {
  95. return Stack(
  96. children: [
  97. Column(
  98. children: [
  99. TextField(
  100. controller: _controller,
  101. decoration: InputDecoration(hintText: 'Channel ID'),
  102. onChanged: (text) {
  103. setState(() {
  104. channelId = text;
  105. });
  106. },
  107. ),
  108. Row(
  109. children: [
  110. Expanded(
  111. flex: 1,
  112. child: ElevatedButton(
  113. onPressed:
  114. isJoined ? this._leaveChannel : this._joinChannel,
  115. child: Text('${isJoined ? 'Leave' : 'Join'} channel'),
  116. ),
  117. )
  118. ],
  119. ),
  120. _renderVideo(),
  121. ],
  122. ),
  123. Align(
  124. alignment: Alignment.bottomRight,
  125. child: Column(
  126. mainAxisSize: MainAxisSize.min,
  127. children: [
  128. ElevatedButton(
  129. onPressed: this._switchCamera,
  130. child: Text('Camera ${switchCamera ? 'front' : 'rear'}'),
  131. ),
  132. ],
  133. ),
  134. )
  135. ],
  136. );
  137. }
  138. _renderVideo() {
  139. return Expanded(
  140. child: Stack(
  141. children: [
  142. RtcLocalView.SurfaceView(),
  143. Align(
  144. alignment: Alignment.topLeft,
  145. child: SingleChildScrollView(
  146. scrollDirection: Axis.horizontal,
  147. child: Row(
  148. children: List.of(remoteUid.map(
  149. (e) => GestureDetector(
  150. onTap: this._switchRender,
  151. child: Container(
  152. width: 120,
  153. height: 120,
  154. child: RtcRemoteView.SurfaceView(
  155. uid: e,
  156. ),
  157. ),
  158. ),
  159. )),
  160. ),
  161. ),
  162. )
  163. ],
  164. ),
  165. );
  166. }
  167. }