/* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */ /* * Copyright 2000-2013 by Eric House (xwords@eehouse.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 "xptypes.h" #include "main.h" #include "gamesdb.h" #include "gtkboard.h" #include "linuxmain.h" static void onNewData( GTKGamesGlobals* gg, sqlite3_int64 rowid, XP_Bool isNew ); static void recordOpened( GTKGamesGlobals* gg, GtkAppGlobals* globals ) { gg->globalsList = g_slist_prepend( gg->globalsList, globals ); globals->gg = gg; } static void recordClosed( GTKGamesGlobals* gg, GtkAppGlobals* globals ) { gg->globalsList = g_slist_remove( gg->globalsList, globals ); } static XP_Bool gameIsOpen( GTKGamesGlobals* gg, sqlite3_int64 rowid ) { XP_Bool found = XP_FALSE; GSList* iter; for ( iter = gg->globalsList; !!iter && !found; iter = iter->next ) { GtkAppGlobals* globals = (GtkAppGlobals*)iter->data; found = globals->cGlobals.selRow == rowid; } return found; } enum { CHECK_ITEM, ROW_ITEM, NAME_ITEM, ROOM_ITEM, OVER_ITEM, TURN_ITEM, NMOVES_ITEM, MISSING_ITEM, N_ITEMS }; static void tree_selection_changed_cb( GtkTreeSelection* selection, gpointer data ) { GTKGamesGlobals* gg = (GTKGamesGlobals*)data; GtkTreeIter iter; GtkTreeModel *model; if ( gtk_tree_selection_get_selected( selection, &model, &iter ) ) { gtk_tree_model_get( model, &iter, ROW_ITEM, &gg->selRow, -1 ); } } static void addTextColumn( GtkWidget* list, const gchar* title, int item ) { GtkCellRenderer* renderer = gtk_cell_renderer_text_new(); GtkTreeViewColumn* column = gtk_tree_view_column_new_with_attributes( title, renderer, "text", item, NULL ); gtk_tree_view_append_column( GTK_TREE_VIEW(list), column ); } static GtkWidget* init_games_list( GTKGamesGlobals* gg ) { GtkWidget* list = gtk_tree_view_new(); GtkCellRenderer* renderer; GtkTreeViewColumn* column; renderer = gtk_cell_renderer_toggle_new(); column = gtk_tree_view_column_new_with_attributes( "", renderer, "active", CHECK_ITEM, NULL ); gtk_tree_view_append_column( GTK_TREE_VIEW(list), column ); addTextColumn( list, "Row", ROW_ITEM ); addTextColumn( list, "Name", NAME_ITEM ); addTextColumn( list, "Room", ROOM_ITEM ); addTextColumn( list, "Ended", OVER_ITEM ); addTextColumn( list, "Turn", TURN_ITEM ); addTextColumn( list, "NMoves", NMOVES_ITEM ); addTextColumn( list, "NMissing", MISSING_ITEM ); GtkListStore* store = gtk_list_store_new( N_ITEMS, G_TYPE_BOOLEAN, G_TYPE_INT64, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_INT, G_TYPE_INT, G_TYPE_INT ); gtk_tree_view_set_model( GTK_TREE_VIEW(list), GTK_TREE_MODEL(store) ); g_object_unref( store ); GtkTreeSelection* select = gtk_tree_view_get_selection( GTK_TREE_VIEW (list) ); gtk_tree_selection_set_mode( select, GTK_SELECTION_SINGLE ); g_signal_connect( G_OBJECT(select), "changed", G_CALLBACK (tree_selection_changed_cb), gg ); return list; } static void add_to_list( GtkWidget* list, sqlite3_int64 rowid, XP_Bool isNew, const GameInfo* gib ) { GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(list)); GtkListStore* store = GTK_LIST_STORE( model ); GtkTreeIter iter; if ( isNew ) { gtk_list_store_append( store, &iter ); } else { gboolean valid; for ( valid = gtk_tree_model_get_iter_first( model, &iter ); valid; valid = gtk_tree_model_iter_next( model, &iter ) ) { sqlite3_int64 tmpid; gtk_tree_model_get( model, &iter, ROW_ITEM, &tmpid, -1 ); if ( tmpid == rowid ) { break; } } } gtk_list_store_set( store, &iter, ROW_ITEM, rowid, NAME_ITEM, gib->name, ROOM_ITEM, gib->room, OVER_ITEM, gib->gameOver, TURN_ITEM, gib->turn, NMOVES_ITEM, gib->nMoves, MISSING_ITEM, gib->nMissing, -1 ); XP_LOGF( "DONE adding" ); } static void handle_newgame_button( GtkWidget* XP_UNUSED(widget), void* closure ) { GTKGamesGlobals* gg = (GTKGamesGlobals*)closure; XP_LOGF( "%s called", __func__ ); GtkAppGlobals* globals = malloc( sizeof(*globals) ); gg->params->needsNewGame = XP_FALSE; initGlobals( globals, gg->params ); if ( !makeNewGame( globals ) ) { freeGlobals( globals ); } else { GtkWidget* gameWindow = globals->window; globals->cGlobals.pDb = gg->pDb; globals->cGlobals.selRow = -1; recordOpened( gg, globals ); gtk_widget_show( gameWindow ); } } static void handle_open_button( GtkWidget* XP_UNUSED(widget), void* closure ) { GTKGamesGlobals* gg = (GTKGamesGlobals*)closure; if ( -1 != gg->selRow && !gameIsOpen( gg, gg->selRow ) ) { gg->params->needsNewGame = XP_FALSE; GtkAppGlobals* globals = malloc( sizeof(*globals) ); initGlobals( globals, gg->params ); globals->cGlobals.pDb = gg->pDb; globals->cGlobals.selRow = gg->selRow; recordOpened( gg, globals ); gtk_widget_show( globals->window ); } } static void handle_quit_button( GtkWidget* XP_UNUSED(widget), gpointer data ) { GTKGamesGlobals* gg = (GTKGamesGlobals*)data; gg = gg; gtk_main_quit(); } static void handle_destroy( GtkWidget* XP_UNUSED(widget), gpointer data ) { LOG_FUNC(); GTKGamesGlobals* gg = (GTKGamesGlobals*)data; GSList* iter; for ( iter = gg->globalsList; !!iter; iter = iter->next ) { GtkAppGlobals* globals = (GtkAppGlobals*)iter->data; freeGlobals( globals ); } g_slist_free( gg->globalsList ); gtk_main_quit(); } static void addButton( gchar* label, GtkWidget* parent, GCallback proc, void* closure ) { GtkWidget* button = gtk_button_new_with_label( label ); gtk_container_add( GTK_CONTAINER(parent), button ); g_signal_connect( GTK_OBJECT(button), "clicked", G_CALLBACK(proc), closure ); gtk_widget_show( button ); } static GtkWidget* makeGamesWindow( GTKGamesGlobals* gg ) { GtkWidget* window; window = gtk_window_new( GTK_WINDOW_TOPLEVEL ); g_signal_connect( G_OBJECT(window), "destroy", G_CALLBACK(handle_destroy), gg ); GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); gtk_container_add( GTK_CONTAINER(window), vbox ); gtk_widget_show( vbox ); GtkWidget* list = init_games_list( gg ); gg->listWidget = list; gtk_container_add( GTK_CONTAINER(vbox), list ); gtk_widget_show( list ); GSList* games = listGames( gg ); for ( GSList* iter = games; !!iter; iter = iter->next ) { sqlite3_int64* rowid = (sqlite3_int64*)iter->data; onNewData( gg, *rowid, XP_TRUE ); } g_slist_free( games ); GtkWidget* hbox = gtk_hbox_new( FALSE, 0 ); gtk_widget_show( hbox ); gtk_container_add( GTK_CONTAINER(vbox), hbox ); addButton( "New game", hbox, G_CALLBACK(handle_newgame_button), gg ); addButton( "Open", hbox, G_CALLBACK(handle_open_button), gg ); addButton( "Quit", hbox, G_CALLBACK(handle_quit_button), gg ); gtk_widget_show( window ); return window; } static gint freeGameGlobals( gpointer data ) { LOG_FUNC(); GtkAppGlobals* globals = (GtkAppGlobals*)data; GTKGamesGlobals* gg = globals->gg; recordClosed( gg, globals ); freeGlobals( globals ); return 0; /* don't run again */ } void windowDestroyed( GtkAppGlobals* globals ) { /* schedule to run after we're back to main loop */ (void)g_idle_add( freeGameGlobals, globals ); } static void onNewData( GTKGamesGlobals* gg, sqlite3_int64 rowid, XP_Bool isNew ) { GameInfo gib; if ( getGameInfo( gg, rowid, &gib ) ) { add_to_list( gg->listWidget, rowid, isNew, &gib ); } } void onGameSaved( void* closure, sqlite3_int64 rowid, XP_Bool firstTime ) { GtkAppGlobals* globals = (GtkAppGlobals*)closure; GTKGamesGlobals* gg = globals->gg; onNewData( gg, rowid, firstTime ); } int gtkmain( LaunchParams* params ) { GTKGamesGlobals gg = {0}; gg.selRow = -1; gg.params = params; gg.pDb = openGamesDB( params->dbName ); (void)makeGamesWindow( &gg ); gtk_main(); closeGamesDB( gg.pDb ); return 0; } /* gtkmain */ #endif /* PLATFORM_GTK */