js录制音频
创始人
2024-11-15 04:34:03
0

整体思路

1、监听用户onTouchStart事件,设置一个定时器记录是否长按,然后调用JSBridge开始录制事件
2、通过JSBridge监听录音过程,拿到录音的数据,前端用数组变量存放
3、监听用户onTouchEnd松开事件,清除定时器,处理录音数组转换成一个文件上传到oss

难点

难点一:将base64录音片段转为WAV文件
首先将一系列Base64编码的音频段合并成一个PCM数据流;
然后创建一个WAV文件的头部信息;
最后合并WAV文件头和PCM数据

难点二:TypedArray数据的合并
TypedArray: 存储的是固定类型的数值数据,如整数或浮点数。
Array: 可以存储任何类型的数据,包括数字、字符串、对象等

开始录音

  /**    * 开始录音    */   const handleTouchStart = (event) => {     event.preventDefault();     timerId = setTimeout(() => {       setLongPress(true);       console.log('handleTouchStart 长按了');       JSBridge(XX.startRecording', {         numberOfChannels: 1, // 声道数         // sampleRate: 16000, // 采样率         sampleRate: 44100, // 更改采样率为 44100 Hz         bitsPerChannel: 16, // 位深         format: 'PCM',       }).then(() => {         setRecordStatus('dialog_listening');       });     }, 100); // 长按时长,这里设置为100ms   }; 

监听录音过程

 const onRecordChange = (event) => {     console.log(event);      const { error, param } = event || {};     const { pcm } = param || {};     const { errorCode, errorMsg } = error || {};      if (errorCode) {       Toast.show({         type: 'error',         content: `录制失败,${errorMsg}`,       });       baseArrayRef.current = [];     } else {       baseArrayRef.current.push(pcm);     }   };    useEffect(() => {     document.addEventListener('RecordingDataBufferTransfer', onRecordChange);      return () => {       // 清除长按定时器       if (timerId !== null) clearTimeout(timerId);     };   }, []); 

结束录制

/**    * 结束录音    * @returns    */   const handleTouchEnd = (event) => {     if (timerId !== null) {       clearTimeout(timerId)       timerId = null     }     if (!longPress) return;     setLongPress(false);     console.log('handleTouchEnd 松开了');     JSBridge('XX.stopRecording').then(() => {       // 移除事件监听器       document.removeEventListener(         'RecordingDataBufferTransfer',         onRecordChange,       );       setRecordStatus('dialog_sleep');       onMerge();     });   }; 

音频波动动画

VoiceAnimation/index.tsx

import cls from 'classnames'; import debounce from 'lodash/debounce'; import { useLayoutEffect, useMemo, useRef } from 'react'; import styles from './index.module.less';  interface IProps {   status: string; } export default function (props: IProps) {   const { status = 'dialog_sleep' } = props;   const list = useMemo(() => new Array(5).fill(true), []);    return (     
{list.map((_, index) => ( ))}
); } function getTransationByStatus(status: string, index?) { return { dialog_sleep: { transition: 'all 0.3s', height: '8px', transform: 'translateY(0)', }, dialog_idle: { transition: 'all 0.3s', height: '8px', transform: 'translateY(0)', }, dialog_listening: { transition: 'all 0.3s', height: '24px', transform: index % 2 ? 'translateY(8px)' : 'translateY(-8px)', onTransitionEnd: debounce( (event) => { if ( event.target.parentElement.className.indexOf('dialog_listening') === -1 ) return; event.target.style.transitionDuration = '0.5s'; event.target.style.height = '24px'; event.target.style.transform = event.target.style.transform === 'translateY(8px)' ? 'translateY(-8px)' : 'translateY(8px)'; }, { leading: true, trailing: false, }, ), }, dialog_thinking: { transition: 'all 0.3s', height: `${[52, 44, 36, 28, 24][index]}px`, transform: 'translateY(0)', onTransitionEnd: debounce( (event) => { if ( event.target.parentElement.className.indexOf('dialog_thinking') === -1 ) return; event.target.style.transitionDuration = '0.5s'; event.target.style.height = { '52px': '24px', '44px': '28px', '36px': '32px', '32px': '36px', '28px': '44px', '24px': '52px', }[event.target.style.height]; }, { leading: true, trailing: false, }, ), }, dialog_responding: { transition: 'all 0.2s', height: `${Math.random() * (index + 1) * 10 + 24}px`, transform: 'translateY(0)', onTransitionEnd: debounce( (event) => { if ( event.target.parentElement.className.indexOf( 'dialog_responding', ) === -1 ) return; event.target.style.transitionDuration = '0.15s'; event.target.style.height = `${Math.random() * (index + 1) * 10 + 24}px`; }, { leading: true, trailing: false, }, ), }, }[status]; } function AnimationItem({ status, index }: { status: string; index?: number }) { const div = useRef(); useLayoutEffect(() => { const container = div.current as HTMLDivElement; function reset() { container.ontransitionend = (e) => {}; container.style.transition = 'all .1s'; container.style.height = '24px'; container.style.transform = 'translateY(0)'; } reset(); const { onTransitionEnd = () => {}, ...style } = getTransationByStatus(status, index) || {}; container.ontransitionend = onTransitionEnd; for (let prop in style) { container.style[prop] = style[prop]; } return () => {}; }, [status]); return (
{ width: 24, height: 24 }} /> ); }

VoiceAnimation/index.module.less

.voice {   display: flex;   justify-content: center;   align-items: center;   height: 56px;    .item {     // width: 24px;     // height: 24px;     background-color: var(--TY-Text-Brand-1);     border-radius: 20px;     margin: 0 4px;     transform: translateY(0);   } }  .loop(@n, @i: 0) when (@i <= @n) {   &:nth-child(@{i}) {     animation-delay: (@i * 0.2s);   }   .loop(@n, (@i + 1)); } 

一个完整的音频录制——播放的例子

            pcmtowav      
getUserMedia需要https,使用localhost或127.0.0.1时,可用http。

相关内容

热门资讯

第4分钟辅助器!兴动休闲辅助(... 第4分钟辅助器!兴动休闲辅助(辅助挂)竟然真的是有挂(详细辅助普及教程)1、进入到兴动休闲辅助黑科技...
第8分钟插件!边锋干瞪眼跟有挂... 您好,边锋干瞪眼跟有挂吗这款游戏可以开挂的,确实是有挂的,需要了解加微【136704302】很多玩家...
黑科技辅助“微信小程序功夫川码... 黑科技辅助“微信小程序功夫川码辅助”外挂透视辅助脚本(原来是有挂)黑科技辅助“微信小程序功夫川码辅助...
5分钟透视!创思维激k有后台吗... 5分钟透视!创思维激k有后台吗(辅助挂)黑科技教程(原生存在有挂);5分钟透视!创思维激k有后台吗(...
3分钟辅助器!微乐家乡自建房辅... 3分钟辅助器!微乐家乡自建房辅助app(辅助挂)好像真的有挂(详细辅助教你攻略)该软件可以轻松地帮助...
黑科技辅助挂“赣湘互娱透视挂”... 黑科技辅助挂“赣湘互娱透视挂”外挂透视辅助科技(本来是真的有挂)1、下载好赣湘互娱透视挂辅助软件之后...
第8分钟辅助器!推筒子压桩公式... 第8分钟辅助器!推筒子压桩公式规律(辅助挂)必赢方法(真是真的是有挂)是一款可以让一直输的玩家,快速...
六分钟辅助!泸州大二新手攻略(... 六分钟辅助!泸州大二新手攻略(辅助挂)切实真的有挂(详细辅助我来教教你)1、泸州大二新手攻略透视辅助...
黑科技辅助“福建老友破解”外挂... 黑科技辅助“福建老友破解”外挂透视辅助软件(确实存在有挂);1、完成福建老友破解的残局,帮助玩家取得...
两分钟插件!闲玩暗宝辅助插件(... 两分钟插件!闲玩暗宝辅助插件(辅助挂)AI教程(一贯是有挂)(1)两分钟插件!闲玩暗宝辅助插件(辅助...