473 lines
19 KiB
Swift
473 lines
19 KiB
Swift
import Foundation
|
|
import SwiftUI
|
|
import AVFoundation
|
|
import SwiftData
|
|
import AppKit
|
|
import KeyboardShortcuts
|
|
import os
|
|
|
|
@MainActor
|
|
class WhisperState: NSObject, ObservableObject, AVAudioRecorderDelegate {
|
|
@Published var isModelLoaded = false
|
|
@Published var canTranscribe = false
|
|
@Published var isRecording = false
|
|
@Published var loadedLocalModel: WhisperModel?
|
|
@Published var currentTranscriptionModel: (any TranscriptionModel)?
|
|
@Published var isModelLoading = false
|
|
@Published var availableModels: [WhisperModel] = []
|
|
@Published var allAvailableModels: [any TranscriptionModel] = PredefinedModels.models
|
|
@Published var clipboardMessage = ""
|
|
@Published var miniRecorderError: String?
|
|
@Published var isProcessing = false
|
|
@Published var shouldCancelRecording = false
|
|
@Published var isTranscribing = false
|
|
@Published var isAutoCopyEnabled: Bool = UserDefaults.standard.object(forKey: "IsAutoCopyEnabled") as? Bool ?? true {
|
|
didSet {
|
|
UserDefaults.standard.set(isAutoCopyEnabled, forKey: "IsAutoCopyEnabled")
|
|
}
|
|
}
|
|
@Published var recorderType: String = UserDefaults.standard.string(forKey: "RecorderType") ?? "mini" {
|
|
didSet {
|
|
UserDefaults.standard.set(recorderType, forKey: "RecorderType")
|
|
}
|
|
}
|
|
|
|
@Published var isVisualizerActive = false
|
|
|
|
@Published var isMiniRecorderVisible = false {
|
|
didSet {
|
|
if isMiniRecorderVisible {
|
|
showRecorderPanel()
|
|
} else {
|
|
hideRecorderPanel()
|
|
}
|
|
}
|
|
}
|
|
|
|
var whisperContext: WhisperContext?
|
|
let recorder = Recorder()
|
|
var recordedFile: URL? = nil
|
|
let whisperPrompt = WhisperPrompt()
|
|
|
|
// Prompt detection service for trigger word handling
|
|
private let promptDetectionService = PromptDetectionService()
|
|
|
|
let modelContext: ModelContext
|
|
|
|
// Transcription Services
|
|
private var localTranscriptionService: LocalTranscriptionService
|
|
private let cloudTranscriptionService = CloudTranscriptionService()
|
|
|
|
private var modelUrl: URL? {
|
|
let possibleURLs = [
|
|
Bundle.main.url(forResource: "ggml-base.en", withExtension: "bin", subdirectory: "Models"),
|
|
Bundle.main.url(forResource: "ggml-base.en", withExtension: "bin"),
|
|
Bundle.main.bundleURL.appendingPathComponent("Models/ggml-base.en.bin")
|
|
]
|
|
|
|
for url in possibleURLs {
|
|
if let url = url, FileManager.default.fileExists(atPath: url.path) {
|
|
return url
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
private enum LoadError: Error {
|
|
case couldNotLocateModel
|
|
}
|
|
|
|
let modelsDirectory: URL
|
|
let recordingsDirectory: URL
|
|
let enhancementService: AIEnhancementService?
|
|
var licenseViewModel: LicenseViewModel
|
|
let logger = Logger(subsystem: "com.prakashjoshipax.voiceink", category: "WhisperState")
|
|
var notchWindowManager: NotchWindowManager?
|
|
var miniWindowManager: MiniWindowManager?
|
|
|
|
// For model progress tracking
|
|
@Published var downloadProgress: [String: Double] = [:]
|
|
|
|
init(modelContext: ModelContext, enhancementService: AIEnhancementService? = nil) {
|
|
self.modelContext = modelContext
|
|
let appSupportDirectory = FileManager.default.urls(for: .applicationSupportDirectory, in: .userDomainMask)[0]
|
|
.appendingPathComponent("com.prakashjoshipax.VoiceInk")
|
|
|
|
self.modelsDirectory = appSupportDirectory.appendingPathComponent("WhisperModels")
|
|
self.recordingsDirectory = appSupportDirectory.appendingPathComponent("Recordings")
|
|
|
|
// Initialize services without whisperState reference first
|
|
self.localTranscriptionService = LocalTranscriptionService(modelsDirectory: self.modelsDirectory)
|
|
|
|
self.enhancementService = enhancementService
|
|
self.licenseViewModel = LicenseViewModel()
|
|
|
|
super.init()
|
|
|
|
// Set the whisperState reference after super.init()
|
|
self.localTranscriptionService = LocalTranscriptionService(modelsDirectory: self.modelsDirectory, whisperState: self)
|
|
|
|
setupNotifications()
|
|
createModelsDirectoryIfNeeded()
|
|
createRecordingsDirectoryIfNeeded()
|
|
loadAvailableModels()
|
|
loadCurrentTranscriptionModel()
|
|
}
|
|
|
|
private func createRecordingsDirectoryIfNeeded() {
|
|
do {
|
|
try FileManager.default.createDirectory(at: recordingsDirectory, withIntermediateDirectories: true, attributes: nil)
|
|
} catch {
|
|
logger.error("Error creating recordings directory: \(error.localizedDescription)")
|
|
}
|
|
}
|
|
|
|
func toggleRecord() async {
|
|
if isRecording {
|
|
logger.notice("🛑 Stopping recording")
|
|
await MainActor.run {
|
|
isRecording = false
|
|
isVisualizerActive = false
|
|
}
|
|
await recorder.stopRecording()
|
|
if let recordedFile {
|
|
if !shouldCancelRecording {
|
|
await transcribeAudio(recordedFile)
|
|
} else {
|
|
logger.info("🛑 Transcription and paste aborted in toggleRecord due to shouldCancelRecording flag.")
|
|
await MainActor.run {
|
|
isProcessing = false
|
|
isTranscribing = false
|
|
canTranscribe = true
|
|
}
|
|
await cleanupModelResources()
|
|
}
|
|
} else {
|
|
logger.error("❌ No recorded file found after stopping recording")
|
|
}
|
|
} else {
|
|
guard currentTranscriptionModel != nil else {
|
|
await MainActor.run {
|
|
NotificationManager.shared.showNotification(
|
|
title: "No AI Model Selected",
|
|
message: "Please select a default AI model before recording.",
|
|
type: .error
|
|
)
|
|
}
|
|
return
|
|
}
|
|
shouldCancelRecording = false
|
|
logger.notice("🎙️ Starting recording sequence...")
|
|
requestRecordPermission { [self] granted in
|
|
if granted {
|
|
Task {
|
|
do {
|
|
// --- Prepare temporary file URL within Application Support base directory ---
|
|
let baseAppSupportDirectory = self.recordingsDirectory.deletingLastPathComponent()
|
|
let file = baseAppSupportDirectory.appendingPathComponent("output.wav")
|
|
// Ensure the base directory exists
|
|
try? FileManager.default.createDirectory(at: baseAppSupportDirectory, withIntermediateDirectories: true)
|
|
// Clean up any old temporary file first
|
|
self.recordedFile = file
|
|
|
|
try await self.recorder.startRecording(toOutputFile: file)
|
|
self.logger.notice("✅ Audio engine started successfully.")
|
|
|
|
await MainActor.run {
|
|
self.isRecording = true
|
|
self.isVisualizerActive = true
|
|
}
|
|
|
|
await ActiveWindowService.shared.applyConfigurationForCurrentApp()
|
|
|
|
// Only load model if it's a local model and not already loaded
|
|
if let model = self.currentTranscriptionModel, model.provider == .local {
|
|
if let localWhisperModel = self.availableModels.first(where: { $0.name == model.name }),
|
|
self.whisperContext == nil {
|
|
do {
|
|
try await self.loadModel(localWhisperModel)
|
|
} catch {
|
|
self.logger.error("❌ Model loading failed: \(error.localizedDescription)")
|
|
}
|
|
}
|
|
}
|
|
|
|
if let enhancementService = self.enhancementService,
|
|
enhancementService.isEnhancementEnabled &&
|
|
enhancementService.useScreenCaptureContext {
|
|
await enhancementService.captureScreenContext()
|
|
}
|
|
|
|
} catch {
|
|
self.logger.error("❌ Failed to start recording: \(error.localizedDescription)")
|
|
await MainActor.run {
|
|
self.isRecording = false
|
|
self.isVisualizerActive = false
|
|
}
|
|
if let url = self.recordedFile {
|
|
try? FileManager.default.removeItem(at: url)
|
|
self.recordedFile = nil
|
|
self.logger.notice("🗑️ Cleaned up temporary recording file after failed start.")
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
logger.error("❌ Recording permission denied.")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private func requestRecordPermission(response: @escaping (Bool) -> Void) {
|
|
#if os(macOS)
|
|
response(true)
|
|
#else
|
|
AVAudioSession.sharedInstance().requestRecordPermission { granted in
|
|
response(granted)
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// MARK: AVAudioRecorderDelegate
|
|
|
|
nonisolated func audioRecorderEncodeErrorDidOccur(_ recorder: AVAudioRecorder, error: Error?) {
|
|
if let error {
|
|
Task {
|
|
await handleRecError(error)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func handleRecError(_ error: Error) {
|
|
logger.error("Recording error: \(error.localizedDescription)")
|
|
isRecording = false
|
|
}
|
|
|
|
nonisolated func audioRecorderDidFinishRecording(_ recorder: AVAudioRecorder, successfully flag: Bool) {
|
|
Task {
|
|
await onDidFinishRecording(success: flag)
|
|
}
|
|
}
|
|
|
|
private func onDidFinishRecording(success: Bool) {
|
|
if !success {
|
|
logger.error("Recording did not finish successfully")
|
|
}
|
|
}
|
|
|
|
private func transcribeAudio(_ url: URL) async {
|
|
if shouldCancelRecording {
|
|
logger.info("🎤 Transcription and paste aborted at the beginning of transcribeAudio due to shouldCancelRecording flag.")
|
|
await MainActor.run {
|
|
isProcessing = false
|
|
isTranscribing = false
|
|
canTranscribe = true
|
|
}
|
|
await cleanupModelResources()
|
|
return
|
|
}
|
|
|
|
await MainActor.run {
|
|
isProcessing = true
|
|
isTranscribing = true
|
|
canTranscribe = false
|
|
}
|
|
|
|
defer {
|
|
if shouldCancelRecording {
|
|
Task {
|
|
await cleanupModelResources()
|
|
}
|
|
}
|
|
}
|
|
|
|
guard let model = currentTranscriptionModel else {
|
|
logger.error("❌ Cannot transcribe: No model selected")
|
|
return
|
|
}
|
|
|
|
logger.notice("🔄 Starting transcription with model: \(model.displayName)")
|
|
|
|
do {
|
|
// --- Core Transcription Logic ---
|
|
let transcriptionService: TranscriptionService = (model.provider == .local) ? localTranscriptionService : cloudTranscriptionService
|
|
var text = try await transcriptionService.transcribe(audioURL: url, model: model)
|
|
text = text.trimmingCharacters(in: .whitespacesAndNewlines)
|
|
|
|
logger.notice("✅ Transcription completed successfully, length: \(text.count) characters")
|
|
|
|
// --- Post-processing and Saving ---
|
|
let permanentURL = try saveRecordingPermanently(url)
|
|
if UserDefaults.standard.bool(forKey: "IsWordReplacementEnabled") {
|
|
text = WordReplacementService.shared.applyReplacements(to: text)
|
|
logger.notice("✅ Word replacements applied")
|
|
}
|
|
|
|
let audioAsset = AVURLAsset(url: url)
|
|
let actualDuration = CMTimeGetSeconds(try await audioAsset.load(.duration))
|
|
var promptDetectionResult: PromptDetectionService.PromptDetectionResult? = nil
|
|
let originalText = text
|
|
|
|
if let enhancementService = enhancementService, enhancementService.isConfigured {
|
|
let detectionResult = promptDetectionService.analyzeText(text, with: enhancementService)
|
|
promptDetectionResult = detectionResult
|
|
await promptDetectionService.applyDetectionResult(detectionResult, to: enhancementService)
|
|
}
|
|
|
|
if let enhancementService = enhancementService,
|
|
enhancementService.isEnhancementEnabled,
|
|
enhancementService.isConfigured {
|
|
do {
|
|
if shouldCancelRecording { return }
|
|
let textForAI = promptDetectionResult?.processedText ?? text
|
|
let enhancedText = try await enhancementService.enhance(textForAI)
|
|
let newTranscription = Transcription(
|
|
text: originalText,
|
|
duration: actualDuration,
|
|
enhancedText: enhancedText,
|
|
audioFileURL: permanentURL.absoluteString
|
|
)
|
|
modelContext.insert(newTranscription)
|
|
try? modelContext.save()
|
|
text = enhancedText
|
|
} catch {
|
|
let newTranscription = Transcription(
|
|
text: originalText,
|
|
duration: actualDuration,
|
|
audioFileURL: permanentURL.absoluteString
|
|
)
|
|
modelContext.insert(newTranscription)
|
|
try? modelContext.save()
|
|
}
|
|
} else {
|
|
let newTranscription = Transcription(
|
|
text: originalText,
|
|
duration: actualDuration,
|
|
audioFileURL: permanentURL.absoluteString
|
|
)
|
|
modelContext.insert(newTranscription)
|
|
try? modelContext.save()
|
|
}
|
|
|
|
if case .trialExpired = licenseViewModel.licenseState {
|
|
text = """
|
|
Your trial has expired. Upgrade to VoiceInk Pro at tryvoiceink.com/buy
|
|
\n\(text)
|
|
"""
|
|
}
|
|
|
|
text += " "
|
|
|
|
SoundManager.shared.playStopSound()
|
|
if AXIsProcessTrusted() {
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.05) {
|
|
CursorPaster.pasteAtCursor(text, shouldPreserveClipboard: !self.isAutoCopyEnabled)
|
|
}
|
|
}
|
|
try? FileManager.default.removeItem(at: url)
|
|
|
|
if let result = promptDetectionResult,
|
|
let enhancementService = enhancementService,
|
|
result.shouldEnableAI {
|
|
await promptDetectionService.restoreOriginalSettings(result, to: enhancementService)
|
|
}
|
|
|
|
await dismissMiniRecorder()
|
|
await cleanupModelResources()
|
|
|
|
} catch {
|
|
logger.error("❌ Transcription failed: \(error.localizedDescription)")
|
|
await cleanupModelResources()
|
|
await dismissMiniRecorder()
|
|
}
|
|
}
|
|
|
|
private func saveRecordingPermanently(_ tempURL: URL) throws -> URL {
|
|
let fileName = "\(UUID().uuidString).wav"
|
|
let permanentURL = recordingsDirectory.appendingPathComponent(fileName)
|
|
try FileManager.default.copyItem(at: tempURL, to: permanentURL)
|
|
return permanentURL
|
|
}
|
|
|
|
// Loads the default transcription model from UserDefaults
|
|
private func loadCurrentTranscriptionModel() {
|
|
if let savedModelName = UserDefaults.standard.string(forKey: "CurrentTranscriptionModel"),
|
|
let savedModel = allAvailableModels.first(where: { $0.name == savedModelName }) {
|
|
currentTranscriptionModel = savedModel
|
|
}
|
|
}
|
|
|
|
// Function to set any transcription model as default
|
|
func setDefaultTranscriptionModel(_ model: any TranscriptionModel) async {
|
|
await MainActor.run {
|
|
self.currentTranscriptionModel = model
|
|
UserDefaults.standard.set(model.name, forKey: "CurrentTranscriptionModel")
|
|
|
|
// For cloud models, clear the old loadedLocalModel
|
|
if model.provider != .local {
|
|
self.loadedLocalModel = nil
|
|
}
|
|
|
|
// Enable transcription for cloud models immediately since they don't need loading
|
|
if model.provider != .local {
|
|
self.canTranscribe = true
|
|
self.isModelLoaded = true
|
|
}
|
|
}
|
|
|
|
logger.info("Default transcription model set to: \(model.name) (\(model.provider.rawValue))")
|
|
|
|
// Post notification about the model change
|
|
NotificationCenter.default.post(name: .didChangeModel, object: nil, userInfo: ["modelName": model.name])
|
|
}
|
|
|
|
func getEnhancementService() -> AIEnhancementService? {
|
|
return enhancementService
|
|
}
|
|
}
|
|
|
|
struct WhisperModel: Identifiable {
|
|
let id = UUID()
|
|
let name: String
|
|
let url: URL
|
|
var coreMLEncoderURL: URL? // Path to the unzipped .mlmodelc directory
|
|
var isCoreMLDownloaded: Bool { coreMLEncoderURL != nil }
|
|
|
|
var downloadURL: String {
|
|
"https://huggingface.co/ggerganov/whisper.cpp/resolve/main/\(filename)"
|
|
}
|
|
|
|
var filename: String {
|
|
"\(name).bin"
|
|
}
|
|
|
|
// Core ML related properties
|
|
var coreMLZipDownloadURL: String? {
|
|
// Only non-quantized models have Core ML versions
|
|
guard !name.contains("q5") && !name.contains("q8") else { return nil }
|
|
return "https://huggingface.co/ggerganov/whisper.cpp/resolve/main/\(name)-encoder.mlmodelc.zip"
|
|
}
|
|
|
|
var coreMLEncoderDirectoryName: String? {
|
|
guard coreMLZipDownloadURL != nil else { return nil }
|
|
return "\(name)-encoder.mlmodelc"
|
|
}
|
|
}
|
|
|
|
private class TaskDelegate: NSObject, URLSessionTaskDelegate {
|
|
private let continuation: CheckedContinuation<Void, Never>
|
|
|
|
init(_ continuation: CheckedContinuation<Void, Never>) {
|
|
self.continuation = continuation
|
|
}
|
|
|
|
func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
|
|
continuation.resume()
|
|
}
|
|
}
|
|
|
|
extension Notification.Name {
|
|
static let toggleMiniRecorder = Notification.Name("toggleMiniRecorder")
|
|
static let didChangeModel = Notification.Name("didChangeModel")
|
|
}
|