Search
lxdream.org :: lxdream/src/cocoaui/cocoaui.c
lxdream 0.9.1
released Jun 29
Download Now
filename src/cocoaui/cocoaui.c
changeset 685:fa1589b42be7
prev681:1755a126b109
next686:afd1bd3f1acc
author nkeynes
date Tue Jun 03 11:16:51 2008 +0000 (15 years ago)
permissions -rw-r--r--
last change Add alert dialog for errors
Handle the command line when launched from finder
file annotate diff log raw
nkeynes@681
     1
/**
nkeynes@681
     2
 * $Id$
nkeynes@681
     3
 *
nkeynes@681
     4
 * Core Cocoa-based user interface
nkeynes@681
     5
 *
nkeynes@681
     6
 * Copyright (c) 2005 Nathan Keynes.
nkeynes@681
     7
 *
nkeynes@681
     8
 * This program is free software; you can redistribute it and/or modify
nkeynes@681
     9
 * it under the terms of the GNU General Public License as published by
nkeynes@681
    10
 * the Free Software Foundation; either version 2 of the License, or
nkeynes@681
    11
 * (at your option) any later version.
nkeynes@681
    12
 *
nkeynes@681
    13
 * This program is distributed in the hope that it will be useful,
nkeynes@681
    14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
nkeynes@681
    15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
nkeynes@681
    16
 * GNU General Public License for more details.
nkeynes@681
    17
 */
nkeynes@681
    18
nkeynes@681
    19
#include <AppKit/AppKit.h>
nkeynes@681
    20
#include <stdio.h>
nkeynes@681
    21
#include <stdlib.h>
nkeynes@685
    22
#include <string.h>
nkeynes@681
    23
#include <sys/time.h>
nkeynes@681
    24
#include "lxdream.h"
nkeynes@681
    25
#include "dreamcast.h"
nkeynes@681
    26
#include "dream.h"
nkeynes@681
    27
#include "gui.h"
nkeynes@681
    28
#include "cocoaui/cocoaui.h"
nkeynes@681
    29
nkeynes@681
    30
void cocoa_gui_update( void );
nkeynes@681
    31
void cocoa_gui_start( void );
nkeynes@681
    32
void cocoa_gui_stop( void );
nkeynes@681
    33
void cocoa_gui_run_later( void );
nkeynes@681
    34
uint32_t cocoa_gui_run_slice( uint32_t nanosecs );
nkeynes@681
    35
nkeynes@681
    36
struct dreamcast_module cocoa_gui_module = { "gui", NULL,
nkeynes@681
    37
                  cocoa_gui_update, 
nkeynes@681
    38
                  cocoa_gui_start, 
nkeynes@681
    39
                  cocoa_gui_run_slice, 
nkeynes@681
    40
                  cocoa_gui_stop, 
nkeynes@681
    41
                  NULL, NULL };
nkeynes@681
    42
nkeynes@681
    43
/**
nkeynes@681
    44
 * Count of running nanoseconds - used to cut back on the GUI runtime
nkeynes@681
    45
 */
nkeynes@681
    46
static uint32_t cocoa_gui_nanos = 0;
nkeynes@681
    47
static uint32_t cocoa_gui_ticks = 0;
nkeynes@681
    48
static struct timeval cocoa_gui_lasttv;
nkeynes@681
    49
static BOOL cocoa_gui_autorun = NO;
nkeynes@681
    50
nkeynes@681
    51
@interface NSApplication (PrivateAdditions)
nkeynes@681
    52
- (void) setAppleMenu:(NSMenu *)aMenu;
nkeynes@681
    53
@end
nkeynes@681
    54
nkeynes@681
    55
