2019-05-24 11:29:27 -07:00
//
// A p p D e l e g a t e . s w i f t
// A l t S e r v e r
//
// C r e a t e d b y R i l e y T e s t u t o n 5 / 2 4 / 1 9 .
// C o p y r i g h t © 2 0 1 9 R i l e y T e s t u t . A l l r i g h t s r e s e r v e d .
//
import Cocoa
2019-07-01 15:19:22 -07:00
import UserNotifications
import AltSign
2019-05-24 11:29:27 -07:00
2019-09-04 11:58:28 -07:00
import LaunchAtLogin
2019-11-18 14:42:38 -08:00
import STPrivilegedTask
enum PluginError : LocalizedError
{
case installationScriptNotFound
case failedToRun ( Int )
case scriptError ( String )
var errorDescription : String ? {
switch self
{
case . installationScriptNotFound : return NSLocalizedString ( " The installation script could not be found. " , comment : " " )
case . failedToRun ( let errorCode ) : return String ( format : NSLocalizedString ( " The installation script could not be run. (%@) " , comment : " " ) , NSNumber ( value : errorCode ) )
case . scriptError ( let output ) : return output
}
}
}
private let pluginURL = URL ( fileURLWithPath : " /Library/Mail/Bundles/AltPlugin.mailbundle " )
2019-09-04 11:58:28 -07:00
2019-05-24 11:29:27 -07:00
@NSApplicationMain
class AppDelegate : NSObject , NSApplicationDelegate {
2019-07-01 15:19:22 -07:00
private var statusItem : NSStatusItem ?
private var connectedDevices = [ ALTDevice ] ( )
private weak var authenticationAlert : NSAlert ?
@IBOutlet private var appMenu : NSMenu !
@IBOutlet private var connectedDevicesMenu : NSMenu !
2019-09-04 11:58:28 -07:00
@IBOutlet private var launchAtLoginMenuItem : NSMenuItem !
2019-11-18 14:42:38 -08:00
@IBOutlet private var installMailPluginMenuItem : NSMenuItem !
2019-07-01 15:19:22 -07:00
private weak var authenticationAppleIDTextField : NSTextField ?
private weak var authenticationPasswordTextField : NSSecureTextField ?
2019-11-18 14:42:38 -08:00
private var isMailPluginInstalled : Bool {
let isMailPluginInstalled = FileManager . default . fileExists ( atPath : pluginURL . path )
return isMailPluginInstalled
}
2019-07-01 15:19:22 -07:00
func applicationDidFinishLaunching ( _ aNotification : Notification )
{
2019-08-01 10:45:54 -07:00
UserDefaults . standard . registerDefaults ( )
2019-07-01 15:19:22 -07:00
UNUserNotificationCenter . current ( ) . delegate = self
2020-01-13 10:17:30 -08:00
2019-07-01 15:19:22 -07:00
ConnectionManager . shared . start ( )
2020-01-13 10:17:30 -08:00
ALTDeviceManager . shared . start ( )
2019-07-01 15:19:22 -07:00
let item = NSStatusBar . system . statusItem ( withLength : - 1 )
guard let button = item . button else { return }
button . image = NSImage ( named : " MenuBarIcon " )
button . target = self
button . action = #selector ( AppDelegate . presentMenu )
self . statusItem = item
self . connectedDevicesMenu . delegate = self
2019-09-25 01:23:23 -07:00
2019-11-04 15:07:51 -08:00
UNUserNotificationCenter . current ( ) . requestAuthorization ( options : [ . alert ] ) { ( success , error ) in
guard success else { return }
2019-09-25 01:23:23 -07:00
2019-11-04 15:07:51 -08:00
if ! UserDefaults . standard . didPresentInitialNotification
{
let content = UNMutableNotificationContent ( )
content . title = NSLocalizedString ( " AltServer Running " , comment : " " )
content . body = NSLocalizedString ( " AltServer runs in the background as a menu bar app listening for AltStore. " , comment : " " )
let request = UNNotificationRequest ( identifier : UUID ( ) . uuidString , content : content , trigger : nil )
UNUserNotificationCenter . current ( ) . add ( request )
UserDefaults . standard . didPresentInitialNotification = true
}
2019-09-25 01:23:23 -07:00
}
2019-05-24 11:29:27 -07:00
}
2019-07-01 15:19:22 -07:00
func applicationWillTerminate ( _ aNotification : Notification )
{
2019-05-24 11:29:27 -07:00
// I n s e r t c o d e h e r e t o t e a r d o w n y o u r a p p l i c a t i o n
}
2019-07-01 15:19:22 -07:00
}
private extension AppDelegate
{
@objc func presentMenu ( )
{
guard let button = self . statusItem ? . button , let superview = button . superview , let window = button . window else { return }
2020-01-16 16:03:46 -08:00
self . connectedDevices = ALTDeviceManager . shared . availableDevices
2019-07-01 15:19:22 -07:00
2019-09-04 11:58:28 -07:00
self . launchAtLoginMenuItem . state = LaunchAtLogin . isEnabled ? . on : . off
self . launchAtLoginMenuItem . action = #selector ( AppDelegate . toggleLaunchAtLogin ( _ : ) )
2019-11-18 14:42:38 -08:00
if FileManager . default . fileExists ( atPath : pluginURL . path )
{
self . installMailPluginMenuItem . title = NSLocalizedString ( " Uninstall Mail Plug-in " , comment : " " )
}
else
{
self . installMailPluginMenuItem . title = NSLocalizedString ( " Install Mail Plug-in " , comment : " " )
}
self . installMailPluginMenuItem . target = self
self . installMailPluginMenuItem . action = #selector ( AppDelegate . handleInstallMailPluginMenuItem ( _ : ) )
2019-07-01 15:19:22 -07:00
let x = button . frame . origin . x
let y = button . frame . origin . y - 5
let location = superview . convert ( NSMakePoint ( x , y ) , to : nil )
guard let event = NSEvent . mouseEvent ( with : . leftMouseUp , location : location ,
modifierFlags : [ ] , timestamp : 0 , windowNumber : window . windowNumber , context : nil ,
eventNumber : 0 , clickCount : 1 , pressure : 0 )
else { return }
NSMenu . popUpContextMenu ( self . appMenu , with : event , for : button )
}
@objc func installAltStore ( _ item : NSMenuItem )
{
guard case let index = self . connectedDevicesMenu . index ( of : item ) , index != - 1 else { return }
let alert = NSAlert ( )
alert . messageText = NSLocalizedString ( " Please enter your Apple ID and password. " , comment : " " )
2019-11-18 14:17:57 -08:00
alert . informativeText = NSLocalizedString ( " Your Apple ID and password are not saved and are only sent to Apple for authentication. " , comment : " " )
2019-07-01 15:19:22 -07:00
let textFieldSize = NSSize ( width : 300 , height : 22 )
let appleIDTextField = NSTextField ( frame : NSRect ( x : 0 , y : 0 , width : textFieldSize . width , height : textFieldSize . height ) )
appleIDTextField . delegate = self
appleIDTextField . translatesAutoresizingMaskIntoConstraints = false
appleIDTextField . placeholderString = NSLocalizedString ( " Apple ID " , comment : " " )
alert . window . initialFirstResponder = appleIDTextField
self . authenticationAppleIDTextField = appleIDTextField
let passwordTextField = NSSecureTextField ( frame : NSRect ( x : 0 , y : 0 , width : textFieldSize . width , height : textFieldSize . height ) )
passwordTextField . delegate = self
passwordTextField . translatesAutoresizingMaskIntoConstraints = false
passwordTextField . placeholderString = NSLocalizedString ( " Password " , comment : " " )
self . authenticationPasswordTextField = passwordTextField
appleIDTextField . nextKeyView = passwordTextField
let stackView = NSStackView ( frame : NSRect ( x : 0 , y : 0 , width : textFieldSize . width , height : textFieldSize . height * 2 ) )
stackView . orientation = . vertical
stackView . distribution = . equalSpacing
stackView . spacing = 0
stackView . addArrangedSubview ( appleIDTextField )
stackView . addArrangedSubview ( passwordTextField )
alert . accessoryView = stackView
alert . addButton ( withTitle : NSLocalizedString ( " Install " , comment : " " ) )
alert . addButton ( withTitle : NSLocalizedString ( " Cancel " , comment : " " ) )
self . authenticationAlert = alert
self . validate ( )
NSRunningApplication . current . activate ( options : . activateIgnoringOtherApps )
let response = alert . runModal ( )
guard response = = . alertFirstButtonReturn else { return }
let username = appleIDTextField . stringValue
let password = passwordTextField . stringValue
let device = self . connectedDevices [ index ]
2019-11-18 14:42:38 -08:00
if ! self . isMailPluginInstalled
{
let result = self . installMailPlugin ( )
guard result else { return }
}
2019-07-01 15:19:22 -07:00
ALTDeviceManager . shared . installAltStore ( to : device , appleID : username , password : password ) { ( result ) in
switch result
{
case . success :
2019-09-04 12:08:27 -07:00
let content = UNMutableNotificationContent ( )
2019-07-01 15:19:22 -07:00
content . title = NSLocalizedString ( " Installation Succeeded " , comment : " " )
content . body = String ( format : NSLocalizedString ( " AltStore was successfully installed on %@. " , comment : " " ) , device . name )
2019-09-04 12:08:27 -07:00
let request = UNNotificationRequest ( identifier : UUID ( ) . uuidString , content : content , trigger : nil )
UNUserNotificationCenter . current ( ) . add ( request )
2019-11-18 14:17:57 -08:00
case . failure ( InstallError . cancelled ) , . failure ( ALTAppleAPIError . requiresTwoFactorAuthentication ) :
2019-09-14 11:28:57 -07:00
// I g n o r e
break
2019-09-13 14:25:26 -07:00
case . failure ( let error as NSError ) :
2019-09-04 12:08:27 -07:00
let alert = NSAlert ( )
2019-09-13 14:25:26 -07:00
alert . alertStyle = . critical
2019-09-04 12:08:27 -07:00
alert . messageText = NSLocalizedString ( " Installation Failed " , comment : " " )
2019-09-13 14:25:26 -07:00
if let underlyingError = error . userInfo [ NSUnderlyingErrorKey ] as ? Error
{
alert . informativeText = underlyingError . localizedDescription
}
else
{
alert . informativeText = error . localizedDescription
}
2019-09-04 12:08:27 -07:00
2019-09-13 14:25:26 -07:00
NSRunningApplication . current . activate ( options : . activateIgnoringOtherApps )
2019-09-04 12:08:27 -07:00
alert . runModal ( )
}
2019-07-01 15:19:22 -07:00
}
}
2019-09-04 11:58:28 -07:00
@objc func toggleLaunchAtLogin ( _ item : NSMenuItem )
{
if item . state = = . on
{
item . state = . off
}
else
{
item . state = . on
}
LaunchAtLogin . isEnabled . toggle ( )
}
2019-11-18 14:42:38 -08:00
@objc func handleInstallMailPluginMenuItem ( _ item : NSMenuItem )
{
installMailPlugin ( )
}
@ discardableResult
func installMailPlugin ( ) -> Bool
{
do
{
let previouslyInstalled = self . isMailPluginInstalled
if ! previouslyInstalled
{
let alert = NSAlert ( )
alert . messageText = NSLocalizedString ( " Install Mail Plug-in " , comment : " " )
alert . informativeText = NSLocalizedString ( " AltServer requires a Mail plug-in in order to retrieve necessary information about your Apple ID. Would you like to install it now? " , comment : " " )
alert . addButton ( withTitle : NSLocalizedString ( " Install Plug-in " , comment : " " ) )
alert . addButton ( withTitle : NSLocalizedString ( " Cancel " , comment : " " ) )
NSRunningApplication . current . activate ( options : . activateIgnoringOtherApps )
let response = alert . runModal ( )
guard response = = . alertFirstButtonReturn else { return false }
}
guard let scriptURL = Bundle . main . url ( forResource : self . isMailPluginInstalled ? " UninstallPlugin " : " InstallPlugin " , withExtension : " sh " ) else { throw PluginError . installationScriptNotFound }
try FileManager . default . setAttributes ( [ . posixPermissions : 0o777 ] , ofItemAtPath : scriptURL . path )
let task = STPrivilegedTask ( )
task . setLaunchPath ( scriptURL . path )
task . setCurrentDirectoryPath ( scriptURL . deletingLastPathComponent ( ) . path )
let errorCode = task . launch ( )
guard errorCode = = 0 else { throw PluginError . failedToRun ( Int ( errorCode ) ) }
task . waitUntilExit ( )
if
let outputData = task . outputFileHandle ( ) ? . readDataToEndOfFile ( ) ,
let outputString = String ( data : outputData , encoding : . utf8 ) , ! outputString . isEmpty
{
throw PluginError . scriptError ( outputString )
}
if ! previouslyInstalled && self . isMailPluginInstalled
{
let alert = NSAlert ( )
alert . messageText = NSLocalizedString ( " Mail Plug-in Installed " , comment : " " )
alert . informativeText = NSLocalizedString ( " Please restart Mail and enable AltPlugin in Mail's Preferences. Mail must be running when installing or refreshing apps with AltServer. " , comment : " " )
alert . runModal ( )
}
return true
}
catch
{
let alert = NSAlert ( )
alert . messageText = self . isMailPluginInstalled ? NSLocalizedString ( " Failed to Uninstall Mail Plug-in " , comment : " " ) : NSLocalizedString ( " Failed to Install Mail Plug-in " , comment : " " )
alert . informativeText = error . localizedDescription
alert . runModal ( )
return false
}
}
2019-07-01 15:19:22 -07:00
}
2019-05-24 11:29:27 -07:00
2019-07-01 15:19:22 -07:00
extension AppDelegate : NSMenuDelegate
{
func numberOfItems ( in menu : NSMenu ) -> Int
{
return self . connectedDevices . isEmpty ? 1 : self . connectedDevices . count
}
func menu ( _ menu : NSMenu , update item : NSMenuItem , at index : Int , shouldCancel : Bool ) -> Bool
{
if self . connectedDevices . isEmpty
{
item . title = NSLocalizedString ( " No Connected Devices " , comment : " " )
item . isEnabled = false
item . target = nil
item . action = nil
}
else
{
let device = self . connectedDevices [ index ]
item . title = device . name
item . isEnabled = true
item . target = self
item . action = #selector ( AppDelegate . installAltStore )
item . tag = index
}
return true
}
}
2019-05-24 11:29:27 -07:00
2019-07-01 15:19:22 -07:00
extension AppDelegate : NSTextFieldDelegate
{
func controlTextDidChange ( _ obj : Notification )
{
self . validate ( )
}
func controlTextDidEndEditing ( _ obj : Notification )
{
self . validate ( )
}
private func validate ( )
{
guard
let appleID = self . authenticationAppleIDTextField ? . stringValue . trimmingCharacters ( in : . whitespacesAndNewlines ) ,
let password = self . authenticationPasswordTextField ? . stringValue . trimmingCharacters ( in : . whitespacesAndNewlines )
else { return }
if appleID . isEmpty || password . isEmpty
{
self . authenticationAlert ? . buttons . first ? . isEnabled = false
}
else
{
self . authenticationAlert ? . buttons . first ? . isEnabled = true
}
self . authenticationAlert ? . layout ( )
}
2019-05-24 11:29:27 -07:00
}
2019-07-01 15:19:22 -07:00
extension AppDelegate : UNUserNotificationCenterDelegate
{
func userNotificationCenter ( _ center : UNUserNotificationCenter , willPresent notification : UNNotification , withCompletionHandler completionHandler : @ escaping ( UNNotificationPresentationOptions ) -> Void )
{
completionHandler ( [ . alert , . sound , . badge ] )
}
}