1
0
mirror of https://git.code.sf.net/p/fuse-emulator/fuse synced 2026-01-30 04:22:18 +03:00
Files
fuse/widget/menu.c
2004-07-03 20:34:14 +00:00

452 lines
9.4 KiB
C

/* menu.c: general menu widget
Copyright (c) 2001-2004 Philip Kendall
$Id$
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
Author contact information:
E-mail: pak21-fuse@srcf.ucam.org
Postal address: 15 Crescent Road, Wokingham, Berks, RG40 2DB, England
*/
#include <config.h>
#ifdef USE_WIDGET
#include <errno.h>
#include <stdio.h>
#include <string.h>
#include "dck.h"
#include "debugger/debugger.h"
#include "event.h"
#include "fuse.h"
#include "joystick.h"
#include "machine.h"
#include "machines/specplus3.h"
#include "menu.h"
#include "psg.h"
#include "rzx.h"
#include "screenshot.h"
#include "settings.h"
#include "simpleide.h"
#include "snapshot.h"
#include "tape.h"
#include "trdos.h"
#include "ui/uidisplay.h"
#include "utils.h"
#include "widget_internals.h"
widget_menu_entry *menu;
int widget_menu_draw( void *data )
{
widget_menu_entry *ptr;
size_t menu_entries, i;
menu = (widget_menu_entry*)data;
/* How many menu items do we have? */
for( ptr = &menu[1]; ptr->text; ptr++ ) ;
menu_entries = ptr - &menu[1];
widget_dialog_with_border( 1, 2, 30, menu_entries + 2 );
widget_printstring( 15 - strlen( menu->text ) / 2, 2,
WIDGET_COLOUR_FOREGROUND, menu->text );
for( i=0; i<menu_entries; i++ )
widget_printstring( 2, i+4, WIDGET_COLOUR_FOREGROUND,
menu[i+1].text );
widget_display_lines( 2, menu_entries + 2 );
return 0;
}
void
widget_menu_keyhandler( input_key key )
{
widget_menu_entry *ptr;
switch( key ) {
#if 0
case INPUT_KEY_Resize: /* Fake keypress used on window resize */
widget_menu_draw( menu );
break;
#endif
case INPUT_KEY_Escape:
widget_end_widget( WIDGET_FINISHED_CANCEL );
return;
case INPUT_KEY_Return:
widget_end_widget( WIDGET_FINISHED_OK );
return;
default: /* Keep gcc happy */
break;
}
for( ptr=&menu[1]; ptr->text; ptr++ ) {
if( key == ptr->key ) {
if( ptr->submenu ) {
widget_do( WIDGET_TYPE_MENU, ptr->submenu );
} else {
ptr->callback( ptr->action );
}
break;
}
}
}
/* General callbacks */
char *
menu_get_filename( const char *title )
{
char *filename = NULL;
widget_do( WIDGET_TYPE_FILESELECTOR, NULL );
if( widget_filesel_name ) {
filename = strdup( widget_filesel_name );
if( !filename )
ui_error( UI_ERROR_ERROR, "Out of memory at %s:%d", __FILE__, __LINE__ );
}
return filename;
}
void
menu_file_savesnapshot( int action )
{
widget_end_all( WIDGET_FINISHED_OK );
if( settings_current.snaps_as_z80 ) {
snapshot_write( "snapshot.z80" );
} else {
snapshot_write( "snapshot.szx" );
}
}
void
menu_file_recording_record( int action )
{
if( rzx_playback || rzx_recording ) return;
widget_end_all( WIDGET_FINISHED_OK );
rzx_start_recording( "record.rzx", 1 );
}
void
menu_file_recording_recordfromsnapshot( int action )
{
int error;
if( rzx_playback || rzx_recording ) return;
/* Get a snapshot name */
widget_do( WIDGET_TYPE_FILESELECTOR, NULL );
if( !widget_filesel_name ) widget_end_widget( WIDGET_FINISHED_CANCEL );
error = snapshot_read( widget_filesel_name );
if( error ) {
if( widget_filesel_name ) free( widget_filesel_name );
return;
}
rzx_start_recording( "record.rzx", settings_current.embed_snapshot );
}
void
menu_file_aylogging_record( int action )
{
if( psg_recording ) return;
widget_end_all( WIDGET_FINISHED_OK );
psg_start_recording( "ay.psg" );
}
void
menu_file_savescreenasscr( int action )
{
widget_end_all( WIDGET_FINISHED_OK );
screenshot_scr_write( "fuse.scr" );
}
#ifdef USE_LIBPNG
void
menu_file_savescreenaspng( int action )
{
scaler_type scaler;
scaler = menu_get_scaler( screenshot_available_scalers );
if( scaler == SCALER_NUM ) return;
screenshot_write( "fuse.png", scaler );
}
#endif /* #ifdef USE_LIBPNG */
scaler_type
menu_get_scaler( scaler_available_fn selector )
{
size_t count, i;
const char *options[ SCALER_NUM ];
widget_select_t info;
int error;
count = 0; info.current = 0;
for( i = 0; i < SCALER_NUM; i++ )
if( selector( i ) ) {
if( current_scaler == i ) info.current = count;
options[ count++ ] = scaler_name( i );
}
info.title = "Select scaler";
info.options = options;
info.count = count;
error = widget_do( WIDGET_TYPE_SELECT, &info );
if( error ) return SCALER_NUM;
if( info.result == -1 ) return SCALER_NUM;
for( i = 0; i < SCALER_NUM; i++ )
if( selector( i ) && !info.result-- ) return i;
ui_error( UI_ERROR_ERROR, "widget_select_scaler: ran out of scalers" );
fuse_abort();
}
void
menu_file_exit( int action )
{
fuse_exiting = 1;
widget_end_all( WIDGET_FINISHED_OK );
}
void
menu_options_general( int action )
{
widget_do( WIDGET_TYPE_GENERAL, NULL );
}
void
menu_options_peripherals( int action )
{
widget_do( WIDGET_TYPE_PERIPHERALS, NULL );
}
void
menu_options_sound( int action )
{
widget_do( WIDGET_TYPE_SOUND, NULL );
}
void
menu_options_rzx( int action )
{
widget_do( WIDGET_TYPE_RZX, NULL );
}
void
menu_options_joysticks_select( int action )
{
widget_select_t info;
int *setting, error;
setting = NULL;
switch( action - 1 ) {
case 0: setting = &( settings_current.joystick_1_output ); break;
case 1: setting = &( settings_current.joystick_2_output ); break;
case JOYSTICK_KEYBOARD:
setting = &( settings_current.joystick_keyboard_output ); break;
}
info.title = "Select joystick";
info.options = joystick_name;
info.count = JOYSTICK_TYPE_COUNT;
info.current = *setting;
error = widget_do( WIDGET_TYPE_SELECT, &info );
if( error ) return;
if( info.result != -1 ) *setting = info.result;
}
/* Options/Select ROMs/<type> */
int
menu_select_roms( libspectrum_machine machine, size_t start, size_t count )
{
widget_roms_info info;
info.machine = machine;
info.start = start;
info.count = count;
info.initialised = 0;
return widget_do( WIDGET_TYPE_ROM, &info );
}
void
menu_machine_reset( int action )
{
widget_end_all( WIDGET_FINISHED_OK );
machine_reset();
}
void
menu_machine_select( int action )
{
widget_select_t info;
char **options, *buffer;
size_t i;
int error;
libspectrum_machine new_machine;
options = malloc( machine_count * sizeof( const char * ) );
if( !options ) {
ui_error( UI_ERROR_ERROR, "out of memory at %s:%d", __FILE__, __LINE__ );
return;
}
buffer = malloc( 40 * machine_count );
if( !buffer ) {
ui_error( UI_ERROR_ERROR, "out of memory at %s:%d", __FILE__, __LINE__ );
free( options );
return;
}
for( i = 0; i < machine_count; i++ ) {
options[i] = &buffer[ i * 40 ];
snprintf( options[i], 40,
libspectrum_machine_name( machine_types[i]->machine ) );
if( machine_current->machine == machine_types[i]->machine )
info.current = i;
}
info.title = "Select machine";
info.options = (const char**)options;
info.count = machine_count;
error = widget_do( WIDGET_TYPE_SELECT, &info );
free( buffer ); free( options );
if( error ) return;
if( info.result == -1 ) return;
new_machine = machine_types[ info.result ]->machine;
if( machine_current->machine != new_machine ) machine_select( new_machine );
}
void
menu_machine_debugger( int action )
{
debugger_mode = DEBUGGER_MODE_HALTED;
widget_do( WIDGET_TYPE_DEBUGGER, NULL );
}
void
menu_media_tape_browse( int action )
{
widget_do( WIDGET_TYPE_BROWSE, NULL );
}
int
ui_tape_write( void )
{
widget_end_all( WIDGET_FINISHED_OK );
return tape_write( "tape.tzx" );
}
#ifdef HAVE_765_H
int
ui_plus3_disk_write( specplus3_drive_number which )
{
char filename[ 20 ];
snprintf( filename, 20, "drive%c.dsk", (char)( 'a' + which ) );
return specplus3_disk_write( which, filename );
}
#endif /* #ifdef HAVE_765_H */
int
ui_trdos_disk_write( trdos_drive_number which )
{
char filename[ 20 ];
snprintf( filename, 20, "drive%c.trd", (char)( 'a' + which ) );
return trdos_disk_write( which, filename );
}
void
menu_help_keyboard( int action )
{
int error, fd;
utils_file file;
widget_picture_data info;
static const char *filename = "keyboard.scr";
fd = utils_find_auxiliary_file( filename, UTILS_AUXILIARY_LIB );
if( fd == -1 ) {
ui_error( UI_ERROR_ERROR, "couldn't find keyboard picture ('%s')",
filename );
return;
}
error = utils_read_fd( fd, filename, &file ); if( error ) return;
if( file.length != 6912 ) {
ui_error( UI_ERROR_ERROR, "keyboard picture ('%s') is not 6912 bytes long",
filename );
utils_close_file( &file );
return;
}
info.filename = filename;
info.screen = file.buffer;
widget_do( WIDGET_TYPE_PICTURE, &info );
if( utils_close_file( &file ) ) return;
}
/* Function to (de)activate specific menu items */
/* FIXME: make this work */
int
ui_menu_activate( ui_menu_item item, int active )
{
return 0;
}
#endif /* #ifdef USE_WIDGET */