voice_change.dart 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. import 'dart:developer';
  2. import 'package:agora_rtc_engine/rtc_engine.dart';
  3. import 'package:agora_rtc_engine_example/config/agora.config.dart' as config;
  4. import 'package:agora_rtc_engine_example/examples/config/voice_changer.config.dart';
  5. import 'package:flutter/cupertino.dart';
  6. import 'package:flutter/foundation.dart';
  7. import 'package:flutter/material.dart';
  8. import 'package:permission_handler/permission_handler.dart';
  9. /// VoiceChange Example
  10. class VoiceChange extends StatefulWidget {
  11. @override
  12. State<StatefulWidget> createState() => _State();
  13. }
  14. class _State extends State<VoiceChange> {
  15. late final RtcEngine _engine;
  16. bool isJoined = false;
  17. List<int> remoteUids = [];
  18. int? uidMySelf;
  19. int? selectedVoiceToolBtn;
  20. AudioEffectPreset currentAudioEffectPreset = AudioEffectPreset.AudioEffectOff;
  21. bool isEnableSlider1 = false, isEnableSlider2 = false;
  22. String sliderTitle1 = '', sliderTitle2 = '';
  23. double minimumValue1 = 0,
  24. maximumValue1 = 0,
  25. minimumValue2 = 0,
  26. maximumValue2 = 0;
  27. double? sliderValue1, sliderValue2;
  28. Map selectedFreq = FreqOptions[0];
  29. Map selectedReverbKey = ReverbKeyOptions[0];
  30. double voicePitchValue = 0.5;
  31. double bandGainValue = 0;
  32. double reverbValue = 1;
  33. @override
  34. void initState() {
  35. super.initState();
  36. }
  37. @override
  38. void dispose() {
  39. super.dispose();
  40. _engine.destroy();
  41. }
  42. _initEngine() async {
  43. if (defaultTargetPlatform == TargetPlatform.android) {
  44. await Permission.microphone.request();
  45. }
  46. _engine = await RtcEngine.createWithContext(RtcEngineContext(config.appId));
  47. this._addListener();
  48. // make this room live broadcasting room
  49. await _engine.setChannelProfile(ChannelProfile.LiveBroadcasting);
  50. await _engine.setClientRole(ClientRole.Broadcaster);
  51. // Set audio route to speaker
  52. await _engine.setDefaultAudioRoutetoSpeakerphone(true);
  53. // start joining channel
  54. // 1. Users can only see each other after they join the
  55. // same channel successfully using the same app id.
  56. // 2. If app certificate is turned on at dashboard, token is needed
  57. // when joining channel. The channel name and uid used to calculate
  58. // the token has to match the ones used for channel join
  59. await _engine.joinChannel(config.token, config.channelId, null, 0, null);
  60. }
  61. _addListener() {
  62. _engine.setEventHandler(RtcEngineEventHandler(warning: (warningCode) {
  63. log('Warning ${warningCode}');
  64. }, error: (errorCode) {
  65. log('Warning ${errorCode}');
  66. }, joinChannelSuccess: (channel, uid, elapsed) {
  67. log('joinChannelSuccess ${channel} ${uid} ${elapsed}');
  68. ;
  69. setState(() {
  70. isJoined = true;
  71. uidMySelf = uid;
  72. });
  73. }, userJoined: (uid, elapsed) {
  74. log('userJoined $uid $elapsed');
  75. this.setState(() {
  76. remoteUids.add(uid);
  77. });
  78. }, userOffline: (uid, reason) {
  79. log('userOffline $uid $reason');
  80. this.setState(() {
  81. remoteUids.remove(uid);
  82. });
  83. }));
  84. }
  85. _onPressBFButton(dynamic type, int index) async {
  86. switch (index) {
  87. case 0:
  88. case 1:
  89. await _engine.setVoiceBeautifierPreset(type as VoiceBeautifierPreset);
  90. this._updateSliderUI(AudioEffectPreset.AudioEffectOff);
  91. break;
  92. case 2:
  93. case 3:
  94. case 4:
  95. case 5:
  96. await _engine.setAudioEffectPreset(type as AudioEffectPreset);
  97. this._updateSliderUI(type);
  98. break;
  99. default:
  100. break;
  101. }
  102. }
  103. _updateSliderUI(AudioEffectPreset type) {
  104. this.setState(() {
  105. currentAudioEffectPreset = type;
  106. switch (type) {
  107. case AudioEffectPreset.RoomAcoustics3DVoice:
  108. isEnableSlider1 = true;
  109. isEnableSlider2 = false;
  110. sliderTitle1 = 'Cycle';
  111. minimumValue1 = 1;
  112. sliderValue1 = 1;
  113. maximumValue1 = 3;
  114. break;
  115. case AudioEffectPreset.PitchCorrection:
  116. isEnableSlider1 = true;
  117. isEnableSlider2 = true;
  118. sliderTitle1 = 'Tonic Mode';
  119. sliderTitle2 = 'Tonic Pitch';
  120. minimumValue1 = 1;
  121. sliderValue1 = 1;
  122. maximumValue1 = 3;
  123. minimumValue2 = 1;
  124. sliderValue2 = 1;
  125. maximumValue2 = 12;
  126. break;
  127. default:
  128. isEnableSlider1 = false;
  129. isEnableSlider2 = false;
  130. break;
  131. }
  132. });
  133. }
  134. _onAudioEffectUpdate({
  135. double? value1,
  136. double? value2,
  137. }) async {
  138. this.setState(() {
  139. if (value1 != null) {
  140. sliderValue1 = value1;
  141. }
  142. if (value2 != null) {
  143. sliderValue2 = value2;
  144. }
  145. _engine.setAudioEffectParameters(
  146. currentAudioEffectPreset,
  147. (isEnableSlider1 ? sliderValue1 ?? minimumValue1 : 0).toInt(),
  148. (isEnableSlider2 ? sliderValue2 ?? minimumValue2 : 0).toInt());
  149. });
  150. }
  151. _onPressChangeFreq() {
  152. return showDialog(
  153. context: context,
  154. barrierDismissible: false,
  155. builder: (BuildContext context) {
  156. return AlertDialog(
  157. title: Text('Set Band Frequency'),
  158. actions: <Widget>[
  159. for (var freqOpt in FreqOptions)
  160. TextButton(
  161. child: Text(freqOpt['text'] as String),
  162. onPressed: () {
  163. setState(() {
  164. selectedFreq = freqOpt;
  165. _engine.setLocalVoiceEqualization(
  166. freqOpt['type'] as AudioEqualizationBandFrequency,
  167. bandGainValue.toInt());
  168. });
  169. Navigator.of(context).pop();
  170. },
  171. ),
  172. ],
  173. );
  174. },
  175. );
  176. }
  177. _onPressChangeReverbKey() {
  178. return showDialog(
  179. context: context,
  180. barrierDismissible: false,
  181. builder: (BuildContext context) {
  182. return AlertDialog(
  183. title: Text('Set Reverb Key'),
  184. actions: <Widget>[
  185. for (var reverbKey in ReverbKeyOptions)
  186. TextButton(
  187. child: Text(reverbKey['text'] as String),
  188. onPressed: () {
  189. setState(() {
  190. selectedReverbKey = reverbKey;
  191. });
  192. Navigator.of(context).pop();
  193. },
  194. ),
  195. ],
  196. );
  197. },
  198. );
  199. }
  200. _renderToolBar() {
  201. return Padding(
  202. padding: EdgeInsets.symmetric(horizontal: 8, vertical: 0),
  203. child: Column(
  204. children: [
  205. Row(children: [
  206. Text('Voice Beautifier & Effects Preset',
  207. style: TextStyle(fontWeight: FontWeight.bold))
  208. ]),
  209. Container(
  210. width: MediaQuery.of(context).size.width,
  211. child: Wrap(children: [
  212. for (var i = 0; i < VoiceChangeConfig.length; i++)
  213. _renderBtnItem(VoiceChangeConfig[i], i)
  214. ])),
  215. if (isEnableSlider1)
  216. Row(
  217. mainAxisAlignment: MainAxisAlignment.end,
  218. mainAxisSize: MainAxisSize.max,
  219. children: [
  220. Expanded(
  221. child: Text(sliderTitle1),
  222. flex: 1,
  223. ),
  224. Expanded(
  225. child: Slider(
  226. value: sliderValue1!,
  227. min: minimumValue1,
  228. max: maximumValue1,
  229. divisions: 5,
  230. label: sliderTitle1,
  231. onChanged: (double value) {
  232. _onAudioEffectUpdate(value1: value);
  233. },
  234. ),
  235. flex: 2,
  236. )
  237. ],
  238. ),
  239. if (isEnableSlider2)
  240. Row(
  241. mainAxisAlignment: MainAxisAlignment.end,
  242. mainAxisSize: MainAxisSize.max,
  243. children: [
  244. Expanded(child: Text(sliderTitle2), flex: 1),
  245. Expanded(
  246. child: Slider(
  247. value: sliderValue2!,
  248. min: minimumValue2,
  249. max: maximumValue2,
  250. divisions: 5,
  251. label: sliderTitle1,
  252. onChanged: (double value) {
  253. _onAudioEffectUpdate(value2: value);
  254. },
  255. ),
  256. flex: 2)
  257. ],
  258. ),
  259. Row(children: [
  260. Text('Customize Voice Effects',
  261. style: TextStyle(fontWeight: FontWeight.bold))
  262. ]),
  263. Row(
  264. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  265. children: [
  266. Text('Pitch:'),
  267. Slider(
  268. value: voicePitchValue,
  269. min: 0.5,
  270. max: 2,
  271. divisions: 5,
  272. onChanged: (double value) {
  273. setState(() {
  274. voicePitchValue = value;
  275. });
  276. _engine.setLocalVoicePitch(value);
  277. },
  278. )
  279. ],
  280. ),
  281. Row(
  282. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  283. children: [
  284. Text('BandFreq'),
  285. TextButton(
  286. child: Text(selectedFreq['text']),
  287. onPressed: _onPressChangeFreq,
  288. )
  289. ],
  290. ),
  291. Row(
  292. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  293. children: [
  294. Text('BandGain:'),
  295. Slider(
  296. value: bandGainValue,
  297. min: 0,
  298. max: 9,
  299. divisions: 5,
  300. onChanged: (double value) async {
  301. this.setState(() {
  302. bandGainValue = value;
  303. });
  304. _engine.setLocalVoiceEqualization(
  305. selectedFreq['type'], value.toInt());
  306. },
  307. )
  308. ],
  309. ),
  310. Row(
  311. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  312. children: [
  313. Text('BandKey'),
  314. TextButton(
  315. child: Text(selectedReverbKey['text']),
  316. onPressed: _onPressChangeReverbKey,
  317. )
  318. ],
  319. ),
  320. Row(
  321. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  322. children: [
  323. Text('ReverbValue:'),
  324. Slider(
  325. value: reverbValue,
  326. min: selectedReverbKey['min'],
  327. max: selectedReverbKey['max'],
  328. divisions: 5,
  329. onChanged: (double value) async {
  330. setState(() {
  331. reverbValue = value;
  332. });
  333. await _engine.setLocalVoiceReverb(
  334. selectedReverbKey['type'], value.toInt());
  335. },
  336. )
  337. ],
  338. ),
  339. ],
  340. ));
  341. }
  342. _renderBtnItem(Map<String, dynamic> config, int index) {
  343. return _CusBtn(
  344. config['alertTitle'], selectedVoiceToolBtn != index, config['options'],
  345. (type) {
  346. setState(() {
  347. selectedVoiceToolBtn = index;
  348. });
  349. _onPressBFButton(type, index);
  350. });
  351. }
  352. @override
  353. Widget build(BuildContext context) {
  354. return Stack(
  355. children: [
  356. Column(
  357. children: [
  358. !isJoined
  359. ? Row(
  360. children: [
  361. Expanded(
  362. flex: 1,
  363. child: ElevatedButton(
  364. onPressed: _initEngine,
  365. child: Text('Join channel'),
  366. ),
  367. )
  368. ],
  369. )
  370. : _renderUserUid(),
  371. if (isJoined) _renderToolBar()
  372. ],
  373. ),
  374. ],
  375. );
  376. }
  377. _renderUserUid() {
  378. final size = MediaQuery.of(context).size;
  379. var list = [uidMySelf, ...remoteUids];
  380. return Container(
  381. width: size.width,
  382. height: 200,
  383. child: ListView.builder(
  384. itemCount: list.length,
  385. itemBuilder: (context, index) {
  386. return Center(
  387. child: Padding(
  388. child: Text(
  389. 'AUDIO ONLY ${index == 0 ? 'LOCAL' : 'REMOTE'} UID: ${list[index]}',
  390. style: TextStyle(fontSize: 14, color: Colors.black),
  391. ),
  392. padding: EdgeInsets.all(4.0),
  393. ),
  394. );
  395. },
  396. ),
  397. );
  398. }
  399. }
  400. class _CusBtn extends StatefulWidget {
  401. String alertTitle;
  402. dynamic options;
  403. bool isOff = false;
  404. void Function(dynamic type) onPressed;
  405. _CusBtn(this.alertTitle, this.isOff, this.options, this.onPressed);
  406. @override
  407. State<StatefulWidget> createState() => _CusBtnState(isOff);
  408. }
  409. class _CusBtnState extends State<_CusBtn> {
  410. String title = "Off";
  411. bool isEnable;
  412. _CusBtnState(this.isEnable);
  413. @override
  414. void didUpdateWidget(covariant _CusBtn oldWidget) {
  415. super.didUpdateWidget(oldWidget);
  416. this.setState(() {
  417. isEnable = !widget.isOff;
  418. });
  419. }
  420. @override
  421. Widget build(BuildContext context) {
  422. return TextButton(
  423. child: Text(isEnable ? title : 'Off'),
  424. onPressed: _showMyDialog,
  425. );
  426. }
  427. Future<void> _showMyDialog() async {
  428. return showDialog(
  429. context: context,
  430. barrierDismissible: false,
  431. builder: (BuildContext context) {
  432. return AlertDialog(
  433. title: Text(widget.alertTitle),
  434. actions: <Widget>[
  435. for (var option in widget.options)
  436. TextButton(
  437. child: Text(option['text']),
  438. onPressed: () {
  439. setState(() {
  440. isEnable = true;
  441. title = option['text'];
  442. widget.onPressed(option['type']);
  443. });
  444. Navigator.of(context).pop();
  445. },
  446. ),
  447. ],
  448. );
  449. },
  450. );
  451. }
  452. }