2016-03-30 23:07:55 +03:00
# import "AppDelegate.h"
2017-01-24 21:00:56 +02:00
# include "GBButtons.h"
2020-03-26 20:54:18 +02:00
# include "GBView.h"
2017-10-13 00:02:02 +03:00
# include < Core / gb . h >
2017-01-24 21:00:56 +02:00
# import < Carbon / Carbon . h >
2019-10-19 19:26:04 +03:00
# import < JoyKit / JoyKit . h >
2021-04-25 22:28:24 +03:00
# import < WebKit / WebKit . h >
# define UPDATE_SERVER "https://sameboy.github.io"
static uint32_t color_to _int ( NSColor * color )
{
color = [ color colorUsingColorSpace : [ NSColorSpace deviceRGBColorSpace ] ] ;
return ( ( ( unsigned ) ( color . redComponent * 0 xFF ) ) < < 16 ) |
( ( ( unsigned ) ( color . greenComponent * 0 xFF ) ) < < 8 ) |
( ( unsigned ) ( color . blueComponent * 0 xFF ) ) ;
}
2016-03-30 23:07:55 +03:00
@ implementation AppDelegate
2016-04-13 22:43:16 +03:00
{
NSWindow * preferences_window ;
2018-12-01 16:08:59 +02:00
NSArray < NSView * > * preferences_tabs ;
2021-04-25 22:28:24 +03:00
NSString * _lastVersion ;
NSString * _updateURL ;
NSURLSessionDownloadTask * _updateTask ;
enum {
UPDATE_DOWNLOADING ,
UPDATE_EXTRACTING ,
UPDATE_WAIT _INSTALL ,
UPDATE_INSTALLING ,
UPDATE_FAILED ,
} _updateState ;
NSString * _downloadDirectory ;
2016-04-13 22:43:16 +03:00
}
- ( void ) applicationDidFinishLaunching : ( NSNotification * ) notification
{
2017-01-24 21:00:56 +02:00
NSUserDefaults * defaults = [ NSUserDefaults standardUserDefaults ] ;
for ( unsigned i = 0 ; i < GBButtonCount ; i + + ) {
2018-12-05 00:00:16 +02:00
if ( [ [ defaults objectForKey : button_to _preference _name ( i , 0 ) ] isKindOfClass : [ NSString class ] ] ) {
[ defaults removeObjectForKey : button_to _preference _name ( i , 0 ) ] ;
2017-01-24 21:00:56 +02:00
}
}
2016-04-13 22:43:16 +03:00
[ [ NSUserDefaults standardUserDefaults ] registerDefaults : @ {
2017-01-24 21:00:56 +02:00
@ "GBRight" : @ ( kVK_RightArrow ) ,
@ "GBLeft" : @ ( kVK_LeftArrow ) ,
@ "GBUp" : @ ( kVK_UpArrow ) ,
@ "GBDown" : @ ( kVK_DownArrow ) ,
2016-04-13 22:43:16 +03:00
2017-01-24 21:00:56 +02:00
@ "GBA" : @ ( kVK_ANSI _X ) ,
@ "GBB" : @ ( kVK_ANSI _Z ) ,
@ "GBSelect" : @ ( kVK_Delete ) ,
@ "GBStart" : @ ( kVK_Return ) ,
2016-04-13 22:43:16 +03:00
2017-01-24 21:00:56 +02:00
@ "GBTurbo" : @ ( kVK_Space ) ,
2018-02-10 14:42:14 +02:00
@ "GBRewind" : @ ( kVK_Tab ) ,
2018-02-10 23:30:30 +02:00
@ "GBSlow-Motion" : @ ( kVK_Shift ) ,
2016-04-28 23:07:05 +03:00
@ "GBFilter" : @ "NearestNeighbor" ,
2017-10-12 17:22:22 +03:00
@ "GBColorCorrection" : @ ( GB_COLOR _CORRECTION _EMULATE _HARDWARE ) ,
2018-02-10 14:42:14 +02:00
@ "GBHighpassFilter" : @ ( GB_HIGHPASS _REMOVE _DC _OFFSET ) ,
2018-12-01 17:16:50 +02:00
@ "GBRewindLength" : @ ( 10 ) ,
2020-03-26 20:54:18 +02:00
@ "GBFrameBlendingMode" : @ ( [ defaults boolForKey : @ "DisableFrameBlending" ] ? GB_FRAME _BLENDING _MODE _DISABLED : GB_FRAME _BLENDING _MODE _ACCURATE ) ,
2018-12-01 17:16:50 +02:00
@ "GBDMGModel" : @ ( GB_MODEL _DMG _B ) ,
@ "GBCGBModel" : @ ( GB_MODEL _CGB _E ) ,
@ "GBSGBModel" : @ ( GB_MODEL _SGB2 ) ,
2020-04-29 16:50:31 +03:00
@ "GBRumbleMode" : @ ( GB_RUMBLE _CARTRIDGE _ONLY ) ,
2021-05-21 18:12:29 +03:00
@ "GBVolume" : @ ( 1.0 ) ,
2016-04-13 22:43:16 +03:00
} ] ;
2019-10-19 19:26:04 +03:00
[ JOYController startOnRunLoop : [ NSRunLoop currentRunLoop ] withOptions : @ {
JOYAxes2DEmulateButtonsKey : @ YES ,
JOYHatsEmulateButtonsKey : @ YES ,
} ] ;
2020-05-23 14:50:54 +03:00
2020-08-22 00:56:12 +03:00
if ( [ [ NSUserDefaults standardUserDefaults ] boolForKey : @ "GBNotificationsUsed" ] ) {
[ NSUserNotificationCenter defaultUserNotificationCenter ] . delegate = self ;
}
2021-04-25 22:28:24 +03:00
[ self askAutoUpdates ] ;
if ( [ [ NSUserDefaults standardUserDefaults ] boolForKey : @ "GBAutoUpdatesEnabled" ] ) {
[ self checkForUpdates ] ;
}
if ( [ [ NSProcessInfo processInfo ] . arguments containsObject : @ "--update-launch" ] ) {
[ NSApp activateIgnoringOtherApps : YES ] ;
}
2016-04-13 22:43:16 +03:00
}
2016-03-30 23:07:55 +03:00
2017-01-24 21:00:56 +02:00
- ( IBAction ) toggleDeveloperMode : ( id ) sender
{
2016-04-08 13:54:34 +03:00
NSUserDefaults * defaults = [ NSUserDefaults standardUserDefaults ] ;
[ defaults setBool : ! [ defaults boolForKey : @ "DeveloperMode" ] forKey : @ "DeveloperMode" ] ;
2016-03-30 23:07:55 +03:00
}
2018-12-01 16:08:59 +02:00
- ( IBAction ) switchPreferencesTab : ( id ) sender
{
for ( NSView * view in preferences_tabs ) {
[ view removeFromSuperview ] ;
}
NSView * tab = preferences_tabs [ [ sender tag ] ] ;
NSRect old = [ _preferencesWindow frame ] ;
NSRect new = [ _preferencesWindow frameRectForContentRect : tab . frame ] ;
new . origin . x = old . origin . x ;
new . origin . y = old . origin . y + ( old . size . height - new . size . height ) ;
[ _preferencesWindow setFrame : new display : YES animate : _preferencesWindow . visible ] ;
[ _preferencesWindow . contentView addSubview : tab ] ;
}
2016-04-08 13:54:34 +03:00
- ( BOOL ) validateMenuItem : ( NSMenuItem * ) anItem
{
if ( [ anItem action ] = = @ selector ( toggleDeveloperMode : ) ) {
2018-12-01 16:08:59 +02:00
[ ( NSMenuItem * ) anItem setState : [ [ NSUserDefaults standardUserDefaults ] boolForKey : @ "DeveloperMode" ] ] ;
2016-04-08 13:54:34 +03:00
}
2020-11-13 23:07:35 +02:00
if ( anItem = = self . linkCableMenuItem ) {
return [ [ NSDocumentController sharedDocumentController ] documents ] . count > 1 ;
}
2016-04-13 22:43:16 +03:00
return true ;
2016-03-30 23:07:55 +03:00
}
2020-11-13 23:07:35 +02:00
- ( void ) menuNeedsUpdate : ( NSMenu * ) menu
{
NSMutableArray * items = [ NSMutableArray array ] ;
NSDocument * currentDocument = [ [ NSDocumentController sharedDocumentController ] currentDocument ] ;
for ( NSDocument * document in [ [ NSDocumentController sharedDocumentController ] documents ] ) {
if ( document = = currentDocument ) continue ;
NSMenuItem * item = [ [ NSMenuItem alloc ] initWithTitle : document . displayName action : @ selector ( connectLinkCable : ) keyEquivalent : @ "" ] ;
item . representedObject = document ;
item . image = [ [ NSWorkspace sharedWorkspace ] iconForFile : document . fileURL . path ] ;
[ item . image setSize : NSMakeSize ( 16 , 16 ) ] ;
[ items addObject : item ] ;
}
menu . itemArray = items ;
}
2016-04-13 22:43:16 +03:00
- ( IBAction ) showPreferences : ( id ) sender
{
NSArray * objects ;
if ( ! _preferencesWindow ) {
[ [ NSBundle mainBundle ] loadNibNamed : @ "Preferences" owner : self topLevelObjects : & objects ] ;
2018-12-01 16:08:59 +02:00
NSToolbarItem * first_toolbar _item = [ _preferencesWindow . toolbar . items firstObject ] ;
_preferencesWindow . toolbar . selectedItemIdentifier = [ first_toolbar _item itemIdentifier ] ;
2021-04-25 22:28:24 +03:00
preferences_tabs = @ [ self . emulationTab , self . graphicsTab , self . audioTab , self . controlsTab , self . updatesTab ] ;
2018-12-01 16:08:59 +02:00
[ self switchPreferencesTab : first_toolbar _item ] ;
[ _preferencesWindow center ] ;
2021-04-25 22:28:24 +03:00
# ifndef UPDATE_SUPPORT
[ _preferencesWindow . toolbar removeItemAtIndex : 4 ] ;
# endif
2016-04-13 22:43:16 +03:00
}
[ _preferencesWindow makeKeyAndOrderFront : self ] ;
}
2016-09-10 19:46:42 +03:00
2016-10-02 00:10:09 +03:00
- ( BOOL ) applicationOpenUntitledFile : ( NSApplication * ) sender
2016-09-10 19:46:42 +03:00
{
2021-04-25 22:28:24 +03:00
[ self askAutoUpdates ] ;
2021-02-25 17:12:01 +02:00
/ * Bring an existing panel to the foreground * /
for ( NSWindow * window in [ [ NSApplication sharedApplication ] windows ] ) {
if ( [ window isKindOfClass : [ NSOpenPanel class ] ] ) {
[ ( NSOpenPanel * ) window makeKeyAndOrderFront : nil ] ;
return true ;
}
}
2016-10-02 00:10:09 +03:00
[ [ NSDocumentController sharedDocumentController ] openDocument : self ] ;
2021-02-25 17:12:01 +02:00
return true ;
2016-09-10 19:46:42 +03:00
}
2016-10-02 00:10:09 +03:00
2020-05-23 14:50:54 +03:00
- ( void ) userNotificationCenter : ( NSUserNotificationCenter * ) center didActivateNotification : ( NSUserNotification * ) notification
{
[ [ NSDocumentController sharedDocumentController ] openDocumentWithContentsOfFile : notification . identifier display : YES ] ;
}
2020-11-13 23:07:35 +02:00
2021-04-25 22:28:24 +03:00
- ( void ) updateFound
{
[ [ [ NSURLSession sharedSession ] dataTaskWithURL : [ NSURL URLWithString : @ UPDATE_SERVER "/raw_changes" ] completionHandler : ^ ( NSData * data , NSURLResponse * response , NSError * error ) {
NSColor * linkColor = [ NSColor colorWithRed : 0.125 green : 0.325 blue : 1.0 alpha : 1.0 ] ;
if ( @ available ( macOS 10.10 , * ) ) {
linkColor = [ NSColor linkColor ] ;
}
NSString * changes = [ [ NSString alloc ] initWithData : data encoding : NSUTF8StringEncoding ] ;
2021-05-30 20:55:04 +03:00
NSRange cutoffRange = [ changes rangeOfString : @ "<!--(" GB_VERSION ")-->" ] ;
2021-04-25 22:28:24 +03:00
if ( cutoffRange . location ! = NSNotFound ) {
changes = [ changes substringToIndex : cutoffRange . location ] ;
}
NSString * html = [ NSString stringWithFormat : @ "<!DOCTYPE html><html><head><title></title>"
"<style>html {background-color:transparent; color: #%06x; line-height:1.5} a:link, a:visited{color:#%06x; text-decoration:none}</style>"
"</head><body>%@</body></html>" ,
color_to _int ( [ NSColor textColor ] ) ,
color_to _int ( linkColor ) ,
changes ] ;
if ( [ ( NSHTTPURLResponse * ) response statusCode ] = = 200 ) {
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
NSArray * objects ;
[ [ NSBundle mainBundle ] loadNibNamed : @ "UpdateWindow" owner : self topLevelObjects : & objects ] ;
self . updateChanges . preferences . standardFontFamily = [ NSFont systemFontOfSize : 0 ] . familyName ;
self . updateChanges . preferences . fixedFontFamily = @ "Menlo" ;
self . updateChanges . drawsBackground = false ;
[ self . updateChanges . mainFrame loadHTMLString : html baseURL : nil ] ;
} ) ;
}
} ] resume ] ;
}
- ( NSArray * ) webView : ( WebView * ) sender contextMenuItemsForElement : ( NSDictionary * ) element defaultMenuItems : ( NSArray * ) defaultMenuItems
{
// Disable reload context menu
if ( [ defaultMenuItems count ] <= 2 ) {
return nil ;
}
return defaultMenuItems ;
}
- ( void ) webView : ( WebView * ) sender didFinishLoadForFrame : ( WebFrame * ) frame
{
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
sender . mainFrame . frameView . documentView . enclosingScrollView . drawsBackground = true ;
sender . mainFrame . frameView . documentView . enclosingScrollView . backgroundColor = [ NSColor textBackgroundColor ] ;
sender . policyDelegate = self ;
[ self . updateWindow center ] ;
[ self . updateWindow makeKeyAndOrderFront : nil ] ;
} ) ;
}
- ( void ) webView : ( WebView * ) webView decidePolicyForNavigationAction : ( NSDictionary * ) actionInformation request : ( NSURLRequest * ) request frame : ( WebFrame * ) frame decisionListener : ( id < WebPolicyDecisionListener > ) listener
{
[ listener ignore ] ;
[ [ NSWorkspace sharedWorkspace ] openURL : [ request URL ] ] ;
}
- ( void ) checkForUpdates
{
# ifdef UPDATE_SUPPORT
[ [ [ NSURLSession sharedSession ] dataTaskWithURL : [ NSURL URLWithString : @ UPDATE_SERVER "/latest_version" ] completionHandler : ^ ( NSData * data , NSURLResponse * response , NSError * error ) {
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
[ self . updatesSpinner stopAnimation : nil ] ;
[ self . updatesButton setEnabled : YES ] ;
} ) ;
if ( [ ( NSHTTPURLResponse * ) response statusCode ] = = 200 ) {
NSString * string = [ [ NSString alloc ] initWithData : data encoding : NSUTF8StringEncoding ] ;
NSArray < NSString * > * components = [ string componentsSeparatedByString : @ "|" ] ;
if ( components . count ! = 2 ) return ;
_lastVersion = components [ 0 ] ;
_updateURL = components [ 1 ] ;
2021-05-30 20:55:04 +03:00
if ( ! [ @ GB_VERSION isEqualToString : _lastVersion ] &&
2021-04-25 22:28:24 +03:00
! [ [ [ NSUserDefaults standardUserDefaults ] stringForKey : @ "GBSkippedVersion" ] isEqualToString : _lastVersion ] ) {
[ self updateFound ] ;
}
}
} ] resume ] ;
# endif
}
- ( IBAction ) userCheckForUpdates : ( id ) sender
{
if ( self . updateWindow ) {
[ self . updateWindow makeKeyAndOrderFront : sender ] ;
}
else {
[ [ NSUserDefaults standardUserDefaults ] setObject : nil forKey : @ "GBSkippedVersion" ] ;
[ self checkForUpdates ] ;
[ sender setEnabled : false ] ;
[ self . updatesSpinner startAnimation : sender ] ;
}
}
- ( void ) askAutoUpdates
{
# ifdef UPDATE_SUPPORT
if ( ! [ [ NSUserDefaults standardUserDefaults ] boolForKey : @ "GBAskedAutoUpdates" ] ) {
NSAlert * alert = [ [ NSAlert alloc ] init ] ;
alert . messageText = @ "Should SameBoy check for updates when launched?" ;
alert . informativeText = @ "SameBoy is frequently updated with new features, accuracy improvements, and bug fixes. This setting can always be changed in the preferences window." ;
[ alert addButtonWithTitle : @ "Check on Launch" ] ;
[ alert addButtonWithTitle : @ "Don't Check on Launch" ] ;
[ [ NSUserDefaults standardUserDefaults ] setBool : [ alert runModal ] = = NSAlertFirstButtonReturn forKey : @ "GBAutoUpdatesEnabled" ] ;
[ [ NSUserDefaults standardUserDefaults ] setBool : true forKey : @ "GBAskedAutoUpdates" ] ;
}
# endif
}
- ( IBAction ) skipVersion : ( id ) sender
{
[ [ NSUserDefaults standardUserDefaults ] setObject : _lastVersion forKey : @ "GBSkippedVersion" ] ;
[ self . updateWindow performClose : sender ] ;
}
- ( IBAction ) installUpdate : ( id ) sender
{
[ self . updateProgressSpinner startAnimation : nil ] ;
self . updateProgressButton . title = @ "Cancel" ;
self . updateProgressButton . enabled = true ;
self . updateProgressLabel . stringValue = @ "Downloading update..." ;
_updateState = UPDATE_DOWNLOADING ;
_updateTask = [ [ NSURLSession sharedSession ] downloadTaskWithURL : [ NSURL URLWithString : _updateURL ] completionHandler : ^ ( NSURL * location , NSURLResponse * response , NSError * error ) {
_updateTask = nil ;
dispatch_sync ( dispatch_get _main _queue ( ) , ^ {
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Extracting update..." ;
_updateState = UPDATE_EXTRACTING ;
} ) ;
_downloadDirectory = [ [ [ NSFileManager defaultManager ] URLForDirectory : NSItemReplacementDirectory
inDomain : NSUserDomainMask
appropriateForURL : [ [ NSBundle mainBundle ] bundleURL ]
create : YES
error : nil ] path ] ;
NSTask * unzipTask ;
if ( ! _downloadDirectory ) {
dispatch_sync ( dispatch_get _main _queue ( ) , ^ {
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Failed to extract update." ;
_updateState = UPDATE_FAILED ;
self . updateProgressButton . title = @ "Close" ;
self . updateProgressButton . enabled = true ;
[ self . updateProgressSpinner stopAnimation : nil ] ;
} ) ;
}
unzipTask = [ [ NSTask alloc ] init ] ;
unzipTask . launchPath = @ "/usr/bin/unzip" ;
unzipTask . arguments = @ [ location . path , @ "-d" , _downloadDirectory ] ;
[ unzipTask launch ] ;
[ unzipTask waitUntilExit ] ;
if ( unzipTask . terminationStatus ! = 0 || unzipTask . terminationReason ! = NSTaskTerminationReasonExit ) {
[ [ NSFileManager defaultManager ] removeItemAtPath : _downloadDirectory error : nil ] ;
dispatch_sync ( dispatch_get _main _queue ( ) , ^ {
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Failed to extract update." ;
_updateState = UPDATE_FAILED ;
self . updateProgressButton . title = @ "Close" ;
self . updateProgressButton . enabled = true ;
[ self . updateProgressSpinner stopAnimation : nil ] ;
} ) ;
return ;
}
dispatch_sync ( dispatch_get _main _queue ( ) , ^ {
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Update ready, save your game progress and click Install." ;
_updateState = UPDATE_WAIT _INSTALL ;
self . updateProgressButton . title = @ "Install" ;
self . updateProgressButton . enabled = true ;
[ self . updateProgressSpinner stopAnimation : nil ] ;
} ) ;
} ] ;
[ _updateTask resume ] ;
self . updateProgressWindow . preventsApplicationTerminationWhenModal = false ;
[ self . updateWindow beginSheet : self . updateProgressWindow completionHandler : ^ ( NSModalResponse returnCode ) {
[ self . updateWindow close ] ;
} ] ;
}
- ( void ) performUpgrade
{
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Instaling update..." ;
_updateState = UPDATE_INSTALLING ;
self . updateProgressButton . enabled = false ;
[ self . updateProgressSpinner startAnimation : nil ] ;
dispatch_async ( dispatch_get _global _queue ( DISPATCH_QUEUE _PRIORITY _DEFAULT , 0 ) , ^ {
NSString * executablePath = [ [ NSBundle mainBundle ] executablePath ] ;
NSString * contentsPath = [ [ [ NSBundle mainBundle ] bundlePath ] stringByAppendingPathComponent : @ "Contents" ] ;
NSString * contentsTempPath = [ [ [ NSBundle mainBundle ] bundlePath ] stringByAppendingPathComponent : @ "TempContents" ] ;
NSString * updateContentsPath = [ _downloadDirectory stringByAppendingPathComponent : @ "SameBoy.app/Contents" ] ;
NSError * error = nil ;
[ [ NSFileManager defaultManager ] moveItemAtPath : contentsPath toPath : contentsTempPath error : & error ] ;
if ( error ) {
[ [ NSFileManager defaultManager ] removeItemAtPath : _downloadDirectory error : nil ] ;
_downloadDirectory = nil ;
dispatch_sync ( dispatch_get _main _queue ( ) , ^ {
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Failed to install update." ;
_updateState = UPDATE_FAILED ;
self . updateProgressButton . title = @ "Close" ;
self . updateProgressButton . enabled = true ;
[ self . updateProgressSpinner stopAnimation : nil ] ;
} ) ;
return ;
}
[ [ NSFileManager defaultManager ] moveItemAtPath : updateContentsPath toPath : contentsPath error : & error ] ;
if ( error ) {
[ [ NSFileManager defaultManager ] moveItemAtPath : contentsTempPath toPath : contentsPath error : nil ] ;
[ [ NSFileManager defaultManager ] removeItemAtPath : _downloadDirectory error : nil ] ;
_downloadDirectory = nil ;
dispatch_sync ( dispatch_get _main _queue ( ) , ^ {
self . updateProgressButton . enabled = false ;
self . updateProgressLabel . stringValue = @ "Failed to install update." ;
_updateState = UPDATE_FAILED ;
self . updateProgressButton . title = @ "Close" ;
self . updateProgressButton . enabled = true ;
[ self . updateProgressSpinner stopAnimation : nil ] ;
} ) ;
return ;
}
[ [ NSFileManager defaultManager ] removeItemAtPath : _downloadDirectory error : nil ] ;
[ [ NSFileManager defaultManager ] removeItemAtPath : contentsTempPath error : nil ] ;
_downloadDirectory = nil ;
atexit_b ( ^ {
execl ( executablePath . UTF8String , executablePath . UTF8String , "--update-launch" , NULL ) ;
} ) ;
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
[ NSApp terminate : nil ] ;
} ) ;
} ) ;
}
- ( IBAction ) updateAction : ( id ) sender
{
switch ( _updateState ) {
case UPDATE_DOWNLOADING :
[ _updateTask cancelByProducingResumeData : nil ] ;
_updateTask = nil ;
[ self . updateProgressWindow close ] ;
break ;
case UPDATE_WAIT _INSTALL :
[ self performUpgrade ] ;
break ;
case UPDATE_EXTRACTING :
case UPDATE_INSTALLING :
break ;
case UPDATE_FAILED :
[ self . updateProgressWindow close ] ;
break ;
}
}
- ( void ) dealloc
{
if ( _downloadDirectory ) {
[ [ NSFileManager defaultManager ] removeItemAtPath : _downloadDirectory error : nil ] ;
}
}
2020-11-13 23:07:35 +02:00
- ( IBAction ) nop : ( id ) sender
{
}
2016-03-30 23:07:55 +03:00
@ end