static void cocoa_gui_create_menu(void)
nkeynes@681
    56
{
nkeynes@681
    57
    NSMenu *appleMenu, *services;
nkeynes@681
    58
    NSMenuItem *menuItem;
nkeynes@681
    59
    NSString *title;
nkeynes@681
    60
    NSString *appName;
nkeynes@681
    61
    
nkeynes@681
    62
    appName = @"Lxdream";
nkeynes@681
    63
    appleMenu = [[NSMenu alloc] initWithTitle:@""];
nkeynes@681
    64
nkeynes@681
    65
    /* Add menu items */
nkeynes@681
    66
    title = [@"About " stringByAppendingString:appName];
nkeynes@681
    67
    [appleMenu addItemWithTitle:title action:@selector(orderFrontStandardAboutPanel:) keyEquivalent:@""];
nkeynes@681
    68
    [appleMenu addItem:[NSMenuItem separatorItem]];
nkeynes@681
    69
nkeynes@681
    70
    // Services Menu
nkeynes@681
    71
    services = [[[NSMenu alloc] init] autorelease];
nkeynes@681
    72
    [appleMenu addItemWithTitle:@"Services" action:nil keyEquivalent:@""];
nkeynes@681
    73
    [appleMenu setSubmenu: services forItem: [appleMenu itemWithTitle: @"Services"]];
nkeynes@681
    74
nkeynes@681
    75
    // Hide AppName
nkeynes@681
    76
    title = [@"Hide " stringByAppendingString:appName];
nkeynes@681
    77
    [appleMenu addItemWithTitle:title action:@selector(hide:) keyEquivalent:@"h"];
nkeynes@681
    78
nkeynes@681
    79
    // Hide Others
nkeynes@681
    80
    menuItem = (NSMenuItem *)[appleMenu addItemWithTitle:@"Hide Others" 
nkeynes@681
    81
                                                  action:@selector(hideOtherApplications:) 
nkeynes@681
    82
                                           keyEquivalent:@"h"];
nkeynes@681
    83
    [menuItem setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
nkeynes@681
    84
nkeynes@681
    85
    // Show All
nkeynes@681
    86
    [appleMenu addItemWithTitle:@"Show All" action:@selector(unhideAllApplications:) keyEquivalent:@""];
nkeynes@681
    87
    [appleMenu addItem:[NSMenuItem separatorItem]];
nkeynes@681
    88
nkeynes@681
    89
    // Quit AppName
nkeynes@681
    90
    title = [@"Quit " stringByAppendingString:appName];
nkeynes@681
    91
    [appleMenu addItemWithTitle:title action:@selector(terminate:) keyEquivalent:@"q"];
nkeynes@681
    92
nkeynes@681
    93
    /* Put menu into the menubar */
nkeynes@681
    94
    menuItem = [[NSMenuItem alloc] initWithTitle:@"" action:nil  keyEquivalent:@""];
nkeynes@681
    95
    [menuItem setSubmenu: appleMenu];
nkeynes@681
    96
    NSMenu *menu = [NSMenu new];
nkeynes@681
    97
    [menu addItem: menuItem];
nkeynes@681
    98
    
nkeynes@681
    99
    NSMenu *fileMenu = [[NSMenu alloc] initWithTitle: NS_("File")];
nkeynes@681
   100
    [fileMenu addItemWithTitle: NS_("Load Binary") action: @selector(load_binary_action:) keyEquivalent: @"b"];
nkeynes@681
   101
    [fileMenu addItemWithTitle: NS_("GD-Rom") action: @selector(mount_action:) keyEquivalent: @"g"];
nkeynes@681
   102
    [fileMenu addItem: [NSMenuItem separatorItem]];
nkeynes@681
   103
    [[fileMenu addItemWithTitle: NS_("Reset") action: @selector(reset_action:) keyEquivalent: @"r"]
nkeynes@681
   104
      setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
nkeynes@681
   105
    [fileMenu addItemWithTitle: NS_("Pause") action: @selector(pause_action:) keyEquivalent: @"p"];
nkeynes@681
   106
    [fileMenu addItemWithTitle: NS_("Resume") action: @selector(run_action:) keyEquivalent: @"r"];
nkeynes@681
   107
    [fileMenu addItem: [NSMenuItem separatorItem]];
nkeynes@681
   108
    [fileMenu addItemWithTitle: NS_("Load State") action: @selector(load_action:) keyEquivalent: @"o"];
nkeynes@681
   109
    [fileMenu addItemWithTitle: NS_("Save State") action: @selector(save_action:) keyEquivalent: @"s"];
nkeynes@681
   110
    
nkeynes@681
   111
    menuItem = [[NSMenuItem alloc] initWithTitle:NS_("File") action: nil keyEquivalent: @""];
nkeynes@681
   112
    [menuItem setSubmenu: fileMenu];
nkeynes@681
   113
    [menu addItem: menuItem];
nkeynes@681
   114
    
nkeynes@681
   115
    /* Tell the application object that this is now the application menu */
nkeynes@681
   116
    [NSApp setMainMenu: menu];
nkeynes@681
   117
    [NSApp setAppleMenu: appleMenu];
nkeynes@681
   118
    [NSApp setServicesMenu: services];
nkeynes@681
   119
nkeynes@681
   120
    /* Finally give up our references to the objects */
nkeynes@681
   121
    [appleMenu release];
nkeynes@681
   122
    [menuItem release];
nkeynes@681
   123
    [menu release];
nkeynes@681
   124
}
nkeynes@681
   125
nkeynes@681
   126
@interface LxdreamDelegate : NSObject
nkeynes@681
   127
@end
nkeynes@681
   128
nkeynes@681
   129
@implementation LxdreamDelegate
nkeynes@681
   130
- (void)windowWillClose: (NSNotification *)notice
nkeynes@681
   131
{
nkeynes@681
   132
    dreamcast_shutdown();
nkeynes@681
   133
    exit(0);
nkeynes@681
   134
}
nkeynes@681
   135
- (void)windowDidBecomeMain: (NSNotification *)notice
nkeynes@681
   136
{
nkeynes@681
   137
    if( cocoa_gui_autorun ) {
nkeynes@681
   138
        cocoa_gui_autorun = NO;
nkeynes@681
   139
        cocoa_gui_run_later();
nkeynes@681
   140
    }
nkeynes@681
   141
}
nkeynes@681
   142
- (void) load_action: (id)sender
nkeynes@681
   143
{
nkeynes@681
   144
    NSOpenPanel *panel = [NSOpenPanel openPanel];
nkeynes@681
   145
    NSArray *fileTypes = [NSArray arrayWithObject: @"dst"];
nkeynes@681
   146
    int result = [panel runModalForDirectory: NSHomeDirectory() file: nil types: fileTypes];
nkeynes@681
   147
    if( result == NSOKButton && [[panel filenames] count] > 0 ) {
nkeynes@681
   148
        NSString *filename = [[panel filenames] objectAtIndex: 0];
nkeynes@681
   149
        dreamcast_load_state( [filename UTF8String] );
nkeynes@681
   150
    }
nkeynes@681
   151
}
nkeynes@681
   152
- (void) save_action: (id)sender
nkeynes@681
   153
{
nkeynes@681
   154
    NSSavePanel *panel = [NSSavePanel savePanel];
nkeynes@681
   155
    [panel setRequiredFileType: @"dst"];
nkeynes@681
   156
    int result = [panel runModalForDirectory: NSHomeDirectory() file:@""];
nkeynes@681
   157
    if( result == NSOKButton ) {
nkeynes@681
   158
        NSString *filename = [panel filename];
nkeynes@681
   159
        dreamcast_save_state( [filename UTF8String] );
nkeynes@681
   160
    }
nkeynes@681
   161
}
nkeynes@681
   162
- (void) load_binary_action: (id)sender
nkeynes@681
   163
{
nkeynes@681
   164
    NSOpenPanel *panel = [NSOpenPanel openPanel];
nkeynes@681
   165
    int result = [panel runModalForDirectory: NSHomeDirectory() file: nil types: nil];
nkeynes@681
   166
    if( result == NSOKButton && [[panel filenames] count] > 0 ) {
nkeynes@681
   167
        NSString *filename = [[panel filenames] objectAtIndex: 0];
nkeynes@681
   168
        file_load_magic( [filename UTF8String] );
nkeynes@681
   169
    }
nkeynes@681
   170
}
nkeynes@681
   171
- (void) mount_action: (id)sender
nkeynes@681
   172
{
nkeynes@681
   173
    NSOpenPanel *panel = [NSOpenPanel openPanel];
nkeynes@681
   174
    int result = [panel runModalForDirectory: NSHomeDirectory() file: nil types: nil];
nkeynes@681
   175
    if( result == NSOKButton && [[panel filenames] count] > 0 ) {
nkeynes@681
   176
        NSString *filename = [[panel filenames] objectAtIndex: 0];
nkeynes@681
   177
        gdrom_mount_image( [filename UTF8String] );
nkeynes@681
   178
    }
nkeynes@681
   179
}
nkeynes@681
   180
- (void) pause_action: (id)sender
nkeynes@681
   181
{
nkeynes@681
   182
    dreamcast_stop();
nkeynes@681
   183
}
nkeynes@681
   184
nkeynes@681
   185
- (void) reset_action: (id)sender
nkeynes@681
   186
{
nkeynes@681
   187
    dreamcast_reset();
nkeynes@681
   188
}
nkeynes@681
   189
- (void) run_action: (id)sender
nkeynes@681
   190
{
nkeynes@681
   191
    cocoa_gui_run_later();
nkeynes@681
   192
}
nkeynes@681
   193
- (void) run_immediate
nkeynes@681
   194
{
nkeynes@681
   195
    dreamcast_run();
nkeynes@681
   196
}
nkeynes@681
   197
@end
nkeynes@681
   198
nkeynes@681
   199
nkeynes@681
   200
gboolean gui_parse_cmdline( int *argc, char **argv[] )
nkeynes@681
   201
{
nkeynes@685
   202
    /* If started from the finder, the first (and only) arg will look something like 
nkeynes@685
   203
    * -psn_0_... - we want to remove this so that lxdream doesn't try to process it 
nkeynes@685
   204
    * normally
nkeynes@685
   205
    */
nkeynes@685
   206
    if( *argc == 2 && strncmp((*argv)[1], "-psn_", 5) == 0 ) {
nkeynes@685
   207
        *argc = 1;
nkeynes@685
   208
    }
nkeynes@685
   209
    return TRUE;
nkeynes@681
   210
}
nkeynes@681
   211
nkeynes@681
   212
gboolean gui_init( gboolean withDebug )
nkeynes@681
   213
{
nkeynes@681
   214
    dreamcast_register_module( &cocoa_gui_module );
nkeynes@681
   215
    
nkeynes@681
   216
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
nkeynes@681
   217
    [NSApplication sharedApplication];
nkeynes@681
   218
    LxdreamDelegate *delegate = [[LxdreamDelegate alloc] init];
nkeynes@681
   219
    [NSApp setDelegate: delegate];
nkeynes@681
   220
    cocoa_gui_create_menu();
nkeynes@681
   221
    NSWindow *window = cocoa_gui_create_main_window();
nkeynes@681
   222
    [window makeKeyAndOrderFront: nil];
nkeynes@681
   223
    [NSApp activateIgnoringOtherApps: YES];   
nkeynes@681
   224
    [pool release];
nkeynes@681
   225
}
nkeynes@681
   226
nkeynes@681
   227
void gui_main_loop( gboolean run )
nkeynes@681
   228
{
nkeynes@681
   229
    if( run ) {
nkeynes@681
   230
        cocoa_gui_autorun = YES;
nkeynes@681
   231
        /*
nkeynes@681
   232
             */
nkeynes@681
   233
    }
nkeynes@681
   234
	[NSApp run];
nkeynes@681
   235
}
nkeynes@681
   236
nkeynes@681
   237
void gui_update_state(void)
nkeynes@681
   238
{
nkeynes@681
   239
    cocoa_gui_update();
nkeynes@681
   240
}
nkeynes@681
   241
nkeynes@681
   242
gboolean gui_error_dialog( const char *msg, ... )
nkeynes@681
   243
{
nkeynes@685
   244
    NSString *error_string;
nkeynes@685
   245
    
nkeynes@685
   246
    va_list args;
nkeynes@685
   247
    va_start(args, msg);
nkeynes@685
   248
    error_string = [[NSString alloc] initWithFormat: [NSString stringWithCString: msg] arguments: args];
nkeynes@685
   249
    NSRunAlertPanel(@"Error in lxdream", error_string, nil, nil, nil);
nkeynes@685
   250
    va_end(args);
nkeynes@681
   251
}
nkeynes@681
   252
nkeynes@681
   253
void gui_update_io_activity( io_activity_type io, gboolean active )
nkeynes@681
   254
{
nkeynes@681
   255
nkeynes@681
   256
}
nkeynes@681
   257
nkeynes@681
   258
nkeynes@681
   259
uint32_t cocoa_gui_run_slice( uint32_t nanosecs )
nkeynes@681
   260
{
nkeynes@681
   261
    NSEvent *event;
nkeynes@681
   262
    NSAutoreleasePool *pool;
nkeynes@681
   263
nkeynes@681
   264
    cocoa_gui_nanos += nanosecs;
nkeynes@681
   265
    if( cocoa_gui_nanos > GUI_TICK_PERIOD ) { /* 10 ms */
nkeynes@681
   266
        cocoa_gui_nanos -= GUI_TICK_PERIOD;
nkeynes@681
   267
        cocoa_gui_ticks ++;
nkeynes@681
   268
        uint32_t current_period = cocoa_gui_ticks * GUI_TICK_PERIOD;
nkeynes@681
   269
nkeynes@681
   270
        // Run the event loop
nkeynes@681
   271
        pool = [NSAutoreleasePool new];
nkeynes@681
   272
        while( (event = [NSApp nextEventMatchingMask: NSAnyEventMask untilDate: nil 
nkeynes@681
   273
                         inMode: NSDefaultRunLoopMode dequeue: YES]) != nil ) {
nkeynes@681
   274
            [NSApp sendEvent: event];
nkeynes@681
   275
        }
nkeynes@681
   276
        [pool release];
nkeynes@681
   277
nkeynes@681
   278
        struct timeval tv;
nkeynes@681
   279
        gettimeofday(&tv,NULL);
nkeynes@681
   280
        uint32_t ns = ((tv.tv_sec - cocoa_gui_lasttv.tv_sec) * 1000000000) + 
nkeynes@681
   281
        (tv.tv_usec - cocoa_gui_lasttv.tv_usec)*1000;
nkeynes@681
   282
        if( (ns * 1.05) < current_period ) {
nkeynes@681
   283
            // We've gotten ahead - sleep for a little bit
nkeynes@681
   284
            struct timespec tv;
nkeynes@681
   285
            tv.tv_sec = 0;
nkeynes@681
   286
            tv.tv_nsec = current_period - ns;
nkeynes@681
   287
            nanosleep(&tv, &tv);
nkeynes@681
   288
        }
nkeynes@681
   289
nkeynes@681
   290
        /* Update the display every 10 ticks (ie 10 times a second) and 
nkeynes@681
   291
         * save the current tv value */
nkeynes@681
   292
        if( cocoa_gui_ticks > 10 ) {
nkeynes@681
   293
            gchar buf[32];
nkeynes@681
   294
            cocoa_gui_ticks -= 10;
nkeynes@681
   295
nkeynes@681
   296
            double speed = (float)( (double)current_period * 100.0 / ns );
nkeynes@681
   297
            cocoa_gui_lasttv.tv_sec = tv.tv_sec;
nkeynes@681
   298
            cocoa_gui_lasttv.tv_usec = tv.tv_usec;
nkeynes@681
   299
            snprintf( buf, 32, _("Running (%2.4f%%)"), speed );
nkeynes@681
   300
            [((LxdreamMainWindow *)[NSApp mainWindow]) setStatusText: buf];
nkeynes@681
   301
nkeynes@681
   302
        }
nkeynes@681
   303
    }
nkeynes@681
   304
    return nanosecs;
nkeynes@681
   305
}
nkeynes@681
   306
nkeynes@681
   307
void cocoa_gui_update( void )
nkeynes@681
   308
{
nkeynes@681
   309
    
nkeynes@681
   310
}
nkeynes@681
   311
nkeynes@681
   312
void cocoa_gui_start( void )
nkeynes@681
   313
{
nkeynes@681
   314
    LxdreamMainWindow *win = (LxdreamMainWindow *)[NSApp mainWindow];
nkeynes@681
   315
    [win setRunning: YES];
nkeynes@681
   316
    cocoa_gui_nanos = 0;
nkeynes@681
   317
    gettimeofday(&cocoa_gui_lasttv,NULL);
nkeynes@681
   318
}
nkeynes@681
   319
nkeynes@681
   320
void cocoa_gui_stop( void )
nkeynes@681
   321
{
nkeynes@681
   322
    LxdreamMainWindow *win = (LxdreamMainWindow *)[NSApp mainWindow];
nkeynes@681
   323
    [win setRunning: NO];
nkeynes@681
   324
}
nkeynes@681
   325
nkeynes@681
   326
/**
nkeynes@681
   327
 * Queue a dreamcast_run() to execute after the currently event(s)
nkeynes@681
   328
 */
nkeynes@681
   329
void cocoa_gui_run_later( void )
nkeynes@681
   330
{
nkeynes@681
   331
    [[NSRunLoop currentRunLoop] performSelector: @selector(run_immediate) 
nkeynes@681
   332
         target: [NSApp delegate] argument: nil order: 1 
nkeynes@681
   333
         modes: [NSArray arrayWithObject: NSDefaultRunLoopMode] ];
nkeynes@681
   334
}
.