Search
lxdream.org :: lxdream/src/cocoaui/cocoaui.c
lxdream 0.9.1
released Jun 29
Download Now
filename src/cocoaui/cocoaui.c
changeset 718:047a244936d1
prev693:b16608c4ad1e
next725:4d4018e8eeb8
author nkeynes
date Mon Jun 30 04:58:26 2008 +0000 (12 years ago)
permissions -rw-r--r--
last change Use the configured paths for the dialog boxes by default
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@718
    25
#include "dream.h"
nkeynes@681
    26
#include "dreamcast.h"
nkeynes@718
    27
#include "config.h"
nkeynes@718
    28
#include "display.h"
nkeynes@681
    29
#include "gui.h"
nkeynes@681
    30
#include "cocoaui/cocoaui.h"
nkeynes@681
    31
nkeynes@681
    32
void cocoa_gui_update( void );
nkeynes@681
    33
void cocoa_gui_start( void );
nkeynes@681
    34
void cocoa_gui_stop( void );
nkeynes@681
    35
void cocoa_gui_run_later( void );
nkeynes@681
    36
uint32_t cocoa_gui_run_slice( uint32_t nanosecs );
nkeynes@681
    37
nkeynes@681
    38
struct dreamcast_module cocoa_gui_module = { "gui", NULL,
nkeynes@681
    39
                  cocoa_gui_update, 
nkeynes@681
    40
                  cocoa_gui_start, 
nkeynes@681
    41
                  cocoa_gui_run_slice, 
nkeynes@681
    42
                  cocoa_gui_stop, 
nkeynes@681
    43
                  NULL, NULL };
nkeynes@681
    44
nkeynes@681
    45
/**
nkeynes@681
    46
 * Count of running nanoseconds - used to cut back on the GUI runtime
nkeynes@681
    47
 */
nkeynes@681
    48
static uint32_t cocoa_gui_nanos = 0;
nkeynes@681
    49
static uint32_t cocoa_gui_ticks = 0;
nkeynes@681
    50
static struct timeval cocoa_gui_lasttv;
nkeynes@681
    51
static BOOL cocoa_gui_autorun = NO;
nkeynes@693
    52
static BOOL cocoa_gui_is_running = NO;
nkeynes@681
    53
nkeynes@681
    54
@interface NSApplication (PrivateAdditions)
nkeynes@681
    55
- (void) setAppleMenu:(NSMenu *)aMenu;
nkeynes@681
    56
@end
nkeynes@681
    57
nkeynes@681
    58
