pull/256/head
Ty Fiero 9 months ago
parent cd22f22300
commit 4647b24ccc

@ -14,6 +14,7 @@
"expo-av": "~13.10.5", "expo-av": "~13.10.5",
"expo-barcode-scanner": "~12.9.3", "expo-barcode-scanner": "~12.9.3",
"expo-camera": "~14.0.5", "expo-camera": "~14.0.5",
"expo-haptics": "~12.8.1",
"expo-status-bar": "~1.11.1", "expo-status-bar": "~1.11.1",
"react": "18.2.0", "react": "18.2.0",
"react-native": "0.73.4", "react-native": "0.73.4",
@ -7719,6 +7720,14 @@
"expo": "*" "expo": "*"
} }
}, },
"node_modules/expo-haptics": {
"version": "12.8.1",
"resolved": "https://registry.npmjs.org/expo-haptics/-/expo-haptics-12.8.1.tgz",
"integrity": "sha512-ntLsHkfle8K8w9MW8pZEw92ZN3sguaGUSSIxv30fPKNeQFu7Cq/h47Qv3tONv2MO3wU48N9FbKnant6XlfptpA==",
"peerDependencies": {
"expo": "*"
}
},
"node_modules/expo-image-loader": { "node_modules/expo-image-loader": {
"version": "4.6.0", "version": "4.6.0",
"resolved": "https://registry.npmjs.org/expo-image-loader/-/expo-image-loader-4.6.0.tgz", "resolved": "https://registry.npmjs.org/expo-image-loader/-/expo-image-loader-4.6.0.tgz",

@ -24,7 +24,8 @@
"react-native-safe-area-context": "4.8.2", "react-native-safe-area-context": "4.8.2",
"react-native-screens": "~3.29.0", "react-native-screens": "~3.29.0",
"text-encoding": "^0.7.0", "text-encoding": "^0.7.0",
"zustand": "^4.5.2" "zustand": "^4.5.2",
"expo-haptics": "~12.8.1"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "^7.20.0", "@babel/core": "^7.20.0",

@ -7,7 +7,7 @@ const HomeScreen = () => {
return ( return (
<View style={styles.container}> <View style={styles.container}>
<View style={styles.circle} /> {/* <View style={styles.circle} /> */}
<TouchableOpacity <TouchableOpacity
style={styles.button} style={styles.button}
onPress={() => navigation.navigate("Camera")} onPress={() => navigation.navigate("Camera")}

@ -1,9 +1,12 @@
import React, { useState, useEffect, useCallback } from "react"; import React, { useState, useEffect, useCallback, useRef } from "react";
import { View, Text, TouchableOpacity, StyleSheet } from "react-native"; 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'; import { create } from "zustand";
import useStore from "../lib/state";
import { Animated } from "react-native";
import * as Haptics from "expo-haptics";
interface MainProps { interface MainProps {
route: { route: {
@ -20,7 +23,8 @@ interface AudioQueueState {
const useAudioQueueStore = create<AudioQueueState>((set) => ({ const useAudioQueueStore = create<AudioQueueState>((set) => ({
audioQueue: [], // initial state audioQueue: [], // initial state
addToQueue: (uri) => set((state) => ({ audioQueue: [...state.audioQueue, uri] })), // action to set audio queue addToQueue: (uri) =>
set((state) => ({ audioQueue: [...state.audioQueue, uri] })), // action to set audio queue
})); }));
interface SoundState { interface SoundState {
@ -35,85 +39,105 @@ const useSoundStore = create<SoundState>((set) => ({
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 [isPressed, setIsPressed] = useState(false);
const [recording, setRecording] = useState<Audio.Recording | null>(null); const [recording, setRecording] = useState<Audio.Recording | null>(null);
const addToQueue = useAudioQueueStore((state) => state.addToQueue); const addToQueue = useAudioQueueStore((state) => state.addToQueue);
const audioQueue = useAudioQueueStore((state) => state.audioQueue); const audioQueue = useAudioQueueStore((state) => state.audioQueue);
const setSound = useSoundStore((state) => state.setSound); const setSound = useSoundStore((state) => state.setSound);
const sound = useSoundStore((state) => state.sound); const sound = useSoundStore((state) => state.sound);
const audioDir = FileSystem.documentDirectory + '01/audio/'; const [soundUriMap, setSoundUriMap] = useState<Map<Audio.Sound, string>>(
new Map()
);
const audioDir = FileSystem.documentDirectory + "01/audio/";
const [permissionResponse, requestPermission] = Audio.usePermissions(); const [permissionResponse, requestPermission] = Audio.usePermissions();
polyfillEncoding(); polyfillEncoding();
const backgroundColorAnim = useRef(new Animated.Value(0)).current;
const constructTempFilePath = async (buffer: string) => { const buttonBackgroundColorAnim = useRef(new Animated.Value(0)).current;
const backgroundColor = backgroundColorAnim.interpolate({
inputRange: [0, 1],
outputRange: ["black", "white"], // Change as needed
});
const buttonBackgroundColor = backgroundColorAnim.interpolate({
inputRange: [0, 1],
outputRange: ["white", "black"], // Inverse of the container
});
const constructTempFilePath = async (buffer: string) => {
try {
await dirExists(); await dirExists();
if (!buffer) {
console.log("Buffer is undefined or empty.");
return null;
}
const tempFilePath = `${audioDir}${Date.now()}.wav`; const tempFilePath = `${audioDir}${Date.now()}.wav`;
await FileSystem.writeAsStringAsync( await FileSystem.writeAsStringAsync(tempFilePath, buffer, {
tempFilePath, encoding: FileSystem.EncodingType.Base64,
buffer, });
{
encoding: FileSystem.EncodingType.Base64,
}
);
return tempFilePath; return tempFilePath;
}; } catch (error) {
console.log("Failed to construct temp file path:", error);
return null; // Return null to prevent crashing, error is logged
}
};
async function dirExists() { async function dirExists() {
/** /**
* Checks if audio directory exists in device storage, if not creates it. * Checks if audio directory exists in device storage, if not creates it.
*/ */
const dirInfo = await FileSystem.getInfoAsync(audioDir); try {
if (!dirInfo.exists) { const dirInfo = await FileSystem.getInfoAsync(audioDir);
console.log("audio directory doesn't exist, creating..."); if (!dirInfo.exists) {
await FileSystem.makeDirectoryAsync(audioDir, { intermediates: true }); console.error("audio directory doesn't exist, creating...");
await FileSystem.makeDirectoryAsync(audioDir, { intermediates: true });
}
} catch (error) {
console.error("Error checking or creating directory:", error);
} }
} }
const playNextAudio = async () => { const playNextAudio = useCallback(async () => {
console.log(`in playNextAudio audioQueue is ${audioQueue.length} and sound is ${sound}`); console.log(
`in playNextAudio audioQueue is ${audioQueue.length} and sound is ${sound}`
);
if (audioQueue.length > 0 && sound == null) { 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);
try { try {
const { sound } = await Audio.Sound.createAsync({ uri }); const { sound: newSound } = await Audio.Sound.createAsync({ uri });
setSound(sound); setSound(newSound);
setSoundUriMap(new Map(soundUriMap.set(newSound, uri)));
console.log("playing audio from", uri); await newSound.playAsync();
await sound?.playAsync(); newSound.setOnPlaybackStatusUpdate(_onPlayBackStatusUpdate);
} catch (error) {
sound.setOnPlaybackStatusUpdate(_onPlayBackStatusUpdate);
} catch (error){
console.log("Error playing audio", error); console.log("Error playing audio", error);
playNextAudio(); playNextAudio();
} }
} else { } else {
console.log("audioQueue is empty or sound is not null"); console.log("audioQueue is empty or sound is not null");
return; return;
} }
}; }, [audioQueue, sound, soundUriMap]);
const _onPlayBackStatusUpdate = async (status: AVPlaybackStatus) => { const _onPlayBackStatusUpdate = useCallback(
if (isAVPlaybackStatusSuccess(status) && status.didJustFinish === true){ async (status) => {
console.log("on playback status update sound is ", sound); if (status.didJustFinish) {
if (sound != null){ await sound?.unloadAsync();
console.log('Unloading Sound'); soundUriMap.delete(sound);
await sound.unloadAsync(); setSoundUriMap(new Map(soundUriMap));
setSound(null);
playNextAudio();
} }
setSound(null); },
console.log("audio has finished playing, playing next audio"); [sound, soundUriMap, playNextAudio]
console.log(audioQueue); );
playNextAudio();
}
}
const isAVPlaybackStatusSuccess = ( const isAVPlaybackStatusSuccess = (
status: AVPlaybackStatus status: AVPlaybackStatus
@ -121,13 +145,17 @@ const Main: React.FC<MainProps> = ({ route }) => {
return (status as AVPlaybackStatusSuccess).isLoaded !== undefined; return (status as AVPlaybackStatusSuccess).isLoaded !== undefined;
}; };
// useEffect(() => {
// console.log("audioQueue has been updated:", audioQueue.length);
// if (audioQueue.length == 1) {
// playNextAudio();
// }
// }, [audioQueue]);
useEffect(() => { useEffect(() => {
console.log("audioQueue has been updated:", audioQueue.length); if (audioQueue.length > 0 && !sound) {
if (audioQueue.length == 1) {
playNextAudio(); playNextAudio();
} }
}, [audioQueue]); }, [audioQueue, sound, playNextAudio]);
useEffect(() => { useEffect(() => {
console.log("sound has been updated:", sound); console.log("sound has been updated:", sound);
}, [sound]); }, [sound]);
@ -145,14 +173,34 @@ const Main: React.FC<MainProps> = ({ route }) => {
}; };
websocket.onmessage = async (e) => { websocket.onmessage = async (e) => {
try {
const message = JSON.parse(e.data); const message = JSON.parse(e.data);
console.log(message.content.slice(0, 50));
if (message.content && typeof message.content === "string") {
const buffer = await message.content as string; console.log("✅✅✅✅✅✅✅✅✅✅✅✅✅✅✅✅✅✅✅ Audio message");
const filePath = await constructTempFilePath(buffer);
addToQueue(filePath); const buffer = message.content;
console.log("audio file written to", filePath); console.log(buffer.length);
if (buffer && buffer.length > 0) {
const filePath = await constructTempFilePath(buffer);
if (filePath !== null) {
addToQueue(filePath);
console.log("audio file written to", filePath);
} else {
console.error("Failed to create file path");
}
} else {
console.error("Received message is empty or undefined");
}
} else {
// console.log(typeof message);
// console.log(typeof message.content);
console.log("Received message content is not a string.");
console.log(message);
}
} catch (error) {
console.error("Error handling WebSocket message:", error);
}
}; };
websocket.onerror = (error) => { websocket.onerror = (error) => {
@ -185,7 +233,10 @@ const Main: React.FC<MainProps> = ({ route }) => {
} }
try { try {
if (permissionResponse !== null && permissionResponse.status !== `granted`) { if (
permissionResponse !== null &&
permissionResponse.status !== `granted`
) {
console.log("Requesting permission.."); console.log("Requesting permission..");
await requestPermission(); await requestPermission();
} }
@ -197,7 +248,9 @@ const Main: React.FC<MainProps> = ({ route }) => {
console.log("Starting recording.."); console.log("Starting recording..");
const newRecording = new Audio.Recording(); const newRecording = new Audio.Recording();
await newRecording.prepareToRecordAsync(Audio.RecordingOptionsPresets.HIGH_QUALITY); await newRecording.prepareToRecordAsync(
Audio.RecordingOptionsPresets.HIGH_QUALITY
);
await newRecording.startAsync(); await newRecording.startAsync();
setRecording(newRecording); setRecording(newRecording);
@ -228,10 +281,9 @@ const Main: React.FC<MainProps> = ({ route }) => {
} }
*/ */
if (ws && uri) { if (ws && uri) {
const response = await fetch(uri); const response = await fetch(uri);
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(); const reader = new FileReader();
@ -242,16 +294,30 @@ const Main: React.FC<MainProps> = ({ route }) => {
ws.send(audioBytes); ws.send(audioBytes);
const audioArray = new Uint8Array(audioBytes as ArrayBuffer); const audioArray = new Uint8Array(audioBytes as ArrayBuffer);
const decoder = new TextDecoder("utf-8"); const decoder = new TextDecoder("utf-8");
console.log("sent audio bytes to WebSocket", decoder.decode(audioArray).slice(0, 50)); console.log(
"sent audio bytes to WebSocket",
decoder.decode(audioArray).slice(0, 50)
);
} }
}; };
} }
} }
}, [recording]); }, [recording]);
const toggleRecording = (shouldPress: boolean) => {
Animated.timing(backgroundColorAnim, {
toValue: shouldPress ? 1 : 0,
duration: 400,
useNativeDriver: false, // 'backgroundColor' does not support native driver
}).start();
Animated.timing(buttonBackgroundColorAnim, {
toValue: shouldPress ? 1 : 0,
duration: 400,
useNativeDriver: false, // 'backgroundColor' does not support native driver
}).start();
};
return ( return (
<View style={styles.container}> <Animated.View style={[styles.container, { backgroundColor }]}>
<Text <Text
style={[ style={[
styles.statusText, styles.statusText,
@ -262,30 +328,47 @@ const Main: React.FC<MainProps> = ({ route }) => {
</Text> </Text>
<TouchableOpacity <TouchableOpacity
style={styles.button} style={styles.button}
onPressIn={startRecording} onPressIn={() => {
onPressOut={stopRecording} setIsPressed(true);
toggleRecording(true); // Pass true when pressed
startRecording();
Haptics.impactAsync(Haptics.ImpactFeedbackStyle.Heavy);
}}
onPressOut={() => {
setIsPressed(false);
toggleRecording(false); // Pass false when released
stopRecording();
Haptics.impactAsync(Haptics.ImpactFeedbackStyle.Heavy);
}}
> >
<View style={styles.circle}> <Animated.View
<Text style={styles.buttonText}>Record</Text> style={[styles.circle, { backgroundColor: buttonBackgroundColor }]}
</View> >
{/* <Text
style={
recording ? styles.buttonTextRecording : styles.buttonTextDefault
}
>
Record
</Text> */}
</Animated.View>
</TouchableOpacity> </TouchableOpacity>
</View> </Animated.View>
); );
} };
const styles = StyleSheet.create({ const styles = StyleSheet.create({
container: { container: {
flex: 1, flex: 1,
justifyContent: 'center', justifyContent: "center",
alignItems: "center", alignItems: "center",
backgroundColor: '#ecf0f1',
padding: 10, padding: 10,
position: "relative",
}, },
circle: { circle: {
width: 100, width: 100,
height: 100, height: 100,
borderRadius: 50, borderRadius: 50,
backgroundColor: "black",
justifyContent: "center", justifyContent: "center",
alignItems: "center", alignItems: "center",
}, },
@ -296,13 +379,20 @@ const styles = StyleSheet.create({
justifyContent: "center", justifyContent: "center",
alignItems: "center", alignItems: "center",
}, },
buttonText: { buttonTextDefault: {
color: "black",
fontSize: 16,
},
buttonTextRecording: {
color: "white", color: "white",
fontSize: 16, fontSize: 16,
}, },
statusText: { statusText: {
marginBottom: 20, position: "absolute",
fontSize: 16, bottom: 10,
alignSelf: "center",
fontSize: 12,
fontWeight: "bold",
}, },
}); });

Loading…
Cancel
Save