You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

219 lines
6.1 KiB

  1. import 'dart:async';
  2. import 'dart:io';
  3. import 'dart:typed_data';
  4. import 'package:record/record.dart';
  5. import 'package:flutter/material.dart';
  6. import 'package:logger/logger.dart';
  7. import 'package:path_provider/path_provider.dart';
  8. import 'package:permission_handler/permission_handler.dart';
  9. import 'package:demo001/xunfei/xunfei_translate.dart';
  10. /*
  11. 录音测试场景
  12. */
  13. class RecordScene extends StatefulWidget {
  14. @override
  15. _RecordSceneState createState() => _RecordSceneState();
  16. }
  17. class _RecordSceneState extends State<RecordScene> {
  18. late Directory savedirectory;
  19. final XunFeiTranslate xunfei = XunFeiTranslate(
  20. appId: "137dc132",
  21. apiKey: "1c1891a475e71250ecd1320303ad6545",
  22. apiSecret: "MjZhNDA1NTI1NWZkZDQxOTMxYzMyN2Yw");
  23. AudioRecorder _recorder = AudioRecorder();
  24. bool _isRecorderReady = false; //是否录音已准备
  25. bool _isRecording = false; //是否录音中
  26. @override
  27. void initState() {
  28. super.initState();
  29. _requestPermissions();
  30. }
  31. // 初始化录音器
  32. void _initRecorder() async {
  33. try {
  34. // 获取外部存储目录路径
  35. savedirectory = (await getExternalStorageDirectory())!;
  36. setState(() {
  37. _isRecorderReady = true;
  38. });
  39. _log('录音器初始化成功');
  40. } catch (e) {
  41. _log('初始化录音器失败: $e');
  42. setState(() {
  43. _isRecorderReady = false;
  44. });
  45. }
  46. }
  47. // 请求麦克风权限
  48. void _requestPermissions() async {
  49. try {
  50. if (await Permission.microphone.request().isGranted) {
  51. _log('麦克风权限已授予');
  52. } else {
  53. _log('麦克风权限被拒绝');
  54. setState(() {
  55. _isRecorderReady = false;
  56. });
  57. }
  58. } catch (e) {
  59. _log('请求麦克风权限失败: $e');
  60. setState(() {
  61. _isRecorderReady = false;
  62. });
  63. }
  64. }
  65. // 切换按钮状态
  66. void _toggleCallStatus() {
  67. if (!_isRecording) {
  68. //开始通话
  69. _startRecorder();
  70. } else {
  71. //结束通话
  72. _stopRecorder();
  73. }
  74. setState(() {
  75. _isRecording = !_isRecording;
  76. });
  77. }
  78. //开始录音
  79. void _startRecorder() async {
  80. try {
  81. if (!_isRecorderReady) {
  82. _initRecorder();
  83. }
  84. if (_isRecording) return; // 防止重复调用
  85. Stream<Uint8List> dataStream = await _recorder.startStream(RecordConfig(
  86. sampleRate: 16000, encoder: AudioEncoder.pcm16bits, numChannels: 1));
  87. xunfei.starttranslate(dataStream);
  88. setState(() {
  89. _isRecording = true;
  90. });
  91. _log('录音开始');
  92. } catch (e) {
  93. _log('录音开始 异常: $e');
  94. }
  95. }
  96. //结束录音
  97. void _stopRecorder() async {
  98. try {
  99. if (!_isRecording) return; // 防止重复调用
  100. await _recorder.stop();
  101. await _recorder.cancel();
  102. xunfei.stoptranslate();
  103. setState(() {
  104. _isRecorderReady = false;
  105. _isRecording = false;
  106. });
  107. _log('录音停止');
  108. } catch (e) {
  109. _log('录音停止 异常: $e');
  110. }
  111. }
  112. @override
  113. Widget build(BuildContext context) {
  114. return Scaffold(
  115. appBar: AppBar(title: Text("录音测试")),
  116. // body: ListView.builder(
  117. // itemCount: _records.length,
  118. // itemBuilder: (context, index) {
  119. // var audio = _records[index];
  120. // return _buildAudioMessage(audio);
  121. // },
  122. // ),
  123. bottomNavigationBar: Padding(
  124. padding: const EdgeInsets.all(20.0),
  125. child: InkWell(
  126. onTap: _toggleCallStatus,
  127. child: Container(
  128. decoration: BoxDecoration(
  129. borderRadius: BorderRadius.circular(30), // 圆角按钮
  130. color: _isRecording
  131. ? Colors.red
  132. : Colors.green, // 通话状态红色,非通话状态绿色
  133. ),
  134. padding: EdgeInsets.symmetric(
  135. vertical: 15, horizontal: 40), // 调整按钮大小
  136. child: Row(
  137. mainAxisAlignment: MainAxisAlignment.center,
  138. children: [
  139. Icon(
  140. _isRecording ? Icons.call_end : Icons.mic, // 图标变化
  141. color: Colors.white,
  142. size: 30,
  143. ),
  144. SizedBox(width: 10),
  145. Text(
  146. _isRecording ? '挂断' : '开始通话', // 状态文字变化
  147. style: TextStyle(
  148. color: Colors.white,
  149. fontSize: 18,
  150. ),
  151. ),
  152. ],
  153. ),
  154. ))),
  155. );
  156. }
  157. // 构建语音消息
  158. // Widget _buildAudioMessage(RecordData data) {
  159. // Color buttColor = data.state == 0 ? Colors.red : Colors.green;
  160. // return Padding(
  161. // padding: const EdgeInsets.symmetric(vertical: 10, horizontal: 15),
  162. // child: Column(
  163. // crossAxisAlignment: CrossAxisAlignment.start,
  164. // children: [
  165. // // 音频播放按钮
  166. // GestureDetector(
  167. // onTap: () {
  168. // // _playRecording(data);
  169. // },
  170. // child: Container(
  171. // padding: EdgeInsets.symmetric(vertical: 10, horizontal: 20),
  172. // decoration: BoxDecoration(
  173. // color: buttColor,
  174. // borderRadius: BorderRadius.circular(30),
  175. // ),
  176. // child: Row(
  177. // children: [
  178. // Icon(
  179. // Icons.play_arrow,
  180. // color: Colors.white,
  181. // ),
  182. // SizedBox(width: 10),
  183. // Text(
  184. // '播放音频',
  185. // style: TextStyle(color: Colors.white),
  186. // ),
  187. // ],
  188. // ),
  189. // ),
  190. // ),
  191. // SizedBox(height: 5),
  192. // // 文字内容
  193. // // Text(
  194. // // message['text'],
  195. // // style: TextStyle(fontSize: 16),
  196. // // ),
  197. // ],
  198. // ),
  199. // );
  200. // }
  201. void _log(String msg) {
  202. Logger().f("LIWEI---------------:$msg");
  203. }
  204. }