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 [whisperFileExists, setWhisperFileExists] = useState(false); const [isWhisperHashValid, setIsWhisperHashValid] = useState(false); 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) { await whisperFile.syncHfMetadata(); await whisperFile.updateTargetExistence(); await whisperFile.updateTargetHash(); setWhisperFileExists(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); setWhisperFileExists(wFile.does_target_exist); }; const doSetDownloadStatus = (arg0: WhisperFile) => { console.log("Downloading ....") setIsWhisperHashValid(arg0.isHashValid); setBytesDone(arg0.download_data?.totalBytesWritten); setBytesRemaining(arg0.download_data?.totalBytesExpectedToWrite); }; const doOnComplete = (arg0: WhisperFile) => { setWhisperFile(arg0); } 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(); setStatusTimeout(undefined); }; 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]) => ( ))} {((!downloader) && whisperFile) && (whisperFile.does_target_exist && ( DELETE {whisperModel.toUpperCase()} )) } DOWNLOAD {whisperModel.toUpperCase()} )) { downloader && ( STOP DOWNLOAD ) } {bytesDone && bytesRemaining && whisperFile?.does_part_target_exist && ( {whisperFile && ( Downloading to {whisperFile.targetPath} )} {bytesDone} of{" "} {bytesRemaining} ( {bytesDone / bytesRemaining * 100} %){" "} )} ) : ( 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;