import React, { useState, useEffect } from "react"; import { View, Text, TextInput, Pressable, StyleSheet } from "react-native"; import { WhisperFile } from "@/app/lib/whisper"; import { Settings } from "@/app/lib/settings"; import { Picker } from "@react-native-picker/picker"; import { LanguageServer, language_matrix, language_matrix_entry } from "@/app/i18n/api"; const WHISPER_MODELS = { small: new WhisperFile("small"), medium: new WhisperFile("medium"), large: new WhisperFile("large"), }; const LIBRETRANSLATE_BASE_URL = "https://translate.argosopentech.com/translate"; const SettingsComponent = () => { const [hostLanguage, setHostLanguage] = useState(null); const [libretranslateBaseUrl, setLibretranslateBaseUrl] = useState< string | null >(null); const [languageOptions, setLanguageOptions] = useState(); const [langServerConn, setLangServerConn] = useState<{ success: boolean; error?: string; } | null>(null); const [whisperModel, setWhisperModel] = useState("small"); const [downloader, setDownloader] = useState(null); const [whisperDownloadProgress, setWhisperDownloadProgress] = useState< any | null >(null); useEffect(() => { loadSettings(); }, []); useEffect(() => { checkDownloadStatus(whisperModel); }, [whisperModel]); const getLanguageOptions = async () => { const languageServer = await LanguageServer.getDefault(); setLanguageOptions(await languageServer.fetchLanguages()); } const loadSettings = async () => { const settings = await Settings.getDefault(); const hostLanguage = await settings.getHostLanguage(); setHostLanguage(hostLanguage); const libretranslateBaseUrl = await settings.getLibretranslateBaseUrl(); setLibretranslateBaseUrl(libretranslateBaseUrl); const whisperModel = await settings.getWhisperModel(); setWhisperModel(whisperModel as keyof typeof WHISPER_MODELS); }; const handleHostLanguageChange = async (lang: string) => { const settings = await Settings.getDefault(); setHostLanguage(lang); await settings.setHostLanguage(lang); }; const handleLibretranslateBaseUrlChange = async (url: string) => { const settings = await Settings.getDefault(); setLibretranslateBaseUrl(url); await settings.setLibretranslateBaseUrl(url); checkLangServerConnection(url); }; const checkLangServerConnection = async (baseUrl: string) => { try { // Replace with actual connection check logic setLangServerConn({ success: true }); } catch (error) { setLangServerConn({ success: false, error: `${error}` }); } }; const handleWhisperModelChange = async ( model: keyof typeof WHISPER_MODELS ) => { const settings = await Settings.getDefault(); setWhisperModel(model); await settings.setWhisperModel(model); checkDownloadStatus(model); }; const doDownload = async () => { const whisperFile = WHISPER_MODELS[whisperModel]; const resumable = await whisperFile.createDownloadResumable({ onData: (progress) => setWhisperDownloadProgress(progress), }); setDownloader(resumable); try { await resumable.downloadAsync(); checkDownloadStatus(whisperModel); } catch (error) { console.error("Failed to download whisper model:", error); } }; const doStopDownload = async () => { downloader.cancelAsync(); setDownloader(null); }; const doDelete = async () => { const whisperFile = WHISPER_MODELS[whisperModel]; whisperFile.delete(); checkDownloadStatus(whisperModel); }; const checkDownloadStatus = async (model: keyof typeof WHISPER_MODELS) => { const whisperFile = WHISPER_MODELS[model]; const status = await whisperFile.getDownloadStatus(); if ( !status.isDownloadComplete && (!status.doesTargetExist || !status.hasDownloadStarted) ) { setDownloader(null); } }; return hostLanguage && libretranslateBaseUrl ? ( Host Language: { languag } LibreTranslate Base URL: {langServerConn && (langServerConn.success ? ( Success connecting to {libretranslateBaseUrl} ) : ( Error connecting to {libretranslateBaseUrl}: {langServerConn.error} ))} {Object.entries(WHISPER_MODELS).map(([key, whisperFile]) => ( ))} {downloader && whisperDownloadProgress && ( {whisperDownloadProgress.totalBytesWritten} bytes of{" "} {whisperDownloadProgress.totalBytesExpectedToWrite} bytes ( {Math.round( (whisperDownloadProgress.totalBytesWritten / whisperDownloadProgress.totalBytesExpectedToWrite) * 100 )} %) )} {downloader && whisperDownloadProgress && whisperDownloadProgress.totalBytesWritten !== whisperDownloadProgress.totalBytesExpectedToWrite ? ( Pause Download ) : ( DOWNLOAD )} {whisperModel && WHISPER_MODELS[whisperModel] && WHISPER_MODELS[whisperModel].doesTargetExist && ( Delete )} ) : ( Loading ... ); }; // Create styles for the component const styles = StyleSheet.create({ downloadButtonWrapper: { flexDirection: "row", }, downloadButton: { backgroundColor: "darkblue", padding: 20, margin: 10, flex: 3, flexDirection: "column", }, deleteButton: { backgroundColor: "darkred", flex: 1, flexDirection: "column", padding: 10, margin: 10, height: 50, }, pauseDownloadButton: { backgroundColor: "#444444", padding: 10, margin: 10, height: 50, }, buttonText: { color: "#fff", flex: 1, fontSize: 16, alignSelf: "center", textAlign: "center", textAlignVertical: "top", }, container: { flex: 1, padding: 20, }, label: { fontSize: 16, marginBottom: 8, }, input: { height: 40, borderColor: "gray", borderWidth: 1, marginBottom: 20, paddingHorizontal: 8, }, }); export default SettingsComponent;