vOOice/VoiceInk/ViewModels/LicenseViewModel.swift

212 lines
8.2 KiB
Swift

import Foundation
import AppKit
@MainActor
class LicenseViewModel: ObservableObject {
enum LicenseState: Equatable {
case trial(daysRemaining: Int)
case trialExpired
case licensed
}
@Published private(set) var licenseState: LicenseState = .trial(daysRemaining: 7) // Default to trial
@Published var licenseKey: String = ""
@Published var isValidating = false
@Published var validationMessage: String?
@Published private(set) var activationsLimit: Int = 0
private let trialPeriodDays = 7
private let polarService = PolarService()
private let userDefaults = UserDefaults.standard
init() {
checkLicenseState()
}
func startTrial() {
// Only set trial start date if it hasn't been set before
if userDefaults.trialStartDate == nil {
userDefaults.trialStartDate = Date()
licenseState = .trial(daysRemaining: trialPeriodDays)
NotificationCenter.default.post(name: .licenseStatusChanged, object: nil)
}
}
private func checkLicenseState() {
// Check for existing license key
if let licenseKey = userDefaults.licenseKey {
self.licenseKey = licenseKey
// Check if this license requires activation
if userDefaults.bool(forKey: "VoiceInkLicenseRequiresActivation") {
// If we have an activation ID, we need to validate it
if let activationId = userDefaults.activationId {
Task {
do {
let isValid = try await polarService.validateLicenseKeyWithActivation(licenseKey, activationId: activationId)
if isValid {
licenseState = .licensed
} else {
// If validation fails, we'll need to reactivate
userDefaults.activationId = nil
licenseState = .trialExpired
}
} catch {
// If there's an error, we'll need to reactivate
userDefaults.activationId = nil
licenseState = .trialExpired
}
}
} else {
// We have a license key but no activation ID, so we need to activate
licenseState = .licensed
}
} else {
// This license doesn't require activation (unlimited devices)
licenseState = .licensed
}
return
}
// Check if this is first launch
let hasLaunchedBefore = userDefaults.bool(forKey: "VoiceInkHasLaunchedBefore")
if !hasLaunchedBefore {
// First launch - start trial automatically
userDefaults.set(true, forKey: "VoiceInkHasLaunchedBefore")
startTrial()
return
}
// Only check trial if not licensed and not first launch
if let trialStartDate = userDefaults.trialStartDate {
let daysSinceTrialStart = Calendar.current.dateComponents([.day], from: trialStartDate, to: Date()).day ?? 0
if daysSinceTrialStart >= trialPeriodDays {
licenseState = .trialExpired
} else {
licenseState = .trial(daysRemaining: trialPeriodDays - daysSinceTrialStart)
}
} else {
// No trial has been started yet - start it now
startTrial()
}
}
var canUseApp: Bool {
switch licenseState {
case .licensed, .trial:
return true
case .trialExpired:
return false
}
}
func openPurchaseLink() {
if let url = URL(string: "https://tryvoiceink.com/buy") {
NSWorkspace.shared.open(url)
}
}
func validateLicense() async {
guard !licenseKey.isEmpty else {
validationMessage = "Please enter a license key"
return
}
isValidating = true
do {
// First, check if the license is valid and if it requires activation
let licenseCheck = try await polarService.checkLicenseRequiresActivation(licenseKey)
if !licenseCheck.isValid {
validationMessage = "Invalid license key"
isValidating = false
return
}
// Store the license key
userDefaults.licenseKey = licenseKey
// Handle based on whether activation is required
if licenseCheck.requiresActivation {
// If we already have an activation ID, validate with it
if let activationId = userDefaults.activationId {
let isValid = try await polarService.validateLicenseKeyWithActivation(licenseKey, activationId: activationId)
if isValid {
// Existing activation is valid
licenseState = .licensed
validationMessage = "License activated successfully!"
NotificationCenter.default.post(name: .licenseStatusChanged, object: nil)
isValidating = false
return
}
}
// Need to create a new activation
let (activationId, limit) = try await polarService.activateLicenseKey(licenseKey)
// Store activation details
userDefaults.activationId = activationId
userDefaults.set(true, forKey: "VoiceInkLicenseRequiresActivation")
self.activationsLimit = limit
} else {
// This license doesn't require activation (unlimited devices)
userDefaults.activationId = nil
userDefaults.set(false, forKey: "VoiceInkLicenseRequiresActivation")
self.activationsLimit = licenseCheck.activationsLimit ?? 0
}
// Update the license state
licenseState = .licensed
validationMessage = "License activated successfully!"
NotificationCenter.default.post(name: .licenseStatusChanged, object: nil)
} catch LicenseError.activationLimitReached {
validationMessage = "This license has reached its maximum number of activations."
} catch LicenseError.activationNotRequired {
// This is actually a success case for unlimited licenses
userDefaults.licenseKey = licenseKey
userDefaults.activationId = nil
userDefaults.set(false, forKey: "VoiceInkLicenseRequiresActivation")
self.activationsLimit = 0
licenseState = .licensed
validationMessage = "License activated successfully!"
NotificationCenter.default.post(name: .licenseStatusChanged, object: nil)
} catch {
validationMessage = "Error validating license: \(error.localizedDescription)"
}
isValidating = false
}
func removeLicense() {
// Remove both license key and trial data
userDefaults.licenseKey = nil
userDefaults.activationId = nil
userDefaults.set(false, forKey: "VoiceInkLicenseRequiresActivation")
userDefaults.trialStartDate = nil
userDefaults.set(false, forKey: "VoiceInkHasLaunchedBefore") // Allow trial to restart
licenseState = .trial(daysRemaining: trialPeriodDays) // Reset to trial state
licenseKey = ""
validationMessage = nil
NotificationCenter.default.post(name: .licenseStatusChanged, object: nil)
checkLicenseState()
}
}
extension Notification.Name {
static let licenseStatusChanged = Notification.Name("licenseStatusChanged")
}
// Add UserDefaults extensions for storing activation ID
extension UserDefaults {
var activationId: String? {
get { string(forKey: "VoiceInkActivationId") }
set { set(newValue, forKey: "VoiceInkActivationId") }
}
}