vOOice/VoiceInk/PowerMode/PowerModeConfig.swift
Beingpax 4ea8d382a4 Add Power Mode keyboard shortcuts and improve session management
Power Mode Keyboard Shortcuts:
- Add hotkeyShortcut property to PowerModeConfig for storing custom shortcuts
- Implement keyboard shortcut UI in Power Mode configuration view
- Add hotkey registration system in HotkeyManager to manage Power Mode shortcuts
- Support cleanup of shortcuts when Power Mode configurations are removed
- Post notifications when Power Mode configurations change

Explicit Power Mode Activation:
- Add optional powerModeId parameter to toggleRecord and toggleMiniRecorder
- Refactor ActiveWindowService.applyConfigurationForCurrentApp to applyConfiguration
- Support direct Power Mode activation via powerModeId instead of auto-detection
- Pass powerModeId through recording flow for explicit mode selection

Session Management Improvements:
- Fix auto-restore to preserve baseline when switching Power Modes mid-recording
- Only capture baseline state on first session creation
- Prevent subsequent beginSession calls from overwriting original baseline
- Ensure auto-restore returns to settings from before recording started

UI Refinements:
- Remove redundant "Record" label from keyboard shortcut recorder
2026-01-03 08:43:11 +05:45

320 lines
13 KiB
Swift

