486 lines
12 KiB
Objective-C
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
|