/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */ /* * Copyright 2001 by Eric House (fixin@peak.org). All rights reserved. * * 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. */ #ifdef PLATFORM_GTK #include #include "gtknewgame.h" #include "strutils.h" #define MAX_SIZE_CHOICES 10 /* make the appropriate set of entries sensitive or not */ typedef struct ItemNumPair { GtkWidget* item; short index; gboolean found; } ItemNumPair; void countBeforeSame( GtkWidget *widget, gpointer data ) { ItemNumPair* pair = (ItemNumPair*)data; if ( !pair->found ) { if ( pair->item == widget ) { pair->found = TRUE; } else { ++pair->index; } } } /* countBeforeSame */ static void setChildrenSensitivity( GtkWidget* hbox, gboolean enabling ) { gtk_widget_set_sensitive( hbox, enabling ); } /* setChildrenSensitivity */ static void nplayers_menu_select( GtkWidget* item, GtkNewGameState* state ) { short prevNPlayers = state->nPlayers; short newNPlayers; GtkWidget* parent = item->parent; ItemNumPair pair; short high, low; gboolean enabling; pair.item = item; pair.index = 0; pair.found = FALSE; gtk_container_foreach( GTK_CONTAINER(parent), countBeforeSame, &pair ); newNPlayers = pair.index + 1; low = XP_MIN( newNPlayers, prevNPlayers ); high = XP_MAX( newNPlayers, prevNPlayers ); enabling = newNPlayers > prevNPlayers; /* now loop through all the hboxes */ while ( low < high ) { setChildrenSensitivity( state->playerEntries[low], enabling ); ++low; } state->nPlayers = newNPlayers; } /* nplayers_menu_select */ static void role_menu_select( GtkWidget* item, GtkNewGameState* state ) { int i; for ( i = 0; i < 3; ++i ) { if ( item == state->roleMenuItems[i] ) { break; } } XP_ASSERT( i < 3 ); /* did we not find it? */ state->role = (Connectedness)i; } /* role_menu_select */ static void size_menu_select( GtkWidget* item, GtkNewGameState* state ) { ItemNumPair pair; pair.item = item; pair.index = 0; pair.found = FALSE; gtk_container_foreach( GTK_CONTAINER(item->parent), countBeforeSame, &pair ); XP_DEBUGF( "changing nCols from %d to %d\n", state->nCols, MAX_COLS - pair.index ); state->nCols = MAX_COLS - pair.index; } /* size_menu_select */ typedef struct LoadPair { LocalPlayer* info; XP_U16 counter; MPSLOT } LoadPair; static void loadCopyValues( GtkWidget* item, gpointer data ) { LoadPair* lp = (LoadPair*)data; LocalPlayer* player = lp->info; const char* entryText; switch( lp->counter ) { case 0: /* labels */ case 2: break; case 1: /* name field */ entryText = gtk_entry_get_text( GTK_ENTRY(item) ); player->name = copyString( MPPARM(lp->mpool) entryText ); break; case 3: /* passwd field */ entryText = gtk_entry_get_text( GTK_ENTRY(item) ); player->password = copyString( MPPARM(lp->mpool) entryText ); break; case 4: /* is Robot */ player->isRobot = GTK_WIDGET_STATE( item ) == GTK_STATE_ACTIVE; break; case 5: /* is local */ player->isLocal = GTK_WIDGET_STATE( item ) == GTK_STATE_ACTIVE; break; default: XP_ASSERT( 0 ); } ++lp->counter; } /* loadCopyValues */ static void handle_ok( GtkWidget* widget, gpointer closure ) { GtkNewGameState* state = (GtkNewGameState*)closure; CurGameInfo* gi = &state->globals->cGlobals.params->gi; short i; LoadPair lp; MPASSIGN( lp.mpool, state->globals->cGlobals.params->util->mpool ); gi->nPlayers = state->nPlayers; gi->boardSize = state->nCols; /* they're the same for now */ for ( i = 0; i < state->nPlayers; ++i ) { LocalPlayer* player = &gi->players[i]; GtkWidget* hbox = state->playerEntries[i]; lp.info = player; lp.counter = 0; /* Read values out of the items in the hbox, which are, in order, the name entry, passwd entry, isLocal box and isRobot box */ gtk_container_foreach( GTK_CONTAINER(hbox), loadCopyValues, &lp ); } state->cancelled = XP_FALSE; gtk_main_quit(); } static void handle_cancel( GtkWidget* widget, void* closure ) { GtkNewGameState* state = (GtkNewGameState*)closure; state->cancelled = XP_TRUE; gtk_main_quit(); } static void handle_revert( GtkWidget* widget, void* closure ) { GtkNewGameState* state = (GtkNewGameState*)closure; state->revert = TRUE; gtk_main_quit(); } /* handle_revert */ GtkWidget* make_menu_item( gchar* name, GCallback func, gpointer data ) { GtkWidget* item; item = gtk_menu_item_new_with_label( name ); g_signal_connect( GTK_OBJECT(item), "activate", func, data ); gtk_widget_show( item ); return item; } /* make_menu_item */ static GtkWidget* makeButton( char* text, GCallback func, gpointer data ) { GtkWidget* button = gtk_button_new_with_label( text ); g_signal_connect( GTK_OBJECT(button), "clicked", func, data ); gtk_widget_show( button ); return button; } /* makeButton */ static GtkWidget* makeNewGameDialog( GtkNewGameState* state ) { GtkWidget* dialog; GtkWidget* vbox; GtkWidget* hbox; GtkWidget* item; GtkWidget* roleMenu; GtkWidget* nPlayersMenu; GtkWidget* boardSizeMenu; GtkWidget* opt; CurGameInfo* gi; short i; char* roles[] = { "Standalone", "Host", "Guest" }; dialog = gtk_dialog_new(); gtk_window_set_modal( GTK_WINDOW( dialog ), TRUE ); vbox = gtk_vbox_new( FALSE, 0 ); /* Role menu */ state->role = state->globals->cGlobals.params->serverRole; hbox = gtk_hbox_new( FALSE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), gtk_label_new("Role:"), FALSE, TRUE, 0 ); opt = gtk_option_menu_new(); roleMenu = gtk_menu_new(); for ( i = 0; i < sizeof(roles)/sizeof(roles[0]); ++i ) { item = make_menu_item( roles[i], GTK_SIGNAL_FUNC(role_menu_select), state ); state->roleMenuItems[i] = item; gtk_menu_append( GTK_MENU(roleMenu), item ); if ( i == state->role ) { gtk_menu_set_active( GTK_MENU(roleMenu), i ); } } gtk_option_menu_set_menu( GTK_OPTION_MENU(opt), roleMenu ); gtk_widget_show( opt ); gtk_box_pack_start( GTK_BOX(hbox), opt, FALSE, TRUE, 0 ); gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); /* NPlayers menu */ hbox = gtk_hbox_new( FALSE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), gtk_label_new("Number of players"), FALSE, TRUE, 0 ); opt = gtk_option_menu_new(); nPlayersMenu = gtk_menu_new(); gi = &state->globals->cGlobals.params->gi; state->nPlayers = gi->nPlayers; for ( i = 0; i < MAX_NUM_PLAYERS; ++i ) { char buf[2]; snprintf( buf, 2, "%d", i+1 ); item = make_menu_item( buf, GTK_SIGNAL_FUNC(nplayers_menu_select), state ); gtk_menu_append( GTK_MENU(nPlayersMenu), item ); if ( i+1 == state->nPlayers ) { gtk_menu_set_active( GTK_MENU(nPlayersMenu), i ); } } gtk_option_menu_set_menu( GTK_OPTION_MENU(opt), nPlayersMenu ); gtk_widget_show( opt ); gtk_box_pack_start( GTK_BOX(hbox), opt, FALSE, TRUE, 0 ); gtk_widget_show( hbox ); gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); for ( i = 0; i < MAX_NUM_PLAYERS; ++i ) { GtkWidget* label; GtkWidget* nameField = gtk_entry_new(); GtkWidget* passwdField = gtk_entry_new_with_max_length( 6 ); GtkWidget* robotCheck = gtk_check_button_new_with_label( "robot" ); GtkWidget* localCheck = gtk_check_button_new_with_label( "is local" ); hbox = gtk_hbox_new( FALSE, 0 ); state->playerEntries[i] = hbox; label = gtk_label_new("name:"); gtk_box_pack_start( GTK_BOX(hbox), label, FALSE, TRUE, 0 ); gtk_widget_show( label ); gtk_box_pack_start( GTK_BOX(hbox), nameField, FALSE, TRUE, 0 ); gtk_widget_show( nameField ); label = gtk_label_new("passwd:"); gtk_box_pack_start( GTK_BOX(hbox), label, FALSE, TRUE, 0 ); gtk_widget_show( label ); gtk_box_pack_start( GTK_BOX(hbox), passwdField, FALSE, TRUE, 0 ); gtk_widget_show( passwdField ); gtk_box_pack_start( GTK_BOX(hbox), robotCheck, FALSE, TRUE, 0 ); gtk_widget_show( robotCheck ); gtk_box_pack_start( GTK_BOX(hbox), localCheck, FALSE, TRUE, 0 ); gtk_widget_show( localCheck ); if ( i < state->nPlayers ) { XP_Bool isSet; gtk_entry_set_text( GTK_ENTRY(nameField), gi->players[i].name ); isSet = gi->players[i].isRobot; gtk_toggle_button_set_state( GTK_TOGGLE_BUTTON(robotCheck), isSet ); XP_DEBUGF( "isRobot set to %d\n", isSet ); isSet = gi->players[i].isLocal; gtk_toggle_button_set_state( GTK_TOGGLE_BUTTON(localCheck), isSet ); XP_DEBUGF( "isLocal set to %d\n", isSet ); } else { char buf[10]; snprintf( buf, sizeof(buf), "Player %d", i+1 ); gtk_entry_set_text( GTK_ENTRY(nameField), buf ); gtk_widget_set_sensitive( hbox, FALSE ); gtk_toggle_button_set_state( GTK_TOGGLE_BUTTON(localCheck), XP_TRUE ); } gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); gtk_widget_show( hbox ); } /* board size choices */ hbox = gtk_hbox_new( FALSE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), gtk_label_new("Board size"), FALSE, TRUE, 0 ); opt = gtk_option_menu_new(); boardSizeMenu = gtk_menu_new(); state->nCols = gi->boardSize; for ( i = 0; i < MAX_SIZE_CHOICES; ++i ) { char buf[10]; XP_U16 siz = MAX_COLS - i; snprintf( buf, sizeof(buf), "%dx%d", siz, siz ); item = make_menu_item( buf, GTK_SIGNAL_FUNC(size_menu_select), state ); gtk_menu_append( GTK_MENU(boardSizeMenu), item ); if ( siz == state->nCols ) { gtk_menu_set_active( GTK_MENU(boardSizeMenu), i ); } } gtk_option_menu_set_menu( GTK_OPTION_MENU(opt), boardSizeMenu ); gtk_widget_show( opt ); gtk_box_pack_start( GTK_BOX(hbox), opt, FALSE, TRUE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), gtk_label_new("Dictionary: "), FALSE, TRUE, 0 ); XP_ASSERT( gi->dictName ); gtk_box_pack_start( GTK_BOX(hbox), gtk_label_new(gi->dictName), FALSE, TRUE, 0 ); gtk_widget_show( hbox ); gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); /* buttons at the bottom */ hbox = gtk_hbox_new( FALSE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), makeButton( "Ok", GTK_SIGNAL_FUNC(handle_ok) , state ), FALSE, TRUE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), makeButton( "Revert", GTK_SIGNAL_FUNC(handle_revert), state ), FALSE, TRUE, 0 ); gtk_box_pack_start( GTK_BOX(hbox), makeButton( "Cancel", GTK_SIGNAL_FUNC(handle_cancel), state ), FALSE, TRUE, 0 ); gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); gtk_widget_show( vbox ); gtk_container_add( GTK_CONTAINER( GTK_DIALOG(dialog)->action_area), vbox); gtk_widget_show_all (dialog); return dialog; } /* makeNewGameDialog */ gboolean newGameDialog( GtkAppGlobals* globals/* , GtkGameInfo* gameInfo */ ) { GtkNewGameState state; state.globals = globals; /* returns when button handler calls gtk_main_quit */ do { GtkWidget* dialog = makeNewGameDialog( &state ); state.revert = FALSE; gtk_main(); gtk_widget_destroy( dialog ); } while ( state.revert ); return !state.cancelled; } /* newGameDialog */ #endif /* PLATFORM_GTK */