static void cocoa_gui_create_menu(void)
nkeynes@681
    59
{
nkeynes@681
    60
    NSMenu *appleMenu, *services;
nkeynes@681
    61
    NSMenuItem *menuItem;
nkeynes@681
    62
    NSString *title;
nkeynes@681
    63
    NSString *appName;
nkeynes@681
    64
    
nkeynes@681
    65
    appName = @"Lxdream";
nkeynes@681
    66
    appleMenu = [[NSMenu alloc] initWithTitle:@""];
nkeynes@681
    67
nkeynes@681
    68
    /* Add menu items */
nkeynes@681
    69
    title = [@"About " stringByAppendingString:appName];
nkeynes@681
    70
    [appleMenu addItemWithTitle:title action:@selector(orderFrontStandardAboutPanel:) keyEquivalent:@""];
nkeynes@681
    71
    [appleMenu addItem:[NSMenuItem separatorItem]];
nkeynes@681
    72
nkeynes@681
    73
    // Services Menu
nkeynes@681
    74
    services = [[[NSMenu alloc] init] autorelease];
nkeynes@681
    75
    [appleMenu addItemWithTitle:@"Services" action:nil keyEquivalent:@""];
nkeynes@681
    76
    [appleMenu setSubmenu: services forItem: [appleMenu itemWithTitle: @"Services"]];
nkeynes@681
    77
nkeynes@681
    78
    // Hide AppName
nkeynes@681
    79
    title = [@"Hide " stringByAppendingString:appName];
nkeynes@681
    80
    [appleMenu addItemWithTitle:title action:@selector(hide:) keyEquivalent:@"h"];
nkeynes@681
    81
nkeynes@681
    82
    // Hide Others
nkeynes@681
    83
    menuItem = (NSMenuItem *)[appleMenu addItemWithTitle:@"Hide Others" 
nkeynes@681
    84
                                                  action:@selector(hideOtherApplications:) 
nkeynes@681
    85
                                           keyEquivalent:@"h"];
nkeynes@681
    86
    [menuItem setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
nkeynes@681
    87
nkeynes@681
    88
    // Show All
nkeynes@681
    89
    [appleMenu addItemWithTitle:@"Show All" action:@selector(unhideAllApplications:) keyEquivalent:@""];
nkeynes@681
    90
    [appleMenu addItem:[NSMenuItem separatorItem]];
nkeynes@681
    91
nkeynes@681
    92
    // Quit AppName
nkeynes@681
    93
    title = [@"Quit " stringByAppendingString:appName];
nkeynes@681
    94
    [appleMenu addItemWithTitle:title action:@selector(terminate:) keyEquivalent:@"q"];
nkeynes@681
    95
nkeynes@681
    96
    /* Put menu into the menubar */
nkeynes@681
    97
    menuItem = [[NSMenuItem alloc] initWithTitle:@"" action:nil  keyEquivalent:@""];
nkeynes@681
    98
    [menuItem setSubmenu: appleMenu];
nkeynes@681
    99
    NSMenu *menu = [NSMenu new];
nkeynes@681
   100
    [menu addItem: menuItem];
nkeynes@681
   101
    
nkeynes@691
   102
    NSMenu *gdromMenu = cocoa_gdrom_menu_new();
nkeynes@691
   103
    
nkeynes@681
   104
    NSMenu *fileMenu = [[NSMenu alloc] initWithTitle: NS_("File")];
nkeynes@681
   105
    [fileMenu addItemWithTitle: NS_("Load Binary") action: @selector(load_binary_action:) keyEquivalent: @"b"];
nkeynes@691
   106
    [[fileMenu addItemWithTitle: NS_("GD-Rom") action: nil keyEquivalent: @""]
nkeynes@691
   107
      setSubmenu: gdromMenu];
nkeynes@681
   108
    [fileMenu addItem: [NSMenuItem separatorItem]];
nkeynes@681
   109
    [[fileMenu addItemWithTitle: NS_("Reset") action: @selector(reset_action:) keyEquivalent: @"r"]
nkeynes@681
   110
      setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
nkeynes@681
   111
    [fileMenu addItemWithTitle: NS_("Pause") action: @selector(pause_action:) keyEquivalent: @"p"];
nkeynes@681
   112
    [fileMenu addItemWithTitle: NS_("Resume") action: @selector(run_action:) keyEquivalent: @"r"];
nkeynes@681
   113
    [fileMenu addItem: [NSMenuItem separatorItem]];
nkeynes@681
   114
    [fileMenu addItemWithTitle: NS_("Load State") action: @selector(load_action:) keyEquivalent: @"o"];
nkeynes@681
   115
    [fileMenu addItemWithTitle: NS_("Save State") action: @selector(save_action:) keyEquivalent: @"s"];
nkeynes@681
   116
    
nkeynes@681
   117
    menuItem = [[NSMenuItem alloc] initWithTitle:NS_("File") action: nil keyEquivalent: @""];
nkeynes@681
   118
    [menuItem setSubmenu: fileMenu];
nkeynes@681
   119
    [menu addItem: menuItem];
nkeynes@681
   120
    
nkeynes@681
   121
    /* Tell the application object that this is now the application menu */
nkeynes@681
   122
    [NSApp setMainMenu: menu];
nkeynes@681
   123
    [NSApp setAppleMenu: appleMenu];
nkeynes@681
   124
    [NSApp setServicesMenu: services];
nkeynes@681
   125
nkeynes@681
   126
    /* Finally give up our references to the objects */
nkeynes@681
   127
    [appleMenu release];
nkeynes@681
   128
    [menuItem release];
nkeynes@681
   129
    [menu release];
nkeynes@681
   130
}
nkeynes@681
   131
nkeynes@681
   132
@interface LxdreamDelegate : NSObject
nkeynes@681
   133
@end
nkeynes@681
   134
nkeynes@681
   135
@implementation LxdreamDelegate
nkeynes@681
   136
- (void)windowWillClose: (NSNotification *)notice
nkeynes@681
   137
{
nkeynes@681
   138
    dreamcast_shutdown();
nkeynes@681
   139
    exit(0);
nkeynes@681
   140
}
nkeynes@681
   141
- (void)windowDidBecomeMain: (NSNotification *)notice
nkeynes@681
   142
{
nkeynes@681
   143
    if( cocoa_gui_autorun ) {
nkeynes@681
   144
        cocoa_gui_autorun = NO;
nkeynes@681
   145
        cocoa_gui_run_later();
nkeynes@681
   146
    }
nkeynes@681
   147
}
nkeynes@681
   148
- (void) load_action: (id)sender
nkeynes@681
   149
{
nkeynes@681
   150
    NSOpenPanel *panel = [NSOpenPanel openPanel];
nkeynes@718
   151
    const gchar *dir = lxdream_get_config_value(CONFIG_SAVE_PATH);
nkeynes@718
   152
    NSString *path = (dir == NULL ? NSHomeDirectory() : [NSString stringWithCString: dir]);
nkeynes@681
   153
    NSArray *fileTypes = [NSArray arrayWithObject: @"dst"];
nkeynes@718
   154
    int result = [panel runModalForDirectory: path file: nil types: fileTypes];
nkeynes@681
   155
    if( result == NSOKButton && [[panel filenames] count] > 0 ) {
nkeynes@681
   156
        NSString *filename = [[panel filenames] objectAtIndex: 0];
nkeynes@681
   157
        dreamcast_load_state( [filename UTF8String] );
nkeynes@681
   158
    }
nkeynes@681
   159
}
nkeynes@681
   160
- (void) save_action: (id)sender
nkeynes@681
   161
{
nkeynes@681
   162
    NSSavePanel *panel = [NSSavePanel savePanel];
nkeynes@718
   163
    const gchar *dir = lxdream_get_config_value(CONFIG_SAVE_PATH);
nkeynes@718
   164
    NSString *path = (dir == NULL ? NSHomeDirectory() : [NSString stringWithCString: dir]);
nkeynes@681
   165
    [panel setRequiredFileType: @"dst"];
nkeynes@718
   166
    int result = [panel runModalForDirectory: path file:@""];
nkeynes@681
   167
    if( result == NSOKButton ) {
nkeynes@681
   168
        NSString *filename = [panel filename];
nkeynes@681
   169
        dreamcast_save_state( [filename UTF8String] );
nkeynes@681
   170
    }
nkeynes@681
   171
}
nkeynes@681
   172
- (void) load_binary_action: (id)sender
nkeynes@681
   173
{
nkeynes@681
   174
    NSOpenPanel *panel = [NSOpenPanel openPanel];
nkeynes@718
   175
    const gchar *dir = lxdream_get_config_value(CONFIG_DEFAULT_PATH);
nkeynes@718
   176
    NSString *path = (dir == NULL ? NSHomeDirectory() : [NSString stringWithCString: dir]);
nkeynes@718
   177
    int result = [panel runModalForDirectory: path file: nil types: nil];
nkeynes@681
   178
    if( result == NSOKButton && [[panel filenames] count] > 0 ) {
nkeynes@681
   179
        NSString *filename = [[panel filenames] objectAtIndex: 0];
nkeynes@681
   180
        file_load_magic( [filename UTF8String] );
nkeynes@681
   181
    }
nkeynes@681
   182
}
nkeynes@681
   183
- (void) mount_action: (id)sender
nkeynes@681
   184
{
nkeynes@681
   185
    NSOpenPanel *panel = [NSOpenPanel openPanel];
nkeynes@718
   186
    const gchar *dir = lxdream_get_config_value(CONFIG_DEFAULT_PATH);
nkeynes@718
   187
    NSString *path = (dir == NULL ? NSHomeDirectory() : [NSString stringWithCString: dir]);
nkeynes@718
   188
    int result = [panel runModalForDirectory: path file: nil types: nil];
nkeynes@681
   189
    if( result == NSOKButton && [[panel filenames] count] > 0 ) {
nkeynes@681
   190
        NSString *filename = [[panel filenames] objectAtIndex: 0];
nkeynes@681
   191
        gdrom_mount_image( [filename UTF8String] );
nkeynes@681
   192
    }
nkeynes@681
   193
}
nkeynes@681
   194
- (void) pause_action: (id)sender
nkeynes@681
   195
{
nkeynes@681
   196
    dreamcast_stop();
nkeynes@681
   197
}
nkeynes@681
   198
nkeynes@681
   199
- (void) reset_action: (id)sender
nkeynes@681
   200
{
nkeynes@681
   201
    dreamcast_reset();
nkeynes@681
   202
}
nkeynes@681
   203
- (void) run_action: (id)sender
nkeynes@681
   204
{
nkeynes@681
   205
    cocoa_gui_run_later();
nkeynes@681
   206
}
nkeynes@681
   207
- (void) run_immediate
nkeynes@681
   208
{
nkeynes@681
   209
    dreamcast_run();
nkeynes@681
   210
}
nkeynes@691
   211
- (void) gdrom_list_action: (id)sender
nkeynes@691
   212
{
nkeynes@691
   213
    gdrom_list_set_selection( [sender tag] );
nkeynes@691
   214
}
nkeynes@681
   215
@end
nkeynes@681
   216
nkeynes@681
   217
nkeynes@681
   218
gboolean gui_parse_cmdline( int *argc, char **argv[] )
nkeynes@681
   219
{
nkeynes@685
   220
    /* If started from the finder, the first (and only) arg will look something like 
nkeynes@685
   221
    * -psn_0_... - we want to remove this so that lxdream doesn't try to process it 
nkeynes@685
   222
    * normally
nkeynes@685
   223
    */
nkeynes@685
   224
    if( *argc == 2 && strncmp((*argv)[1], "-psn_", 5) == 0 ) {
nkeynes@685
   225
        *argc = 1;
nkeynes@685
   226
    }
nkeynes@685
   227
    return TRUE;
nkeynes@681
   228
}
nkeynes@681
   229
nkeynes@681
   230
gboolean gui_init( gboolean withDebug )
nkeynes@681
   231
{
nkeynes@681
   232
    dreamcast_register_module( &cocoa_gui_module );
nkeynes@681
   233
    
nkeynes@681
   234
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
nkeynes@681
   235
    [NSApplication sharedApplication];
nkeynes@718
   236
    
nkeynes@681
   237
    LxdreamDelegate *delegate = [[LxdreamDelegate alloc] init];
nkeynes@681
   238
    [NSApp setDelegate: delegate];
nkeynes@686
   239
    NSString *iconFile = [[NSBundle mainBundle] pathForResource:@"dcemu" ofType:@"gif"];
nkeynes@686
   240
    NSImage *iconImage = [[NSImage alloc] initWithContentsOfFile: iconFile];
nkeynes@686
   241
    [NSApp setApplicationIconImage: iconImage];   
nkeynes@681
   242
    cocoa_gui_create_menu();
nkeynes@681
   243
    NSWindow *window = cocoa_gui_create_main_window();
nkeynes@681
   244
    [window makeKeyAndOrderFront: nil];
nkeynes@681
   245
    [NSApp activateIgnoringOtherApps: YES];   
nkeynes@718
   246
    
nkeynes@681
   247
    [pool release];
nkeynes@681
   248
}
nkeynes@681
   249
nkeynes@681
   250
void gui_main_loop( gboolean run )
nkeynes@681
   251
{
nkeynes@681
   252
    if( run ) {
nkeynes@681
   253
        cocoa_gui_autorun = YES;
nkeynes@681
   254
    }
nkeynes@693
   255
    cocoa_gui_is_running = YES;
nkeynes@681
   256
	[NSApp run];
nkeynes@693
   257
	cocoa_gui_is_running = NO;
nkeynes@681
   258
}
nkeynes@681
   259
nkeynes@681
   260
void gui_update_state(void)
nkeynes@681
   261
{
nkeynes@681
   262
    cocoa_gui_update();
nkeynes@681
   263
}
nkeynes@681
   264
nkeynes@681
   265
gboolean gui_error_dialog( const char *msg, ... )
nkeynes@681
   266
{
nkeynes@693
   267
    if( cocoa_gui_is_running ) {
nkeynes@693
   268
        NSString *error_string;
nkeynes@693
   269
nkeynes@693
   270
        va_list args;
nkeynes@693
   271
        va_start(args, msg);
nkeynes@693
   272
        error_string = [[NSString alloc] initWithFormat: [NSString stringWithCString: msg] arguments: args];
nkeynes@693
   273
        NSRunAlertPanel(@"Error in lxdream", error_string, nil, nil, nil);
nkeynes@693
   274
        va_end(args);
nkeynes@693
   275
        return TRUE;
nkeynes@693
   276
    } else {
nkeynes@693
   277
        return FALSE;
nkeynes@693
   278
    }
nkeynes@681
   279
}
nkeynes@681
   280
nkeynes@681
   281
void gui_update_io_activity( io_activity_type io, gboolean active )
nkeynes@681
   282
{
nkeynes@681
   283
nkeynes@681
   284
}
nkeynes@681
   285
nkeynes@681
   286
nkeynes@681
   287
uint32_t cocoa_gui_run_slice( uint32_t nanosecs )
nkeynes@681
   288
{
nkeynes@681
   289
    NSEvent *event;
nkeynes@681
   290
    NSAutoreleasePool *pool;
nkeynes@681
   291
nkeynes@681
   292
    cocoa_gui_nanos += nanosecs;
nkeynes@681
   293
    if( cocoa_gui_nanos > GUI_TICK_PERIOD ) { /* 10 ms */
nkeynes@681
   294
        cocoa_gui_nanos -= GUI_TICK_PERIOD;
nkeynes@681
   295
        cocoa_gui_ticks ++;
nkeynes@681
   296
        uint32_t current_period = cocoa_gui_ticks * GUI_TICK_PERIOD;
nkeynes@681
   297
nkeynes@681
   298
        // Run the event loop
nkeynes@681
   299
        pool = [NSAutoreleasePool new];
nkeynes@681
   300
        while( (event = [NSApp nextEventMatchingMask: NSAnyEventMask untilDate: nil 
nkeynes@681
   301
                         inMode: NSDefaultRunLoopMode dequeue: YES]) != nil ) {
nkeynes@681
   302
            [NSApp sendEvent: event];
nkeynes@681
   303
        }
nkeynes@681
   304
        [pool release];
nkeynes@681
   305
nkeynes@681
   306
        struct timeval tv;
nkeynes@681
   307
        gettimeofday(&tv,NULL);
nkeynes@681
   308
        uint32_t ns = ((tv.tv_sec - cocoa_gui_lasttv.tv_sec) * 1000000000) + 
nkeynes@681
   309
        (tv.tv_usec - cocoa_gui_lasttv.tv_usec)*1000;
nkeynes@681
   310
        if( (ns * 1.05) < current_period ) {
nkeynes@681
   311
            // We've gotten ahead - sleep for a little bit
nkeynes@681
   312
            struct timespec tv;
nkeynes@681
   313
            tv.tv_sec = 0;
nkeynes@681
   314
            tv.tv_nsec = current_period - ns;
nkeynes@681
   315
            nanosleep(&tv, &tv);
nkeynes@681
   316
        }
nkeynes@681
   317
nkeynes@681
   318
        /* Update the display every 10 ticks (ie 10 times a second) and 
nkeynes@681
   319
         * save the current tv value */
nkeynes@681
   320
        if( cocoa_gui_ticks > 10 ) {
nkeynes@681
   321
            gchar buf[32];
nkeynes@681
   322
            cocoa_gui_ticks -= 10;
nkeynes@681
   323
nkeynes@681
   324
            double speed = (float)( (double)current_period * 100.0 / ns );
nkeynes@681
   325
            cocoa_gui_lasttv.tv_sec = tv.tv_sec;
nkeynes@681
   326
            cocoa_gui_lasttv.tv_usec = tv.tv_usec;
nkeynes@681
   327
            snprintf( buf, 32, _("Running (%2.4f%%)"), speed );
nkeynes@681
   328
            [((LxdreamMainWindow *)[NSApp mainWindow]) setStatusText: buf];
nkeynes@681
   329
nkeynes@681
   330
        }
nkeynes@681
   331
    }
nkeynes@681
   332
    return nanosecs;
nkeynes@681
   333
}
nkeynes@681
   334
nkeynes@681
   335
void cocoa_gui_update( void )
nkeynes@681
   336
{
nkeynes@681
   337
    
nkeynes@681
   338
}
nkeynes@681
   339
nkeynes@681
   340
void cocoa_gui_start( void )
nkeynes@681
   341
{
nkeynes@681
   342
    LxdreamMainWindow *win = (LxdreamMainWindow *)[NSApp mainWindow];
nkeynes@681
   343
    [win setRunning: YES];
nkeynes@681
   344
    cocoa_gui_nanos = 0;
nkeynes@681
   345
    gettimeofday(&cocoa_gui_lasttv,NULL);
nkeynes@681
   346
}
nkeynes@681
   347
nkeynes@681
   348
void cocoa_gui_stop( void )
nkeynes@681
   349
{
nkeynes@681
   350
    LxdreamMainWindow *win = (LxdreamMainWindow *)[NSApp mainWindow];
nkeynes@681
   351
    [win setRunning: NO];
nkeynes@681
   352
}
nkeynes@681
   353
nkeynes@681
   354
/**
nkeynes@681
   355
 * Queue a dreamcast_run() to execute after the currently event(s)
nkeynes@681
   356
 */
nkeynes@681
   357
void cocoa_gui_run_later( void )
nkeynes@681
   358
{
nkeynes@681
   359
    [[NSRunLoop currentRunLoop] performSelector: @selector(run_immediate) 
nkeynes@681
   360
         target: [NSApp delegate] argument: nil order: 1 
nkeynes@681
   361
         modes: [NSArray arrayWithObject: NSDefaultRunLoopMode] ];
nkeynes@718
   362
}
nkeynes@718
   363
nkeynes@718
   364
NSImage *NSImage_new_from_framebuffer( frame_buffer_t buffer )
nkeynes@718
   365
{
nkeynes@718
   366
    NSBitmapImageRep *rep = 
nkeynes@718
   367
        [[NSBitmapImageRep alloc] initWithBitmapDataPlanes: &buffer->data
nkeynes@718
   368
         pixelsWide: buffer->width  pixelsHigh: buffer->height
nkeynes@718
   369
        bitsPerSample: 8 samplesPerPixel: 3
nkeynes@718
   370
        hasAlpha: NO isPlanar: NO
nkeynes@718
   371
        colorSpaceName: NSDeviceRGBColorSpace  bitmapFormat: 0
nkeynes@718
   372
        bytesPerRow: buffer->rowstride  bitsPerPixel: 24];
nkeynes@718
   373
   
nkeynes@718
   374
    NSImage *image = [[NSImage alloc] initWithSize: NSMakeSize(0.0,0.0)];
nkeynes@718
   375
    [image addRepresentation: rep];
nkeynes@718
   376
    return image;
nkeynes@718
   377
}
.