parent
					
						
							
								e9f197740d
							
						
					
				
			
			
				commit
				
					
						6637ecb460
					
				
			
		
					 3 changed files with 117 additions and 71 deletions
				
			
		
							
								
								
									
										62
									
								
								app/javascript/mastodon/hooks/useAudioContext.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								app/javascript/mastodon/hooks/useAudioContext.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,62 @@ | |||
| import { useCallback, useEffect, useRef } from 'react'; | ||||
| 
 | ||||
| interface AudioContextOptions { | ||||
|   audioElementRef: React.MutableRefObject<HTMLAudioElement | null>; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Create and return an audio context instance for a given audio element [0]. | ||||
|  * Also returns an associated audio source, a gain node, and play and pause actions | ||||
|  * which should be used instead of `audioElementRef.current.play/pause()`. | ||||
|  * | ||||
|  * [0] https://developer.mozilla.org/en-US/docs/Web/API/AudioContext
 | ||||
|  */ | ||||
| 
 | ||||
| export const useAudioContext = ({ audioElementRef }: AudioContextOptions) => { | ||||
|   const audioContextRef = useRef<AudioContext>(); | ||||
|   const sourceRef = useRef<MediaElementAudioSourceNode>(); | ||||
|   const gainNodeRef = useRef<GainNode>(); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (!audioElementRef.current) { | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     const context = audioContextRef.current ?? new AudioContext(); | ||||
|     const source = | ||||
|       sourceRef.current ?? | ||||
|       context.createMediaElementSource(audioElementRef.current); | ||||
| 
 | ||||
|     const gainNode = context.createGain(); | ||||
|     gainNode.connect(context.destination); | ||||
|     source.connect(gainNode); | ||||
| 
 | ||||
|     audioContextRef.current = context; | ||||
|     gainNodeRef.current = gainNode; | ||||
|     sourceRef.current = source; | ||||
| 
 | ||||
|     return () => { | ||||
|       if (context.state !== 'closed') { | ||||
|         void context.close(); | ||||
|       } | ||||
|     }; | ||||
|   }, [audioElementRef]); | ||||
| 
 | ||||
|   const playAudio = useCallback(() => { | ||||
|     void audioElementRef.current?.play(); | ||||
|     void audioContextRef.current?.resume(); | ||||
|   }, [audioElementRef]); | ||||
| 
 | ||||
|   const pauseAudio = useCallback(() => { | ||||
|     audioElementRef.current?.pause(); | ||||
|     void audioContextRef.current?.suspend(); | ||||
|   }, [audioElementRef]); | ||||
| 
 | ||||
|   return { | ||||
|     audioContextRef, | ||||
|     sourceRef, | ||||
|     gainNodeRef, | ||||
|     playAudio, | ||||
|     pauseAudio, | ||||
|   }; | ||||
| }; | ||||
|  | @ -1,4 +1,4 @@ | |||
| import { useState, useEffect, useRef, useCallback } from 'react'; | ||||
| import { useState, useEffect, useRef } from 'react'; | ||||
| 
 | ||||
| const normalizeFrequencies = (arr: Float32Array): number[] => { | ||||
|   return new Array(...arr).map((value: number) => { | ||||
|  | @ -10,12 +10,17 @@ const normalizeFrequencies = (arr: Float32Array): number[] => { | |||
|   }); | ||||
| }; | ||||
| 
 | ||||
| export const useAudioVisualizer = ( | ||||
|   ref: React.MutableRefObject<HTMLAudioElement | null>, | ||||
|   numBands: number, | ||||
| ) => { | ||||
|   const audioContextRef = useRef<AudioContext>(); | ||||
|   const sourceRef = useRef<MediaElementAudioSourceNode>(); | ||||
| interface AudioVisualiserOptions { | ||||
|   audioContextRef: React.MutableRefObject<AudioContext | undefined>; | ||||
|   sourceRef: React.MutableRefObject<MediaElementAudioSourceNode | undefined>; | ||||
|   numBands: number; | ||||
| } | ||||
| 
 | ||||
| export const useAudioVisualizer = ({ | ||||
|   audioContextRef, | ||||
|   sourceRef, | ||||
|   numBands, | ||||
| }: AudioVisualiserOptions) => { | ||||
|   const analyzerRef = useRef<AnalyserNode>(); | ||||
| 
 | ||||
|   const [frequencyBands, setFrequencyBands] = useState<number[]>( | ||||
|  | @ -23,47 +28,31 @@ export const useAudioVisualizer = ( | |||
|   ); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (!audioContextRef.current) { | ||||
|       audioContextRef.current = new AudioContext(); | ||||
|     if (audioContextRef.current) { | ||||
|       analyzerRef.current = audioContextRef.current.createAnalyser(); | ||||
|       analyzerRef.current.smoothingTimeConstant = 0.6; | ||||
|       analyzerRef.current.fftSize = 2048; | ||||
|     } | ||||
| 
 | ||||
|     return () => { | ||||
|       if (audioContextRef.current) { | ||||
|         void audioContextRef.current.close(); | ||||
|       } | ||||
|     }; | ||||
|   }, []); | ||||
|   }, [audioContextRef]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if ( | ||||
|       audioContextRef.current && | ||||
|       analyzerRef.current && | ||||
|       !sourceRef.current && | ||||
|       ref.current | ||||
|     ) { | ||||
|       sourceRef.current = audioContextRef.current.createMediaElementSource( | ||||
|         ref.current, | ||||
|       ); | ||||
|     if (analyzerRef.current && sourceRef.current) { | ||||
|       sourceRef.current.connect(analyzerRef.current); | ||||
|       sourceRef.current.connect(audioContextRef.current.destination); | ||||
|     } | ||||
|     const currentSource = sourceRef.current; | ||||
| 
 | ||||
|     return () => { | ||||
|       if (sourceRef.current) { | ||||
|         sourceRef.current.disconnect(); | ||||
|       if (currentSource && analyzerRef.current) { | ||||
|         currentSource.disconnect(analyzerRef.current); | ||||
|       } | ||||
|     }; | ||||
|   }, [ref]); | ||||
|   }, [audioContextRef, sourceRef]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const source = sourceRef.current; | ||||
|     const analyzer = analyzerRef.current; | ||||
|     const context = audioContextRef.current; | ||||
| 
 | ||||
|     if (!source || !analyzer || !context) { | ||||
|     if (!analyzer || !context) { | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|  | @ -94,19 +83,7 @@ export const useAudioVisualizer = ( | |||
|     return () => { | ||||
|       clearInterval(updateInterval); | ||||
|     }; | ||||
|   }, [numBands]); | ||||
|   }, [numBands, audioContextRef]); | ||||
| 
 | ||||
|   const resume = useCallback(() => { | ||||
|     if (audioContextRef.current) { | ||||
|       void audioContextRef.current.resume(); | ||||
|     } | ||||
|   }, []); | ||||
| 
 | ||||
|   const suspend = useCallback(() => { | ||||
|     if (audioContextRef.current) { | ||||
|       void audioContextRef.current.suspend(); | ||||
|     } | ||||
|   }, []); | ||||
| 
 | ||||
|   return [resume, suspend, frequencyBands] as const; | ||||
|   return frequencyBands; | ||||
| }; | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue