OverSight/MainApp/AppDelegate.m

486 lines
12 KiB
Objective-C

//
// AppDelegate.m
// Test Application
//
// Created by Patrick Wardle on 9/10/16.
// Copyright (c) 2016 Objective-See. All rights reserved.
//
#import "Consts.h"
#import "Logging.h"
#import "Utilities.h"
#import "AppDelegate.h"
@interface AppDelegate ()
@property (weak) IBOutlet NSWindow *window;
@end
@implementation AppDelegate
@synthesize infoWindowController;
@synthesize aboutWindowController;
//center window
// ->also make front, init title bar, etc
-(void)awakeFromNib
{
//center
[self.window center];
//make it key window
[self.window makeKeyAndOrderFront:self];
//make window front
[NSApp activateIgnoringOtherApps:YES];
//set button states
[self setButtonStates];
//set title
self.window.title = [NSString stringWithFormat:@"OverSight Preferences (v. %@)", getAppVersion()];
return;
}
//app interface
// ->init user interface
-(void)applicationDidFinishLaunching:(NSNotification *)aNotification
{
//register for hotkey presses
// ->for now, just cmd+q to quit app
[self registerKeypressHandler];
//start login item in background
// ->checks if already running though
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
^{
//start
[self startLoginItem:NO];
});
return;
}
//automatically close when user closes window
-(BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)theApplication
{
return YES;
}
//set button states from preferences
-(void)setButtonStates
{
//preferences
NSDictionary* preferences = nil;
//load preferences
preferences = [NSDictionary dictionaryWithContentsOfFile:[APP_PREFERENCES stringByExpandingTildeInPath]];
//set 'log activity' button state
self.logActivity.state = [preferences[PREF_LOG_ACTIVITY] boolValue];
//set 'start at login' button state
self.startAtLogin.state = [preferences[PREF_START_AT_LOGIN] boolValue];
//set 'run headless' button state
self.runHeadless.state = [preferences[PREF_RUN_HEADLESS] boolValue];
//set 'automatically check for updates' button state
self.check4Updates.state = [preferences[PREF_CHECK_4_UPDATES] boolValue];
return;
}
//register handler for hot keys
-(void)registerKeypressHandler
{
//event handler
NSEvent* (^keypressHandler)(NSEvent *) = nil;
//init handler block
// ->just call helper function
keypressHandler = ^NSEvent * (NSEvent * theEvent){
//invoke helper
return [self handleKeypress:theEvent];
};
//register for key-down events
[NSEvent addLocalMonitorForEventsMatchingMask:NSKeyDownMask handler:keypressHandler];
return;
}
//helper function for keypresses
// ->for now, only handle cmd+q, to quit
-(NSEvent*)handleKeypress:(NSEvent*)event
{
//flag indicating event was handled
BOOL wasHandled = NO;
//only care about 'cmd' + something
if(NSCommandKeyMask != (event.modifierFlags & NSCommandKeyMask))
{
//bail
goto bail;
}
//handle key-code
// command+q: quite
switch ([event keyCode])
{
//'q' (quit)
case KEYCODE_Q:
//bye!
[[NSApplication sharedApplication] terminate:nil];
//set flag
wasHandled = YES;
break;
//default
// ->do nothing
default:
break;
}
//bail
bail:
//nil out event if it was handled
if(YES == wasHandled)
{
//nil
event = nil;
}
return event;
}
//toggle/set preferences
-(IBAction)togglePreference:(NSButton *)sender
{
//preferences
NSMutableDictionary* preferences = nil;
//path to login item
NSURL* loginItem = nil;
//load preferences
preferences = [NSMutableDictionary dictionaryWithContentsOfFile:[APP_PREFERENCES stringByExpandingTildeInPath]];
//set 'log activity' button
if(sender == self.logActivity)
{
//set
preferences[PREF_LOG_ACTIVITY] = [NSNumber numberWithBool:[sender state]];
}
//set 'automatically check for updates'
else if(sender == self.check4Updates)
{
//set
preferences[PREF_CHECK_4_UPDATES] = [NSNumber numberWithBool:[sender state]];
}
//set 'start at login'
// ->then also toggle for current user
else if(sender == self.startAtLogin)
{
//set
preferences[PREF_START_AT_LOGIN] = [NSNumber numberWithBool:[sender state]];
//init path to login item
loginItem = [NSURL fileURLWithPath:[[[NSBundle mainBundle] bundlePath] stringByAppendingPathComponent:@"/Contents/Library/LoginItems/OverSight Helper.app"]];
//install
toggleLoginItem(loginItem, (int)[sender state]);
}
//set 'run in headless mode'
// ->then restart login item to realize this
else if(sender == self.runHeadless)
{
//set
preferences[PREF_RUN_HEADLESS] = [NSNumber numberWithBool:[sender state]];
//restart login item in background
// ->will read prefs, and run in headless mode
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
^{
//start
[self startLoginItem:YES];
});
}
//save em
[preferences writeToFile:[APP_PREFERENCES stringByExpandingTildeInPath] atomically:YES];
return;
}
//'about' button/menu handler
-(IBAction)about:(id)sender
{
//alloc/init settings window
if(nil == self.aboutWindowController)
{
//alloc/init
aboutWindowController = [[AboutWindowController alloc] initWithWindowNibName:@"AboutWindow"];
}
//center window
[[self.aboutWindowController window] center];
//show it
[self.aboutWindowController showWindow:self];
//invoke function in background that will make window modal
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
//make modal
makeModal(self.aboutWindowController);
});
return;
}
//'check for update' (now) button handler
-(IBAction)check4Update:(id)sender
{
//disable button
self.check4UpdatesNow.enabled = NO;
//reset
self.versionLabel.stringValue = @"";
//re-draw
[self.versionLabel displayIfNeeded];
//show spinner
[self.spinner startAnimation:self];
//check for update
[self isThereAnUpdate];
return;
}
//check for an update
-(void)isThereAnUpdate
{
//version string
NSMutableString* versionString = nil;
//alloc string
versionString = [NSMutableString string];
//check if available version is newer
// ->show update window
if(YES == isNewVersion(versionString))
{
//dbg msg
logMsg(LOG_DEBUG, [NSString stringWithFormat:@"a new version (%@) is available", versionString]);
//hide version message
self.versionLabel.hidden = YES;
//alloc/init about window
infoWindowController = [[InfoWindowController alloc] initWithWindowNibName:@"InfoWindow"];
//configure
[self.infoWindowController configure:[NSString stringWithFormat:@"a new version (%@) is available!", versionString] buttonTitle:@"update"];
//center window
[[self.infoWindowController window] center];
//show it
[self.infoWindowController showWindow:self];
//invoke function in background that will make window modal
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
//make modal
makeModal(self.infoWindowController);
});
//stop/hide spinner
[self.spinner stopAnimation:self];
//re-enable button
self.check4UpdatesNow.enabled = YES;
}
//no new version
// ->stop animations/just (debug) log msg
else
{
//dbg msg
logMsg(LOG_DEBUG, @"no updates available");
//stop/hide spinner
[self.spinner stopAnimation:self];
//re-enable button
self.check4UpdatesNow.enabled = YES;
//show now new version message
self.versionLabel.hidden = NO;
//set message
self.versionLabel.stringValue = @"No new versions";
//re-draw
[self.versionLabel displayIfNeeded];
}
return;
}
//start the login item
-(void)startLoginItem:(BOOL)shouldRestart
{
//path to login item
NSString* loginItem = nil;
//login item's pid
pid_t loginItemPID = -1;
//get pid of login item
loginItemPID = getProcessID(@"OverSight Helper", getuid());
//already running?
// ->kill the login item
if( (YES == shouldRestart) &&
(-1 != loginItemPID) )
{
//kill
kill(loginItemPID, SIGTERM);
//sleep
sleep(2);
//really kill
kill(loginItemPID, SIGKILL);
//reset pid
loginItemPID = -1;
}
//start not already running
if(-1 == loginItemPID)
{
//dbg msg
logMsg(LOG_DEBUG, @"starting login item");
//show overlay view on main thread
dispatch_async(dispatch_get_main_queue(), ^{
//pre-req
[self.overlay setWantsLayer:YES];
//round edges
[self.overlay.layer setCornerRadius: 10];
//set overlay's view color to white
self.overlay.layer.backgroundColor = [NSColor grayColor].CGColor;
//make it semi-transparent
self.overlay.alphaValue = 0.85;
//show it
self.overlay.hidden = NO;
//show message
self.statusMessage.hidden = NO;
//show spinner
self.progressIndicator.hidden = NO;
//animate it
[self.progressIndicator startAnimation:nil];
});
}
//init path
loginItem = [[[NSBundle mainBundle] bundlePath] stringByAppendingPathComponent:@"/Contents/Library/LoginItems/OverSight Helper.app"];
//launch it
if(YES != [[NSWorkspace sharedWorkspace] launchApplication:loginItem])
{
//err msg
logMsg(LOG_ERR, [NSString stringWithFormat:@"failed to start login item, %@", loginItem]);
//bail
goto bail;
}
//(re)obtain focus for app
[[NSRunningApplication currentApplication] activateWithOptions:NSApplicationActivateIgnoringOtherApps];
//bail
bail:
//hide overlay?
if(-1 == loginItemPID)
{
//sleep to give message some more time
[NSThread sleepForTimeInterval:1];
//update message
dispatch_async(dispatch_get_main_queue(), ^{
//stop spinner
[self.progressIndicator stopAnimation:nil];
//update
self.statusMessage.stringValue = @"started!";
});
//sleep to give message some more time
[NSThread sleepForTimeInterval:1];
//hide overlay view on main thread
dispatch_async(dispatch_get_main_queue(), ^{
//hide spinner
self.progressIndicator.hidden = YES;
//hide view
self.overlay.hidden = YES;
//hide message
self.statusMessage.hidden = YES;
});
}
return;
}
//manage rules
-(IBAction)manageRules:(id)sender
{
return;
}
@end