|
|
@ -3,6 +3,7 @@ import { View, Text, TouchableOpacity, StyleSheet } from "react-native";
|
|
|
|
import * as FileSystem from 'expo-file-system';
|
|
|
|
import * as FileSystem from 'expo-file-system';
|
|
|
|
import { AVPlaybackStatus, AVPlaybackStatusSuccess, Audio } from "expo-av";
|
|
|
|
import { AVPlaybackStatus, AVPlaybackStatusSuccess, Audio } from "expo-av";
|
|
|
|
import { polyfill as polyfillEncoding } from 'react-native-polyfill-globals/src/encoding';
|
|
|
|
import { polyfill as polyfillEncoding } from 'react-native-polyfill-globals/src/encoding';
|
|
|
|
|
|
|
|
import { create } from 'zustand';
|
|
|
|
|
|
|
|
|
|
|
|
interface MainProps {
|
|
|
|
interface MainProps {
|
|
|
|
route: {
|
|
|
|
route: {
|
|
|
@ -12,23 +13,43 @@ interface MainProps {
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
interface AudioQueueState {
|
|
|
|
|
|
|
|
audioQueue: string[]; // Define the audio queue type
|
|
|
|
|
|
|
|
addToQueue: (uri: string) => void; // Function to set audio queue
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const useAudioQueueStore = create<AudioQueueState>((set) => ({
|
|
|
|
|
|
|
|
audioQueue: [], // initial state
|
|
|
|
|
|
|
|
addToQueue: (uri) => set((state) => ({ audioQueue: [...state.audioQueue, uri] })), // action to set audio queue
|
|
|
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
interface SoundState {
|
|
|
|
|
|
|
|
sound: Audio.Sound | null; // Define the sound type
|
|
|
|
|
|
|
|
setSound: (newSound: Audio.Sound | null) => void; // Function to set sound
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const useSoundStore = create<SoundState>((set) => ({
|
|
|
|
|
|
|
|
sound: null, // initial state
|
|
|
|
|
|
|
|
setSound: (newSound) => set({ sound: newSound }), // action to set sound
|
|
|
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
|
|
const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
const { scannedData } = route.params;
|
|
|
|
const { scannedData } = route.params;
|
|
|
|
const [connectionStatus, setConnectionStatus] = useState<string>("Connecting...");
|
|
|
|
const [connectionStatus, setConnectionStatus] = useState<string>("Connecting...");
|
|
|
|
const [ws, setWs] = useState<WebSocket | null>(null);
|
|
|
|
const [ws, setWs] = useState<WebSocket | null>(null);
|
|
|
|
const [recording, setRecording] = useState<Audio.Recording | null>(null);
|
|
|
|
const [recording, setRecording] = useState<Audio.Recording | null>(null);
|
|
|
|
const [audioQueue, setAudioQueue] = useState<string[]>([]);
|
|
|
|
const addToQueue = useAudioQueueStore((state) => state.addToQueue);
|
|
|
|
const [sound, setSound] = useState<Audio.Sound | null>();
|
|
|
|
const audioQueue = useAudioQueueStore((state) => state.audioQueue);
|
|
|
|
|
|
|
|
const setSound = useSoundStore((state) => state.setSound);
|
|
|
|
|
|
|
|
const sound = useSoundStore((state) => state.sound);
|
|
|
|
const audioDir = FileSystem.documentDirectory + '01/audio/';
|
|
|
|
const audioDir = FileSystem.documentDirectory + '01/audio/';
|
|
|
|
const [permissionResponse, requestPermission] = Audio.usePermissions();
|
|
|
|
const [permissionResponse, requestPermission] = Audio.usePermissions();
|
|
|
|
polyfillEncoding();
|
|
|
|
polyfillEncoding();
|
|
|
|
const reader = new FileReader();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const constructTempFilePath = async (buffer: string) => {
|
|
|
|
const constructTempFilePath = async (buffer: string) => {
|
|
|
|
await dirExists();
|
|
|
|
await dirExists();
|
|
|
|
const tempFilePath = `${audioDir}${Date.now()}.wav`;
|
|
|
|
const tempFilePath = `${audioDir}${Date.now()}.wav`;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
await FileSystem.writeAsStringAsync(
|
|
|
|
await FileSystem.writeAsStringAsync(
|
|
|
|
tempFilePath,
|
|
|
|
tempFilePath,
|
|
|
|
buffer,
|
|
|
|
buffer,
|
|
|
@ -37,7 +58,6 @@ const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return tempFilePath;
|
|
|
|
return tempFilePath;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
@ -54,15 +74,10 @@ const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const playNextAudio = async () => {
|
|
|
|
const playNextAudio = async () => {
|
|
|
|
console.log("in playNextAudio audioQueue is", audioQueue.length);
|
|
|
|
console.log(`in playNextAudio audioQueue is ${audioQueue.length} and sound is ${sound}`);
|
|
|
|
|
|
|
|
|
|
|
|
if (sound != null){
|
|
|
|
|
|
|
|
console.log('Unloading Sound');
|
|
|
|
|
|
|
|
await sound.unloadAsync();
|
|
|
|
|
|
|
|
setSound(null);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (audioQueue.length > 0) {
|
|
|
|
if (audioQueue.length > 0 && sound == null) {
|
|
|
|
const uri = audioQueue.shift() as string;
|
|
|
|
const uri = audioQueue.shift() as string;
|
|
|
|
console.log("load audio from", uri);
|
|
|
|
console.log("load audio from", uri);
|
|
|
|
|
|
|
|
|
|
|
@ -80,21 +95,32 @@ const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
playNextAudio();
|
|
|
|
playNextAudio();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
console.log("audioQueue is empty or sound is not null");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const _onPlayBackStatusUpdate = async (status: AVPlaybackStatus) => {
|
|
|
|
|
|
|
|
if (isAVPlaybackStatusSuccess(status) && status.didJustFinish === true){
|
|
|
|
|
|
|
|
console.log("on playback status update sound is ", sound);
|
|
|
|
|
|
|
|
if (sound != null){
|
|
|
|
|
|
|
|
console.log('Unloading Sound');
|
|
|
|
|
|
|
|
await sound.unloadAsync();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
setSound(null);
|
|
|
|
|
|
|
|
console.log("audio has finished playing, playing next audio");
|
|
|
|
|
|
|
|
console.log(audioQueue);
|
|
|
|
|
|
|
|
playNextAudio();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const isAVPlaybackStatusSuccess = (
|
|
|
|
const isAVPlaybackStatusSuccess = (
|
|
|
|
status: AVPlaybackStatus
|
|
|
|
status: AVPlaybackStatus
|
|
|
|
): status is AVPlaybackStatusSuccess => {
|
|
|
|
): status is AVPlaybackStatusSuccess => {
|
|
|
|
return (status as AVPlaybackStatusSuccess).isLoaded !== undefined;
|
|
|
|
return (status as AVPlaybackStatusSuccess).isLoaded !== undefined;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const _onPlayBackStatusUpdate = (status: AVPlaybackStatus) => {
|
|
|
|
|
|
|
|
if (isAVPlaybackStatusSuccess(status) && status.didJustFinish){
|
|
|
|
|
|
|
|
playNextAudio();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
useEffect(() => {
|
|
|
|
console.log("audioQueue has been updated:", audioQueue.length);
|
|
|
|
console.log("audioQueue has been updated:", audioQueue.length);
|
|
|
|
if (audioQueue.length == 1) {
|
|
|
|
if (audioQueue.length == 1) {
|
|
|
@ -102,6 +128,10 @@ const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, [audioQueue]);
|
|
|
|
}, [audioQueue]);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
|
|
console.log("sound has been updated:", sound);
|
|
|
|
|
|
|
|
}, [sound]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
useEffect(() => {
|
|
|
|
let websocket: WebSocket;
|
|
|
|
let websocket: WebSocket;
|
|
|
|
try {
|
|
|
|
try {
|
|
|
@ -121,13 +151,8 @@ const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
|
|
|
|
|
|
|
|
const buffer = await message.content as string;
|
|
|
|
const buffer = await message.content as string;
|
|
|
|
const filePath = await constructTempFilePath(buffer);
|
|
|
|
const filePath = await constructTempFilePath(buffer);
|
|
|
|
setAudioQueue((prevQueue) => [...prevQueue, filePath]);
|
|
|
|
addToQueue(filePath);
|
|
|
|
console.log("audio file written to", filePath);
|
|
|
|
console.log("audio file written to", filePath);
|
|
|
|
|
|
|
|
|
|
|
|
if (message.format === "bytes.raw" && message.end && audioQueue.length >= 1) {
|
|
|
|
|
|
|
|
console.log("calling playNextAudio");
|
|
|
|
|
|
|
|
playNextAudio();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
websocket.onerror = (error) => {
|
|
|
|
websocket.onerror = (error) => {
|
|
|
@ -209,6 +234,7 @@ const Main: React.FC<MainProps> = ({ route }) => {
|
|
|
|
console.log("fetched audio file", response);
|
|
|
|
console.log("fetched audio file", response);
|
|
|
|
const blob = await response.blob();
|
|
|
|
const blob = await response.blob();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const reader = new FileReader();
|
|
|
|
reader.readAsArrayBuffer(blob);
|
|
|
|
reader.readAsArrayBuffer(blob);
|
|
|
|
reader.onloadend = () => {
|
|
|
|
reader.onloadend = () => {
|
|
|
|
const audioBytes = reader.result;
|
|
|
|
const audioBytes = reader.result;
|
|
|
|