2014-04-08 11 views
5

Perché il seguente codice funziona? È un piccolo programma Cocoa che utilizza NSOpenPanel per selezionare un file e aprirlo in Emacs.app. Può essere eseguito dalla riga di comando con la directory di partenza come argomento.Com'è possibile eseguire la classe della GUI Cocoa senza richiamare NSApplication o NSRunLoop

Come viene eseguito NSOpenPanel senza richiamare NSApplication o NSRunLoop? Quali sono le limitazioni su un programma Cocoa che non avvia esplicitamente NSApplication o NSRunLoop? Avrei pensato che uno di questi era: non puoi usare alcun tipo di interfaccia grafica. Forse invocando NSOpenPanel, viene chiamato un codice di fallback che richiama NSRunLoop? Ho inserito i breakpoint su + [allocazione NSApplication] e + [allocazione NSRunLoop] e non sono stati attivati.

main.m: 

#import <Cocoa/Cocoa.h> 

NSString *selectFileWithStartPath(NSString *path) { 
    NSString *answer = nil; 
    NSOpenPanel* panel = [NSOpenPanel openPanel]; 
    panel.allowsMultipleSelection = NO; 
    panel.canChooseFiles = YES; 
    panel.canChooseDirectories = NO; 
    panel.resolvesAliases = YES; 
    if([panel runModalForDirectory:path file:nil] == NSOKButton) 
    answer = [[[panel URLs] objectAtIndex:0] path]; 
    return answer; 
} 

int main(int argc, const char * argv[]) { 
    NSString *startPath = argc > 1 ? [NSString stringWithUTF8String:argv[1]] : @"/Users/Me/Docs"; 
    printf("%s\n", argv[1]); 
    BOOL isDir; 
    if([[NSFileManager defaultManager] fileExistsAtPath:startPath isDirectory:&isDir] && isDir) { 
    system([[NSString stringWithFormat:@"find %@ -name \\*~ -exec rm {} \\;", startPath] UTF8String]); 
    NSString *file = selectFileWithStartPath(startPath); 
    if(file) [[NSWorkspace sharedWorkspace] openFile:file withApplication:@"Emacs.app"]; 
    } 
} 

risposta

5

runModalForDirectory:file:types: crea ed esegue il proprio ciclo di eventi. Dalla documentazione:

visualizza il pannello e inizia un ciclo evento modale che è terminato quando l'utente fa clic OK o Annulla.

Si può vedere che anche se si mette in pausa il programma, mentre la finestra di dialogo "Apri" è attivo e stampare il backtrace dello stack nella console debugger:

frame #0: 0x00007fff8b855a1a libsystem_kernel.dylib`mach_msg_trap + 10 
frame #1: 0x00007fff8b854d18 libsystem_kernel.dylib`mach_msg + 64 
frame #2: 0x00007fff8549f155 CoreFoundation`__CFRunLoopServiceMachPort + 181 
frame #3: 0x00007fff8549e779 CoreFoundation`__CFRunLoopRun + 1161 
frame #4: 0x00007fff8549e0b5 CoreFoundation`CFRunLoopRunSpecific + 309 
frame #5: 0x00007fff88381a0d HIToolbox`RunCurrentEventLoopInMode + 226 
frame #6: 0x00007fff883817b7 HIToolbox`ReceiveNextEventCommon + 479 
frame #7: 0x00007fff883815bc HIToolbox`_BlockUntilNextEventMatchingListInModeWithFilter + 65 
frame #8: 0x00007fff838ca3de AppKit`_DPSNextEvent + 1434 
frame #9: 0x00007fff838c9a2b AppKit`-[NSApplication nextEventMatchingMask:untilDate:inMode:dequeue:] + 122 
frame #10: 0x00007fff83c28e2e AppKit`-[NSApplication _realDoModalLoop:peek:] + 642 
frame #11: 0x00007fff83c2754e AppKit`-[NSApplication runModalForWindow:] + 117 
frame #12: 0x00007fff83ef5d0b AppKit`-[NSSavePanel runModal] + 276 
frame #13: 0x0000000100000c61 xxx`selectFileWithStartPath(path=0x00000001000010b8) + 225 at main.m:18 
frame #14: 0x0000000100000e0d xxx`main(argc=1, argv=0x00007fff5fbff9c0) + 189 at main.m:26 

Come si può vedere nel telaio # 11 , -[NSApplication runModalForWindow:] viene utilizzato per eseguire un ciclo di eventi modale per la finestra di dialogo "Apri".

+0

Sì, capisco. Grazie per la risposta veloce. Una cosa che ho appena capito è che le vecchie chiamate main.m di Apple [NSApplication sharedApplication], non [NSApplication alloc]. Immagino che la chiamata a runModalForWindow possa essere stata preceduta istanziando un singleton [NSApplication sharedApplication] nella libreria Apple. – Colin

Problemi correlati