/* -*-mode: C; fill-column: 77; c-basic-offset: 4; -*- */ /* * Copyright 2003 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 BEYOND_IR #include #include "callback.h" #include "connsdlg.h" #include "strutils.h" #include "palmmain.h" #include "palmutil.h" #include "palmir.h" #include "palmbt.h" /* When user pops up via Host gadget, we want to get the port to listen on. * When pops up via the Guest gadget, we want to get the port and IP address * of the host AS WELL AS the local port that we'll tell it we're listening * on. We need local state for both, since user can switch between them and * expect state to live as long as the parent dialog isn't exited. */ typedef struct ConnsDlgState { ListPtr connTypesList; XP_U16 serverRole; XP_Bool isNewGame; CommsAddrRec* addr; XP_BtAddr btAddr; /* since there's no field, save it here */ } ConnsDlgState; static void strFromField( XP_U16 id, XP_UCHAR* buf, XP_U16 max ) { FieldPtr field = getActiveObjectPtr( id ); XP_UCHAR* str = FldGetTextPtr( field ); XP_U16 len = FldGetTextLength( field ); if ( len > max-1 ) { len = max - 1; } XP_MEMCPY( buf, str, len ); buf[len] = '\0'; } /* strFromField */ static void ctlsFromState( PalmAppGlobals* XP_UNUSED_BT(globals), ConnsDlgState* state ) { XP_Bool isNewGame = state->isNewGame; XP_UCHAR buf[16]; CommsAddrRec* addr = state->addr; if ( addr->conType == COMMS_CONN_RELAY ) { setFieldStr( XW_CONNS_RELAY_FIELD_ID, addr->u.ip_relay.hostName ); setFieldEditable( XW_CONNS_RELAY_FIELD_ID, isNewGame ); StrPrintF( buf, "%d", addr->u.ip_relay.port ); setFieldStr( XW_CONNS_PORT_FIELD_ID, buf ); setFieldEditable( XW_CONNS_PORT_FIELD_ID, isNewGame ); setFieldStr( XW_CONNS_COOKIE_FIELD_ID, addr->u.ip_relay.cookie ); setFieldEditable( XW_CONNS_COOKIE_FIELD_ID, isNewGame ); #ifdef XWFEATURE_BLUETOOTH } else if ( addr->conType == COMMS_CONN_BT && state->serverRole == SERVER_ISCLIENT ) { if ( '\0' == addr->u.bt.hostName[0] ) { XP_BtAddrStr tmp; palm_bt_addrString( globals, &addr->u.bt.btAddr, &tmp ); setFieldStr( XW_CONNS_BT_HOSTFIELD_ID, tmp.chars ); } else { setFieldStr( XW_CONNS_BT_HOSTFIELD_ID, addr->u.bt.hostName ); } #endif } } /* ctlsFromState */ static XP_Bool stateFromCtls( ConnsDlgState* state ) { XP_Bool ok = XP_TRUE; XP_UCHAR buf[16]; CommsAddrRec* addr = state->addr; if ( addr->conType == COMMS_CONN_RELAY ) { strFromField( XW_CONNS_RELAY_FIELD_ID, addr->u.ip_relay.hostName, sizeof(addr->u.ip_relay.hostName) ); strFromField( XW_CONNS_PORT_FIELD_ID, buf, sizeof(buf) ); addr->u.ip_relay.port = StrAToI( buf ); strFromField( XW_CONNS_COOKIE_FIELD_ID, addr->u.ip_relay.cookie, sizeof(addr->u.ip_relay.cookie) ); #ifdef XWFEATURE_BLUETOOTH } else if ( addr->conType == COMMS_CONN_BT && state->serverRole == SERVER_ISCLIENT ) { strFromField( XW_CONNS_BT_HOSTFIELD_ID, addr->u.bt.hostName, sizeof(addr->u.bt.hostName) ); /* Not exactly from a control... */ XP_MEMCPY( &addr->u.bt.btAddr, &state->btAddr, sizeof(addr->u.bt.btAddr) ); LOG_HEX( &addr->u.bt.btAddr, sizeof(addr->u.bt.btAddr), __FUNCTION__ ); #endif } return ok; } /* stateFromCtls */ static void updateFormCtls( FormPtr form, ConnsDlgState* state ) { const XP_U16 relayCtls[] = { XW_CONNS_RELAY_LABEL_ID , XW_CONNS_RELAY_FIELD_ID, XW_CONNS_PORT_LABEL_ID, XW_CONNS_PORT_FIELD_ID, XW_CONNS_COOKIE_LABEL_ID, XW_CONNS_COOKIE_FIELD_ID, 0 }; const XP_U16 btGuestCtls[] = { #ifdef XWFEATURE_BLUETOOTH XW_CONNS_BT_HOSTNAME_LABEL_ID, XW_CONNS_BT_HOSTFIELD_ID, XW_CONNS_BT_BROWSEBUTTON_ID, #else XW_CONNS_BT_NOTSUPPORT_LABEL_ID, #endif 0 }; const XP_U16* allCtls[] = { relayCtls, btGuestCtls }; const XP_U16* on; XP_U16 i; if ( state->addr->conType == COMMS_CONN_RELAY ) { on = relayCtls; } else if ( state->addr->conType == COMMS_CONN_BT #ifdef XWFEATURE_BLUETOOTH && state->serverRole == SERVER_ISCLIENT #endif ) { on = btGuestCtls; } else { on = NULL; } for ( i = 0; i < sizeof(allCtls)/sizeof(allCtls[0]); ++i ) { const XP_U16* cur = allCtls[i]; if ( cur != on ) { disOrEnableSet( form, cur, XP_FALSE ); } } if ( on != NULL ) { disOrEnableSet( form, on, XP_TRUE ); } } /* updateFormCtls */ static void cleanupExit( PalmAppGlobals* globals ) { XP_FREE( globals->mpool, globals->connState ); globals->connState = NULL; FrmReturnToForm( 0 ); } /* cleanupExit */ static XP_U16 conTypeToSel( CommsConnType conType ) { XP_U16 result = 0; switch ( conType ) { case COMMS_CONN_BT: /* result = 0; */break; case COMMS_CONN_IR: result = 1; break; case COMMS_CONN_RELAY: result = 2; break; default: XP_ASSERT(0); } return result; } /* conTypeToSel */ static CommsConnType selToConType( XP_U16 sel ) { CommsConnType conType = COMMS_CONN_BT; switch( sel ) { case 0: /* conType = COMMS_CONN_BT; */break; case 1: conType = COMMS_CONN_IR; break; case 2: conType = COMMS_CONN_RELAY; break; default: XP_ASSERT(0); } return conType; } /* selToConType */ #ifdef XWFEATURE_BLUETOOTH static void browseForDeviceName( PalmAppGlobals* globals, ConnsDlgState* state ) { char buf[32]; XP_BtAddr btAddr; if ( palm_bt_browse_device( globals, &btAddr, buf, sizeof( buf ) ) ) { setFieldStr( XW_CONNS_BT_HOSTFIELD_ID, buf ); XP_MEMCPY( &state->btAddr, &btAddr, sizeof(state->btAddr) ); LOG_HEX( &state->btAddr, sizeof(state->btAddr), __FUNCTION__ ); } } /* browseForDeviceName */ #endif Boolean ConnsFormHandleEvent( EventPtr event ) { Boolean result; PalmAppGlobals* globals; ConnsDlgState* state; FormPtr form; XP_S16 chosen; CALLBACK_PROLOGUE(); globals = getFormRefcon(); state = globals->connState; if ( !state ) { state = globals->connState = XP_MALLOC( globals->mpool, sizeof(*state) ); XP_MEMSET( state, 0, sizeof(*state) ); } form = FrmGetActiveForm(); switch ( event->eType ) { case frmOpenEvent: state->serverRole = (Connectedness)globals->dlgParams[CONNS_PARAM_ROLE_INDEX]; state->addr = (CommsAddrRec*)globals->dlgParams[CONNS_PARAM_ADDR_INDEX]; state->isNewGame = globals->isNewGame; XP_MEMCPY( &state->btAddr, &state->addr->u.bt.btAddr, sizeof(state->btAddr) ); /* setup connection popup */ state->connTypesList = getActiveObjectPtr( XW_CONNS_TYPE_LIST_ID ); XP_ASSERT( state->addr->conType == COMMS_CONN_IR || state->addr->conType == COMMS_CONN_RELAY || state->addr->conType == COMMS_CONN_BT ); setSelectorFromList( XW_CONNS_TYPE_TRIGGER_ID, state->connTypesList, conTypeToSel(state->addr->conType) ); ctlsFromState( globals, state ); updateFormCtls( form, state ); case frmUpdateEvent: FrmDrawForm( form ); result = true; break; case ctlSelectEvent: result = true; switch ( event->data.ctlSelect.controlID ) { #ifdef XWFEATURE_BLUETOOTH case XW_CONNS_BT_BROWSEBUTTON_ID: browseForDeviceName( globals, state ); break; #endif case XW_CONNS_TYPE_TRIGGER_ID: if ( state->isNewGame ) { chosen = LstPopupList( state->connTypesList ); if ( chosen >= 0 ) { setSelectorFromList( XW_CONNS_TYPE_TRIGGER_ID, state->connTypesList, chosen ); state->addr->conType = selToConType( chosen ); updateFormCtls( form, state ); } } break; case XW_CONNS_OK_BUTTON_ID: if ( !state->isNewGame ) { /* do nothing; same as cancel */ } else if ( !stateFromCtls( state ) ) { beep(); break; } else { EventType eventToPost; eventToPost.eType = connsSettingChgEvent; EvtAddEventToQueue( &eventToPost ); } case XW_CONNS_CANCEL_BUTTON_ID: cleanupExit( globals ); break; } break; default: result = false; } CALLBACK_EPILOGUE(); return result; } /* ConnsFormHandleEvent */ #endif /* BEYOND_IR */