speechtotext.vue 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. <template>
  2. <s-layout class="chat-wrap" title="语音转文字">
  3. <!-- {{messages}} -->
  4. <div class="chat-container">
  5. <div class="messages" :style="{height:messagesHeight+'px'}" id="messages">
  6. <div v-for="(message, index) in messages" :key="index" class="message" @click="playRecording(index)">
  7. <div class="bubble">
  8. <span class="duration">{{ message.duration }}" </span><span class="ss-m-l-10">
  9. <image src="@/static/icon/audioPaly.png" class="audioPaly" />
  10. </span>
  11. </div>
  12. </div>
  13. </div>
  14. <div class="input-area">
  15. <button @mousedown="startRecording" @mouseup="stopRecording" @mouseleave="cancelRecording"
  16. @touchstart="startRecording" @touchend="stopRecording" @touchcancel="cancelRecording"
  17. @touchmove="handleTouchMove">
  18. 按住 说话
  19. </button>
  20. <div v-if="isRecording" class="recording-overlay">
  21. <div>{{ recordingDuration }}</div>
  22. <div>上滑至此取消</div>
  23. </div>
  24. </div>
  25. </div>
  26. </s-layout>
  27. </template>
  28. <script setup>
  29. import {
  30. ref,
  31. nextTick,
  32. onMounted
  33. } from 'vue';
  34. import sheep from '@/sheep';
  35. import VoiceApi from '@/sheep/api/system/voice';
  36. const {
  37. safeAreaInsets,
  38. safeArea
  39. } = sheep.$platform.device;
  40. const sysNavBar = sheep.$platform.navbar;
  41. const messagesHeight = (safeArea.height) - sysNavBar - 20 - 60;
  42. const messages = ref([]);
  43. const isRecording = ref(false);
  44. let startTime = ref(null);
  45. const recordingDuration = ref('');
  46. const minRecordingTime = 500; // 设置最小录音时间为500毫秒
  47. let intervalId = null;
  48. const cancelOffset = 280; // 上滑取消的距离阈值
  49. const startTouchY = ref(0);
  50. let mediaRecorder = null;
  51. let audioChunks = [];
  52. let isCancelled = ref(false);
  53. // 初始化录音功能
  54. onMounted(() => {
  55. navigator.mediaDevices.getUserMedia({
  56. audio: true
  57. })
  58. .then(stream => {
  59. mediaRecorder = new MediaRecorder(stream);
  60. mediaRecorder.ondataavailable = (event) => {
  61. audioChunks.push(event.data);
  62. };
  63. mediaRecorder.onstop = () => {
  64. if (!isCancelled.value) {
  65. sendDuration()
  66. }
  67. audioChunks = [];
  68. };
  69. })
  70. .catch(error => {
  71. console.error("Error accessing media devices.", error);
  72. });
  73. });
  74. async function sendAudioToServer(audioBlob) {
  75. const formData = new FormData();
  76. formData.append('audio_file', audioBlob);
  77. try {
  78. const response = await fetch('https://zxgz.newfeifan.cn/voice2text/', {
  79. method: 'POST',
  80. body: formData,
  81. });
  82. const data = await response.json();
  83. console.log('Server response:', data);
  84. } catch (error) {
  85. console.error('Error sending audio file:', error);
  86. }
  87. }
  88. // 发送录音
  89. const sendDuration = () => {
  90. const audioBlob = new Blob(audioChunks, {
  91. type: 'audio/mpeg'
  92. });
  93. const audioUrl = URL.createObjectURL(audioBlob);
  94. const duration = Math.max(Math.floor((new Date() - startTime.value) / 1000), 1);
  95. messages.value.push({
  96. duration,
  97. audioUrl
  98. });
  99. sendAudioToServer(audioBlob);
  100. nextTick(() => {
  101. let messagesElement = document.getElementById('messages');
  102. messagesElement.scrollTop = messagesElement.scrollHeight;
  103. });
  104. }
  105. // 更新录音
  106. const updateDuration = () => {
  107. const currentDuration = Math.floor((new Date() - startTime.value) / 1000);
  108. recordingDuration.value = currentDuration + 's';
  109. if (currentDuration >= 60) { // 如果到达60秒自动停止录音
  110. stopRecording(new Event('mouseup'));
  111. }
  112. };
  113. // 开始录音
  114. const startRecording = (event) => {
  115. if (!isRecording.value && (event.type === 'mousedown' || event.type === 'touchstart')) {
  116. startTime.value = new Date();
  117. isRecording.value = true;
  118. intervalId = setInterval(updateDuration, 1000); // 每秒更新一次时间
  119. recordingDuration.value = '1s'; // 开始时显示1秒
  120. // console.log('开始录音...');
  121. event.preventDefault();
  122. startTouchY.value = event.touches ? event.touches[0].clientY : 0;
  123. audioChunks = [];
  124. isCancelled.value = false; // 确保开始录音时取消标志为假
  125. mediaRecorder.start();
  126. }
  127. };
  128. // 停止录音
  129. const stopRecording = (event) => {
  130. if (isRecording.value && (event.type === 'mouseup' || event.type === 'touchend' || event.type ===
  131. 'mouseleave' || event.type === 'touchcancel')) {
  132. clearInterval(intervalId); // 停止定时器
  133. isRecording.value = false;
  134. recordingDuration.value = ''; // 清空显示的时间
  135. if (new Date() - startTime.value >= minRecordingTime) {
  136. mediaRecorder.stop();
  137. } else {
  138. console.log('录音时间太短,不保存');
  139. }
  140. }
  141. };
  142. // 取消录音
  143. const cancelRecording = () => {
  144. if (isRecording.value) {
  145. clearInterval(intervalId);
  146. console.log('录音取消');
  147. isCancelled.value = true; // 设置取消标志为真
  148. mediaRecorder.stop();
  149. resetRecording();
  150. }
  151. };
  152. // 重置录音
  153. const resetRecording = () => {
  154. isRecording.value = false;
  155. recordingDuration.value = 0;
  156. startTime.value = null;
  157. audioChunks = [];
  158. };
  159. // 处理触摸移动事件
  160. const handleTouchMove = (event) => {
  161. // 此处应添加逻辑来检测触摸位置,如果滑动到取消区域,调用 cancelRecording
  162. const currentTouchY = event.touches[0].clientY;
  163. if (startTouchY.value - currentTouchY > cancelOffset) { // 检查是否上滑超过阈值
  164. cancelRecording();
  165. }
  166. };
  167. // 播放录音
  168. const playRecording = (index) => {
  169. const message = messages.value[index];
  170. const audio = new Audio(message.audioUrl);
  171. audio.play();
  172. };
  173. </script>
  174. <style scoped>
  175. .chat-container {
  176. display: flex;
  177. flex-direction: column;
  178. overflow-y: auto;
  179. background-color: #f0f0f0;
  180. padding: 10px;
  181. }
  182. .messages {
  183. flex-grow: 1;
  184. overflow-y: auto;
  185. }
  186. .message {
  187. display: flex;
  188. justify-content: flex-end;
  189. /* 消息靠右 */
  190. margin-bottom: 10px;
  191. }
  192. .bubble {
  193. padding: 10px 20px 10px 50px;
  194. margin-right: 10px;
  195. background-color: rgb(131 235 96);
  196. border-radius: 5px;
  197. display: flex;
  198. align-items: center;
  199. position: relative;
  200. }
  201. .bubble::after {
  202. content: "";
  203. width: 0px;
  204. height: 0px;
  205. border-top: 5px solid transparent;
  206. border-bottom: 5px solid transparent;
  207. border-left: 5px solid #83eb60;
  208. position: absolute;
  209. top: 14px;
  210. right: -5px;
  211. }
  212. .duration {
  213. color: #000;
  214. font-size: 16px;
  215. }
  216. .audioPaly {
  217. width: 12px;
  218. height: 12px;
  219. }
  220. .input-area {
  221. position: fixed;
  222. bottom: 0;
  223. left: 0;
  224. width: 100%;
  225. background-color: rgb(245, 245, 245);
  226. display: flex;
  227. justify-content: center;
  228. padding: 10px;
  229. box-sizing: border-box;
  230. }
  231. .input-area>button {
  232. display: block;
  233. width: 100%;
  234. background-color: #FFFFFF;
  235. border: none;
  236. padding: 3px 0;
  237. border-radius: 5px;
  238. color: #333333;
  239. font-size: 16px;
  240. font-weight: bold;
  241. text-align: center;
  242. outline: none;
  243. cursor: pointer;
  244. }
  245. .input-area>.button-hover {
  246. background-color: #f0f0f0 !important;
  247. }
  248. .input-area>button:after {
  249. border: none;
  250. }
  251. .recording-overlay {
  252. position: fixed;
  253. top: 50%;
  254. left: 50%;
  255. transform: translate(-50%, -50%);
  256. background-color: rgba(0, 0, 0, 0.7);
  257. color: white;
  258. padding: 20px;
  259. border-radius: 10px;
  260. text-align: center;
  261. }
  262. .recording-overlay div {
  263. margin: 5px 0;
  264. }
  265. </style>