import React, { useState, useEffect } from "react"; import { View, Text, TextInput, Pressable, StyleSheet } from "react-native"; import { WHISPER_FILES, WhisperFile, download_status_t, whisper_tag_t, } 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< language_matrix | undefined >(); const [langServerConn, setLangServerConn] = useState<{ success: boolean; error?: string; } | null>(null); const [whisperModel, setWhisperModel] = useState("small"); const [whisperFile, setWhisperFile] = useState(); const [downloader, setDownloader] = useState(null); const [bytesDone, setBytesDone] = useState(); const [bytesRemaining, setBytesRemaining] = useState(); const [statusTimeout, setStatusTimeout] = useState< NodeJS.Timeout | undefined >(); useEffect(() => { (async function () { const settings = await Settings.getDefault(); setHostLanguage((await settings.getHostLanguage()) || "en"); setLibretranslateBaseUrl( (await settings.getLibretranslateBaseUrl()) || LIBRETRANSLATE_BASE_URL ); setWhisperModel((await settings.getWhisperModel()) || "small"); setWhisperFile(WHISPER_FILES[whisperModel]); if (!whisperFile) { throw new Error("Invalid Whisper file!"); } await whisperFile.syncHfMetadata(); await whisperFile.updateTargetExistence(); await whisperFile.updateTargetHash(); console.log("Does %s exist? part=%s, target=%s", whisperFile.label, whisperFile.does_part_target_exist, whisperFile.does_target_exist) })(); }, [whisperFile]); const getLanguageOptions = async () => { const languageServer = await LanguageServer.getDefault(); setLanguageOptions(await languageServer.fetchLanguages()); }; 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: whisper_tag_t) => { const settings = await Settings.getDefault(); await settings.setWhisperModel(model); setWhisperModel(model); const wFile = WHISPER_FILES[whisperModel]; await wFile.syncHfMetadata(); await wFile.updateTargetExistence(); // await wFile.updateTargetHash(); // setIsWhisperHashValid(wFile.isHashValid); setWhisperFile(wFile); }; const doSetDownloadStatus = (arg0: WhisperFile) => { // console.log("Downloading ...."); setBytesDone(arg0.download_data?.totalBytesWritten); setBytesRemaining(arg0.download_data?.totalBytesExpectedToWrite); }; const doOnComplete = async (arg0: WhisperFile) => { console.log("✅ Download complete."); setDownloader(undefined); await arg0.updateTargetExistence(); setWhisperFile(arg0); await whisperFile?.updateTargetExistence(); }; const doDownload = async () => { if (!whisperModel) { throw new Error("Could not start download because whisperModel not set."); } console.log("Starting download of %s", whisperModel); if (!whisperFile) throw new Error("No whisper file"); try { const resumable = await whisperFile.createDownloadResumable({ onData: doSetDownloadStatus, onComplete: doOnComplete, }); setDownloader(resumable); if (!resumable) throw new Error("Could not construct resumable"); await resumable.resumeAsync(); } 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(); await whisperFile.updateTargetExistence(); }; return hostLanguage && libretranslateBaseUrl ? ( Host Language: { {languageOptions && Object.entries(languageOptions).map(([key, value]) => { return ; })} } LibreTranslate Base URL: {langServerConn && (langServerConn.success ? ( Success connecting to {libretranslateBaseUrl} ) : ( Error connecting to {libretranslateBaseUrl}: {langServerConn.error} ))} {Object.entries(WHISPER_MODELS).map(([key, whisperFile]) => ( ))} {/* The target is completely downloaded */} {!downloader && whisperFile?.does_target_exist && ( DELETE {whisperModel.toUpperCase()} )} {/* The target "part" is present and is downloading */} {downloader && whisperFile?.does_part_target_exist && ( STOP DOWNLOAD )} {/* The target "part" is present and we are NOT downloading */} {!downloader && whisperFile?.does_part_target_exist && ( <> RESUME DOWNLOAD DELETE {whisperModel.toUpperCase()} )} {/* Anything else -- usually if the file has not yet been downloaded */} {!( downloader || whisperFile?.does_target_exist || whisperFile?.does_part_target_exist ) && ( DOWNLOAD {whisperModel.toUpperCase()} )} {downloader && bytesDone && bytesRemaining && whisperFile?.does_part_target_exist && ( {whisperFile && ( Downloading to {whisperFile.targetPath} )} {bytesDone} of {bytesRemaining} ( {(bytesDone / bytesRemaining) * 100} %){" "} )} Debug Panel downloader is null? {downloader ? "no" : "yes"} whisperFile.does_target_exist{" "} {whisperFile?.does_target_exist ? "yes" : "no"} whisperFile.does_part_target_exist{" "} {whisperFile?.does_part_target_exist ? "yes" : "no"} ) : ( Loading ... ); }; // Create styles for the component const styles = StyleSheet.create({ downloadButtonWrapper: { flexDirection: "row", }, downloadButton: { backgroundColor: "#236b9f", padding: 20, margin: 10, flex: 1, flexDirection: "column", alignItems: "center", }, 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;