import Foundation
import KeyboardShortcuts
struct PowerModeConfig: Codable, Identifiable, Equatable {
var id: UUID
var name: String
var emoji: String
var appConfigs: [AppConfig]?
var urlConfigs: [URLConfig]?
var isAIEnhancementEnabled: Bool
var selectedPrompt: String?
var selectedTranscriptionModelName: String?
var selectedLanguage: String?
var useScreenCapture: Bool
var selectedAIProvider: String?
var selectedAIModel: String?
var isAutoSendEnabled: Bool = false
var isEnabled: Bool = true
var isDefault: Bool = false
var hotkeyShortcut: String? = nil
enum CodingKeys: String, CodingKey {
case id, name, emoji, appConfigs, urlConfigs, isAIEnhancementEnabled, selectedPrompt, selectedLanguage, useScreenCapture, selectedAIProvider, selectedAIModel, isAutoSendEnabled, isEnabled, isDefault, hotkeyShortcut
case selectedWhisperModel
case selectedTranscriptionModelName
}
init(id: UUID = UUID(), name: String, emoji: String, appConfigs: [AppConfig]? = nil,
urlConfigs: [URLConfig]? = nil, isAIEnhancementEnabled: Bool, selectedPrompt: String? = nil,
selectedTranscriptionModelName: String? = nil, selectedLanguage: String? = nil, useScreenCapture: Bool = false,
selectedAIProvider: String? = nil, selectedAIModel: String? = nil, isAutoSendEnabled: Bool = false, isEnabled: Bool = true, isDefault: Bool = false, hotkeyShortcut: String? = nil) {
self.id = id
self.name = name
self.emoji = emoji
self.appConfigs = appConfigs
self.urlConfigs = urlConfigs
self.isAIEnhancementEnabled = isAIEnhancementEnabled
self.selectedPrompt = selectedPrompt
self.useScreenCapture = useScreenCapture
self.isAutoSendEnabled = isAutoSendEnabled
self.selectedAIProvider = selectedAIProvider ?? UserDefaults.standard.string(forKey: "selectedAIProvider")
self.selectedAIModel = selectedAIModel
self.selectedTranscriptionModelName = selectedTranscriptionModelName ?? UserDefaults.standard.string(forKey: "CurrentTranscriptionModel")
self.selectedLanguage = selectedLanguage ?? UserDefaults.standard.string(forKey: "SelectedLanguage") ?? "en"
self.isEnabled = isEnabled
self.isDefault = isDefault
self.hotkeyShortcut = hotkeyShortcut
}
init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
id = try container.decode(UUID.self, forKey: .id)
name = try container.decode(String.self, forKey: .name)
emoji = try container.decode(String.self, forKey: .emoji)
appConfigs = try container.decodeIfPresent([AppConfig].self, forKey: .appConfigs)
urlConfigs = try container.decodeIfPresent([URLConfig].self, forKey: .urlConfigs)
isAIEnhancementEnabled = try container.decode(Bool.self, forKey: .isAIEnhancementEnabled)
selectedPrompt = try container.decodeIfPresent(String.self, forKey: .selectedPrompt)
selectedLanguage = try container.decodeIfPresent(String.self, forKey: .selectedLanguage)
useScreenCapture = try container.decode(Bool.self, forKey: .useScreenCapture)
selectedAIProvider = try container.decodeIfPresent(String.self, forKey: .selectedAIProvider)
selectedAIModel = try container.decodeIfPresent(String.self, forKey: .selectedAIModel)
isAutoSendEnabled = try container.decodeIfPresent(Bool.self, forKey: .isAutoSendEnabled) ?? false
isEnabled = try container.decodeIfPresent(Bool.self, forKey: .isEnabled) ?? true
isDefault = try container.decodeIfPresent(Bool.self, forKey: .isDefault) ?? false
hotkeyShortcut = try container.decodeIfPresent(String.self, forKey: .hotkeyShortcut)
if let newModelName = try container.decodeIfPresent(String.self, forKey: .selectedTranscriptionModelName) {
selectedTranscriptionModelName = newModelName
} else if let oldModelName = try container.decodeIfPresent(String.self, forKey: .selectedWhisperModel) {
selectedTranscriptionModelName = oldModelName
} else {
selectedTranscriptionModelName = nil
}
}
func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(id, forKey: .id)
try container.encode(name, forKey: .name)
try container.encode(emoji, forKey: .emoji)
try container.encodeIfPresent(appConfigs, forKey: .appConfigs)
try container.encodeIfPresent(urlConfigs, forKey: .urlConfigs)
try container.encode(isAIEnhancementEnabled, forKey: .isAIEnhancementEnabled)
try container.encodeIfPresent(selectedPrompt, forKey: .selectedPrompt)
try container.encodeIfPresent(selectedLanguage, forKey: .selectedLanguage)
try container.encode(useScreenCapture, forKey: .useScreenCapture)
try container.encodeIfPresent(selectedAIProvider, forKey: .selectedAIProvider)
try container.encodeIfPresent(selectedAIModel, forKey: .selectedAIModel)
try container.encode(isAutoSendEnabled, forKey: .isAutoSendEnabled)
try container.encodeIfPresent(selectedTranscriptionModelName, forKey: .selectedTranscriptionModelName)
try container.encode(isEnabled, forKey: .isEnabled)
try container.encode(isDefault, forKey: .isDefault)
try container.encodeIfPresent(hotkeyShortcut, forKey: .hotkeyShortcut)
}
static func == (lhs: PowerModeConfig, rhs: PowerModeConfig) -> Bool {
lhs.id == rhs.id
}
}
struct AppConfig: Codable, Identifiable, Equatable {
let id: UUID
var bundleIdentifier: String
var appName: String
init(id: UUID = UUID(), bundleIdentifier: String, appName: String) {
self.id = id
self.bundleIdentifier = bundleIdentifier
self.appName = appName
}
static func == (lhs: AppConfig, rhs: AppConfig) -> Bool {
lhs.id == rhs.id
}
}
struct URLConfig: Codable, Identifiable, Equatable {
let id: UUID
var url: String
init(id: UUID = UUID(), url: String) {
self.id = id
self.url = url
}
static func == (lhs: URLConfig, rhs: URLConfig) -> Bool {
lhs.id == rhs.id
}
}
class PowerModeManager: ObservableObject {
static let shared = PowerModeManager()
@Published var configurations: [PowerModeConfig] = []
@Published var activeConfiguration: PowerModeConfig?
private let configKey = "powerModeConfigurationsV2"
private let activeConfigIdKey = "activeConfigurationId"
private init() {
loadConfigurations()
if let activeConfigIdString = UserDefaults.standard.string(forKey: activeConfigIdKey),
let activeConfigId = UUID(uuidString: activeConfigIdString) {
activeConfiguration = configurations.first { $0.id == activeConfigId }
} else {
activeConfiguration = nil
}
}
private func loadConfigurations() {
if let data = UserDefaults.standard.data(forKey: configKey),
let configs = try? JSONDecoder().decode([PowerModeConfig].self, from: data) {
configurations = configs
}
}
func saveConfigurations() {
if let data = try? JSONEncoder().encode(configurations) {
UserDefaults.standard.set(data, forKey: configKey)
}
NotificationCenter.default.post(name: NSNotification.Name("PowerModeConfigurationsDidChange"), object: nil)
}
func addConfiguration(_ config: PowerModeConfig) {
if !configurations.contains(where: { $0.id == config.id }) {
configurations.append(config)
saveConfigurations()
}
}
func removeConfiguration(with id: UUID) {
KeyboardShortcuts.setShortcut(nil, for: .powerMode(id: id))
configurations.removeAll { $0.id == id }
saveConfigurations()
}
func getConfiguration(with id: UUID) -> PowerModeConfig? {
return configurations.first { $0.id == id }
}
func updateConfiguration(_ config: PowerModeConfig) {
if let index = configurations.firstIndex(where: { $0.id == config.id }) {
configurations[index] = config
saveConfigurations()
}
}
func moveConfigurations(fromOffsets: IndexSet, toOffset: Int) {
configurations.move(fromOffsets: fromOffsets, toOffset: toOffset)
saveConfigurations()
}
func getConfigurationForURL(_ url: String) -> PowerModeConfig? {
let cleanedURL = cleanURL(url)
for config in configurations.filter({ $0.isEnabled }) {
if let urlConfigs = config.urlConfigs {
for urlConfig in urlConfigs {
let configURL = cleanURL(urlConfig.url)
if cleanedURL.contains(configURL) {
return config
}
}
}
}
return nil
}
func getConfigurationForApp(_ bundleId: String) -> PowerModeConfig? {
for config in configurations.filter({ $0.isEnabled }) {
if let appConfigs = config.appConfigs {
if appConfigs.contains(where: { $0.bundleIdentifier == bundleId }) {
return config
}
}
}
return nil
}
func getDefaultConfiguration() -> PowerModeConfig? {
return configurations.first { $0.isEnabled && $0.isDefault }
}
func hasDefaultConfiguration() -> Bool {
return configurations.contains { $0.isDefault }
}
func setAsDefault(configId: UUID, skipSave: Bool = false) {
for index in configurations.indices {
configurations[index].isDefault = false
}
if let index = configurations.firstIndex(where: { $0.id == configId }) {
configurations[index].isDefault = true
}
if !skipSave {
saveConfigurations()
}
}
func enableConfiguration(with id: UUID) {
if let index = configurations.firstIndex(where: { $0.id == id }) {
configurations[index].isEnabled = true
saveConfigurations()
}
}
func disableConfiguration(with id: UUID) {
if let index = configurations.firstIndex(where: { $0.id == id }) {
configurations[index].isEnabled = false
saveConfigurations()
}
}
var enabledConfigurations: [PowerModeConfig] {
return configurations.filter { $0.isEnabled }
}
func addAppConfig(_ appConfig: AppConfig, to config: PowerModeConfig) {
if var updatedConfig = configurations.first(where: { $0.id == config.id }) {
var configs = updatedConfig.appConfigs ?? []
configs.append(appConfig)
updatedConfig.appConfigs = configs
updateConfiguration(updatedConfig)
}
}
func removeAppConfig(_ appConfig: AppConfig, from config: PowerModeConfig) {
if var updatedConfig = configurations.first(where: { $0.id == config.id }) {
updatedConfig.appConfigs?.removeAll(where: { $0.id == appConfig.id })
updateConfiguration(updatedConfig)
}
}
func addURLConfig(_ urlConfig: URLConfig, to config: PowerModeConfig) {
if var updatedConfig = configurations.first(where: { $0.id == config.id }) {
var configs = updatedConfig.urlConfigs ?? []
configs.append(urlConfig)
updatedConfig.urlConfigs = configs
updateConfiguration(updatedConfig)
}
}
func removeURLConfig(_ urlConfig: URLConfig, from config: PowerModeConfig) {
if var updatedConfig = configurations.first(where: { $0.id == config.id }) {
updatedConfig.urlConfigs?.removeAll(where: { $0.id == urlConfig.id })
updateConfiguration(updatedConfig)
}
}
func cleanURL(_ url: String) -> String {
return url.lowercased()
.replacingOccurrences(of: "https://", with: "")
.replacingOccurrences(of: "http://", with: "")
.replacingOccurrences(of: "www.", with: "")
.trimmingCharacters(in: .whitespacesAndNewlines)
}
func setActiveConfiguration(_ config: PowerModeConfig?) {
activeConfiguration = config
UserDefaults.standard.set(config?.id.uuidString, forKey: activeConfigIdKey)
self.objectWillChange.send()
}
var currentActiveConfiguration: PowerModeConfig? {
return activeConfiguration
}
func getAllAvailableConfigurations() -> [PowerModeConfig] {
return configurations
}
func isEmojiInUse(_ emoji: String) -> Bool {
return configurations.contains { $0.emoji == emoji }
}
}