warzone2100/src/transporter.c

1901 lines
50 KiB
C

/*
This file is part of Warzone 2100.
Copyright (C) 1999-2004 Eidos Interactive
Copyright (C) 2005-2010 Warzone 2100 Project
Warzone 2100 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.
Warzone 2100 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 Warzone 2100; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
/**
* @file transporter.c
*
* Code to deal with loading/unloading, interface and flight of transporters.
*/
#include <string.h>
#include "lib/framework/frame.h"
#include "lib/framework/strres.h"
#include "lib/framework/math_ext.h"
#include "lib/widget/label.h"
#include "lib/widget/widget.h"
#include "lib/ivis_common/textdraw.h"
#include "stats.h"
#include "hci.h"
#include "intdisplay.h"
#include "objmem.h"
#include "transporter.h"
#include "group.h"
#include "display3d.h"
#include "mission.h"
#include "objects.h"
#include "display.h"
#include "lib/script/script.h"
#include "scripttabs.h"
#include "order.h"
#include "action.h"
#include "lib/gamelib/gtime.h"
#include "console.h"
#include "lib/ivis_common/bitimage.h"
#include "warcam.h"
#include "selection.h"
#include "lib/sound/audio.h"
#include "lib/sound/audio_id.h"
// FIXME Direct iVis implementation include!
#include "lib/framework/fixedpoint.h"
#include "lib/ivis_opengl/piematrix.h"
#include "mapgrid.h"
#include "visibility.h"
#include "multiplay.h"
//#define IDTRANS_FORM 9000 //The Transporter base form
#define IDTRANS_TABFORM 9001 //The Transporter tabbed form
#define IDTRANS_CLOSE 9002 //The close button icon
//#define IDTRANS_CONTENTFORM 9003 //The Transporter Contents form
#define IDTRANS_CONTABFORM 9004 //The Transporter Contents tabbed form
#define IDTRANS_CONTCLOSE 9005 //The close icon on the Contents form
//#define IDTRANS_DROIDS 9006 //The Droid base form
#define IDTRANS_DROIDTAB 9007 //The Droid tab form
#define IDTRANS_DROIDCLOSE 9008 //The close icon for the Droid form
//#define IDTRANS_LAUNCH 9010 //The Transporter Launch button
#define IDTRANS_START 9100 //The first button on the Transporter tab form
#define IDTRANS_END 9199 //The last button on the Transporter tab form
#define IDTRANS_STATSTART 9200 //The status button for the first Transporter
#define IDTRANS_STATEND 9299 //The status button for the last Transporter
#define IDTRANS_CONTSTART 9300 //The first button on the Transporter contents tab form
#define IDTRANS_CONTEND 9399 //The last button on the Transporter contents tab form
#define IDTRANS_DROIDSTART 9400 //The first button on the Droid tab form
#define IDTRANS_DROIDEND 9499 //The last button on the Droid tab form
#define IDTRANS_REPAIRBARSTART 9600 //The first repair status bar on Droid button
#define IDTRANS_REPAIRBAREND 9699 //The last repair status bar on Droid button
//#define IDTRANS_CAPACITY 9500 //The capacity label
/* Transporter screen positions */
#define TRANS_X OBJ_BACKX
#define TRANS_Y OBJ_BACKY
#define TRANS_WIDTH OBJ_BACKWIDTH
#define TRANS_HEIGHT OBJ_BACKHEIGHT
/*tabbed form screen positions */
#define TRANS_TABX OBJ_TABX
#define TRANS_TABY OBJ_TABY
#define TRANS_TABWIDTH OBJ_WIDTH
#define TRANS_TABHEIGHT OBJ_HEIGHT
/*Transported contents screen positions */
#define TRANSCONT_X STAT_X
#define TRANSCONT_Y STAT_Y
#define TRANSCONT_WIDTH STAT_WIDTH
#define TRANSCONT_HEIGHT STAT_HEIGHT
/*contents tabbed form screen positions */
#define TRANSCONT_TABX STAT_TABFORMX
#define TRANSCONT_TABY STAT_TABFORMY
#define TRANSCONT_TABWIDTH STAT_TABWIDTH
#define TRANSCONT_TABHEIGHT STAT_TABHEIGHT
/*droid form screen positions */
#define TRANSDROID_X RADTLX
#define TRANSDROID_Y STAT_Y
#define TRANSDROID_WIDTH STAT_WIDTH
#define TRANSDROID_HEIGHT STAT_HEIGHT
/*droid Tab form screen positions */
#define TRANSDROID_TABX STAT_TABFORMX
#define TRANSDROID_TABY STAT_TABFORMY
#define TRANSDROID_TABWIDTH STAT_WIDTH
#define TRANSDROID_TABHEIGHT STAT_HEIGHT
//start y position of the available droids buttons
#define AVAIL_STARTY 0
//defines how much space is on the Transporter
#define TRANSPORTER_CAPACITY 10
//They all take up the same amount of space now - AB 30/10/98
//defines how much space each sized droid takes up on the Transporter
#define LIGHT_DROID 1
#define MEDIUM_DROID 2
#define HEAVY_DROID 3
//max that can be available from home
#define MAX_DROIDS 80
/* the widget screen */
extern W_SCREEN *psWScreen;
/* Static variables */
static DROID *psCurrTransporter;
static DROID *g_psCurScriptTransporter = NULL;
static BOOL onMission;
static UDWORD g_iLaunchTime = 0;
//used for audio message for reinforcements
static BOOL bFirstTransporter;
//the tab positions of the DroidsAvail window
static UWORD objMajor = 0, objMinor = 0;
/*functions */
static BOOL intAddTransporterContents(void);
static void transporterRemoveDroid(UDWORD id);
static void setCurrentTransporter(UDWORD id);
static void intRemoveTransContentNoAnim(void);
static BOOL intAddTransButtonForm(void);
static BOOL intAddTransContentsForm(void);
static BOOL intAddDroidsAvailForm(void);
void intRemoveTransContent(void);
static UDWORD transporterSpaceRequired(DROID *psDroid);
static DROID* transInterfaceDroidList(void);
static void intTransporterAddDroid(UDWORD id);
static void intRemoveTransDroidsAvail(void);
static void intRemoveTransDroidsAvailNoAnim(void);
static BOOL _intRefreshTransporter(void);
static BOOL _intAddTransporter(DROID *psSelected, BOOL offWorld);
static void _intProcessTransporter(UDWORD id);
//initialises Transporter variables
void initTransporters(void)
{
onMission = false;
psCurrTransporter = NULL;
}
// Call to refresh the transporter screen, ie when a droids boards it.
//
BOOL intRefreshTransporter(void)
{
//printf("intRefreshTransporter\n");
return _intRefreshTransporter();
}
static BOOL _intRefreshTransporter(void)
{
// Is the transporter screen up?
if( (intMode == INT_TRANSPORTER) &&
(widgGetFromID(psWScreen,IDTRANS_FORM) != NULL))
{
BOOL Ret;
// Refresh it by re-adding it.
Ret = intAddTransporter(psCurrTransporter,onMission);
intMode = INT_TRANSPORTER;
return Ret;
}
return true;
}
BOOL intAddTransporter(DROID *psSelected, BOOL offWorld)
{
return(_intAddTransporter(psSelected,offWorld));
}
/*Add the Transporter Interface*/
static BOOL _intAddTransporter(DROID *psSelected, BOOL offWorld)
{
W_FORMINIT sFormInit;
W_BUTINIT sButInit;
BOOL Animate = true;
onMission = offWorld;
psCurrTransporter = psSelected;
/*if transporter has died - close the interface - this can only happen in
multiPlayer where the transporter can be killed*/
if (bMultiPlayer)
{
if (psCurrTransporter && isDead((BASE_OBJECT *)psCurrTransporter))
{
intRemoveTransNoAnim();
return true;
}
}
// Add the main Transporter form
// Is the form already up?
if(widgGetFromID(psWScreen,IDTRANS_FORM) != NULL)
{
intRemoveTransNoAnim();
Animate = false;
}
if(intIsRefreshing()) {
Animate = false;
}
memset(&sFormInit, 0, sizeof(W_FORMINIT));
sFormInit.formID = 0;
sFormInit.id = IDTRANS_FORM;
sFormInit.style = WFORM_PLAIN;
sFormInit.x = (SWORD)TRANS_X;
sFormInit.y = (SWORD)TRANS_Y;
sFormInit.width = TRANS_WIDTH;
sFormInit.height = TRANS_HEIGHT;
// If the window was closed then do open animation.
if(Animate)
{
sFormInit.pDisplay = intOpenPlainForm;
sFormInit.disableChildren = true;
}
else
{
// otherwise just recreate it.
sFormInit.pDisplay = intDisplayPlainForm;
}
if (!widgAddForm(psWScreen, &sFormInit))
{
return false;
}
/* Add the close button */
memset(&sButInit, 0, sizeof(W_BUTINIT));
sButInit.formID = IDTRANS_FORM;
sButInit.id = IDTRANS_CLOSE;
sButInit.style = WBUT_PLAIN;
sButInit.x = TRANS_WIDTH - CLOSE_WIDTH;
sButInit.y = 0;
sButInit.width = CLOSE_WIDTH;
sButInit.height = CLOSE_HEIGHT;
sButInit.pTip = _("Close");
sButInit.FontID = font_regular;
sButInit.pDisplay = intDisplayImageHilight;
sButInit.UserData = PACKDWORD_TRI(0,IMAGE_CLOSEHILIGHT , IMAGE_CLOSE);
if (!widgAddButton(psWScreen, &sButInit))
{
return false;
}
if (!intAddTransButtonForm())
{
return false;
}
// Add the Transporter Contents form (and buttons)
if (!intAddTransporterContents())
{
return false;
}
//if on a mission - add the Droids back at home base form
if (onMission)
{
if (!intAddDroidsAvailForm())
{
return false;
}
}
return true;
}
// Add the main Transporter Contents Interface
BOOL intAddTransporterContents(void)
{
W_FORMINIT sFormInit;
W_BUTINIT sButInit;
W_FORMINIT sButFInit;
W_LABINIT sLabInit;
BOOL Animate = true;
BOOL AlreadyUp = false;
// Is the form already up?
if(widgGetFromID(psWScreen,IDTRANS_CONTENTFORM) != NULL)
{
intRemoveTransContentNoAnim();
Animate = false;
AlreadyUp = true;
}
if(intIsRefreshing()) {
Animate = false;
}
memset(&sFormInit, 0, sizeof(W_FORMINIT));
sFormInit.formID = 0;
sFormInit.id = IDTRANS_CONTENTFORM;
sFormInit.style = WFORM_PLAIN;
sFormInit.x = (SWORD)TRANSCONT_X;
sFormInit.y = (SWORD)TRANSCONT_Y;
sFormInit.width = TRANSCONT_WIDTH;
sFormInit.height = TRANSCONT_HEIGHT;
// If the window was closed then do open animation.
if(Animate)
{
sFormInit.pDisplay = intOpenPlainForm;
sFormInit.disableChildren = true;
}
else
{
// otherwise just recreate it.
sFormInit.pDisplay = intDisplayPlainForm;
}
if (!widgAddForm(psWScreen, &sFormInit))
{
return false;
}
/* Add the close button */
memset(&sButInit, 0, sizeof(W_BUTINIT));
sButInit.formID = IDTRANS_CONTENTFORM;
sButInit.id = IDTRANS_CONTCLOSE;
sButInit.style = WBUT_PLAIN;
sButInit.x = STAT_WIDTH - CLOSE_WIDTH;
sButInit.y = 0;
sButInit.width = CLOSE_WIDTH;
sButInit.height = CLOSE_HEIGHT;
sButInit.pTip = _("Close");
sButInit.FontID = font_regular;
sButInit.pDisplay = intDisplayImageHilight;
sButInit.UserData = PACKDWORD_TRI(0,IMAGE_CLOSEHILIGHT , IMAGE_CLOSE);
if (!widgAddButton(psWScreen, &sButInit))
{
return false;
}
if (bMultiPlayer)
{
//add the capacity label
memset(&sLabInit,0,sizeof(W_LABINIT));
sLabInit.formID = IDTRANS_CONTENTFORM;
sLabInit.id = IDTRANS_CAPACITY;
sLabInit.style = WLAB_PLAIN;
sLabInit.x = (SWORD)sButInit.x -40;
sLabInit.y = 0;
sLabInit.width = 16;
sLabInit.height = 16;
sLabInit.pText = "00/10";
sLabInit.FontID = font_regular;
sLabInit.pCallback = intUpdateTransCapacity;
if (!widgAddLabel(psWScreen, &sLabInit))
{
return false;
}
}
//add the Launch button if on a mission
if (onMission)
{
memset(&sButFInit, 0, sizeof(W_FORMINIT));
sButFInit.formID = IDTRANS_CONTENTFORM;
sButFInit.id = IDTRANS_LAUNCH;
sButFInit.style = WFORM_CLICKABLE | WFORM_NOCLICKMOVE;
sButFInit.x = OBJ_STARTX;
sButFInit.y = (UWORD)(STAT_SLDY - 1);
sButFInit.width = iV_GetImageWidth(IntImages,IMAGE_LAUNCHUP);
sButFInit.height = iV_GetImageHeight(IntImages,IMAGE_LAUNCHUP);
sButFInit.pTip = _("Launch Transport");
//sButInit.pText = "Launch";
// sButFInit.FontID = font_regular;
sButFInit.pDisplay = intDisplayImageHilight;
sButFInit.UserData = PACKDWORD_TRI(0,IMAGE_LAUNCHDOWN,IMAGE_LAUNCHUP);
if (!widgAddForm(psWScreen, &sButFInit))
{
return false;
}
}
if (!intAddTransContentsForm())
{
return false;
}
return true;
}
/*This is used to display the transporter button and capacity when at the home base ONLY*/
BOOL intAddTransporterLaunch(DROID *psDroid)
{
//W_BUTINIT sButInit;
W_FORMINIT sButInit; //needs to be a clickable form now
W_LABINIT sLabInit;
UDWORD capacity;
DROID *psCurr, *psNext;
if (bMultiPlayer)
{
return true;
}
//do this first so that if the interface is already up it syncs with this transporter
//set up the static transporter
psCurrTransporter = psDroid;
//check the button is not already up
if(widgGetFromID(psWScreen,IDTRANS_LAUNCH) != NULL)
{
return true;
}
memset(&sButInit, 0, sizeof(W_FORMINIT));
sButInit.formID = 0;
sButInit.id = IDTRANS_LAUNCH;
sButInit.style = WFORM_CLICKABLE | WFORM_NOCLICKMOVE;
sButInit.x = RET_X;
sButInit.y = (SWORD)TIMER_Y;
sButInit.width = (UWORD)(10 + iV_GetImageWidth(IntImages,IMAGE_LAUNCHUP));
sButInit.height = iV_GetImageHeight(IntImages,IMAGE_LAUNCHUP);
sButInit.pTip = _("Launch Transport");
sButInit.pDisplay = intDisplayImageHilight;
sButInit.UserData = PACKDWORD_TRI(0,IMAGE_LAUNCHDOWN,IMAGE_LAUNCHUP);
if (!widgAddForm(psWScreen, &sButInit))
{
return false;
}
//add the capacity label
memset(&sLabInit,0,sizeof(W_LABINIT));
sLabInit.formID = IDTRANS_LAUNCH;
sLabInit.id = IDTRANS_CAPACITY;
sLabInit.style = WLAB_PLAIN;
sLabInit.x = (SWORD)(sButInit.x + 20);
sLabInit.y = 0;
sLabInit.width = 16;
sLabInit.height = 16;
sLabInit.pText = "00/10";
sLabInit.FontID = font_regular;
sLabInit.pCallback = intUpdateTransCapacity;
if (!widgAddLabel(psWScreen, &sLabInit))
{
return false;
}
//when full flash the transporter button
if (psCurrTransporter && psCurrTransporter->psGroup)
{
capacity = TRANSPORTER_CAPACITY;
for (psCurr = psCurrTransporter->psGroup->psList; psCurr != NULL;
psCurr = psNext)
{
psNext = psCurr->psGrpNext;
if (psCurr != psCurrTransporter)
{
capacity -= transporterSpaceRequired(psCurr);
}
}
if (capacity <= 0)
{
flashMissionButton(IDTRANS_LAUNCH);
}
}
return true;
}
/* Remove the Transporter Launch widget from the screen*/
void intRemoveTransporterLaunch(void)
{
if(widgGetFromID(psWScreen,IDTRANS_LAUNCH) != NULL)
{
widgDelete(psWScreen, IDTRANS_LAUNCH);
}
}
/* Add the Transporter Button form */
BOOL intAddTransButtonForm(void)
{
W_FORMINIT sFormInit;
W_FORMINIT sBFormInit, sBFormInit2;
UDWORD numButtons, i;
SDWORD BufferID;
DROID *psDroid;
/* Add the button form */
memset(&sFormInit, 0, sizeof(W_FORMINIT));
sFormInit.formID = IDTRANS_FORM;
sFormInit.id = IDTRANS_TABFORM;
sFormInit.style = WFORM_TABBED;
sFormInit.width = TRANS_TABWIDTH;
sFormInit.height = TRANS_TABHEIGHT;
sFormInit.x = TRANS_TABX;
sFormInit.y = TRANS_TABY;
sFormInit.majorPos = WFORM_TABTOP;
sFormInit.minorPos = WFORM_TABNONE;
sFormInit.majorSize = OBJ_TABWIDTH;
sFormInit.majorOffset = OBJ_TABOFFSET;
sFormInit.tabVertOffset = (OBJ_TABHEIGHT/2);
sFormInit.tabMajorThickness = OBJ_TABHEIGHT;
numButtons = 0;
/*work out the number of buttons */
for(psDroid = transInterfaceDroidList(); psDroid; psDroid = psDroid->psNext)
{
//only interested in Transporter droids
if ( psDroid->droidType == DROID_TRANSPORTER &&
(psDroid->action != DACTION_TRANSPORTOUT &&
psDroid->action != DACTION_TRANSPORTIN ) )
{
//set the first Transporter to be the current one if not already set
if (psCurrTransporter == NULL)
{
psCurrTransporter = psDroid;
}
numButtons++;
}
}
//set the number of tabs required
sFormInit.numMajor = numForms((OBJ_BUTWIDTH + OBJ_GAP) * numButtons,
OBJ_WIDTH - OBJ_GAP);
sFormInit.pUserData = &StandardTab;
sFormInit.pTabDisplay = intDisplayTab;
if (sFormInit.numMajor > MAX_TAB_STD_SHOWN)
{ // we do NOT use smallTab icons here, so be safe and only display max # of
// standard sized tab icons.
sFormInit.numMajor = MAX_TAB_STD_SHOWN;
}
//set minor tabs to 1
for (i=0; i< sFormInit.numMajor; i++)
{
sFormInit.aNumMinors[i] = 1;
}
if (!widgAddForm(psWScreen, &sFormInit))
{
return false;
}
/* Add the transporter and status buttons */
memset(&sBFormInit, 0, sizeof(W_FORMINIT));
memset(&sBFormInit2, 0, sizeof(W_FORMINIT));
sBFormInit.formID = IDTRANS_TABFORM;
sBFormInit.id = IDTRANS_START;
sBFormInit.majorID = 0;
sBFormInit.minorID = 0;
sBFormInit.style = WFORM_CLICKABLE;
sBFormInit.x = OBJ_STARTX;
sBFormInit.y = OBJ_STARTY;
sBFormInit.width = OBJ_BUTWIDTH;
sBFormInit.height = OBJ_BUTHEIGHT;
memcpy(&sBFormInit2,&sBFormInit,sizeof(W_FORMINIT));
sBFormInit2.id = IDTRANS_STATSTART;
sBFormInit2.y = OBJ_STATSTARTY;
ClearObjectBuffers();
ClearTopicBuffers();
//add each button
for(psDroid = transInterfaceDroidList(); psDroid; psDroid = psDroid->psNext)
{
if ( psDroid->droidType == DROID_TRANSPORTER &&
(psDroid->action != DACTION_TRANSPORTOUT &&
psDroid->action != DACTION_TRANSPORTIN ) )
{
/* Set the tip and add the button */
sBFormInit.pTip = droidGetName(psDroid);
BufferID = sBFormInit.id-IDTRANS_START;
ASSERT( BufferID < NUM_TOPICBUFFERS,"BufferID > NUM_TOPICBUFFERS" );
ClearTopicButtonBuffer(BufferID);
RENDERBUTTON_INUSE(&TopicBuffers[BufferID]);
TopicBuffers[BufferID].Data = (void*)psDroid;
sBFormInit.pUserData = &TopicBuffers[BufferID];
sBFormInit.pDisplay = intDisplayObjectButton;
if (!widgAddForm(psWScreen, &sBFormInit))
{
return false;
}
/* if the current droid matches psCurrTransporter lock the button */
if (psDroid == psCurrTransporter)
{
widgSetButtonState(psWScreen, sBFormInit.id, WBUT_LOCK);
widgSetTabs(psWScreen, IDTRANS_TABFORM, sBFormInit.majorID, 0);
}
//now do status button
sBFormInit2.pTip = NULL;
BufferID = (sBFormInit2.id-IDTRANS_STATSTART)*2+1;
ASSERT( BufferID < NUM_OBJECTBUFFERS,"BufferID > NUM_OBJECTBUFFERS" );
ClearObjectButtonBuffer(BufferID);
RENDERBUTTON_INUSE(&ObjectBuffers[BufferID]);
sBFormInit2.pUserData = &ObjectBuffers[BufferID];
sBFormInit2.pDisplay = intDisplayStatusButton;
if (!widgAddForm(psWScreen, &sBFormInit2))
{
return false;
}
/* Update the init struct for the next buttons */
sBFormInit.id += 1;
ASSERT( sBFormInit.id < IDTRANS_END,"Too many Transporter buttons" );
sBFormInit.x += OBJ_BUTWIDTH + OBJ_GAP;
if (sBFormInit.x + OBJ_BUTWIDTH + OBJ_GAP > OBJ_WIDTH)
{
sBFormInit.x = OBJ_STARTX;
sBFormInit.majorID += 1;
}
sBFormInit2.id += 1;
ASSERT( sBFormInit2.id < IDTRANS_STATEND,"Too many Transporter status buttons" );
sBFormInit2.x += OBJ_BUTWIDTH + OBJ_GAP;
if (sBFormInit2.x + OBJ_BUTWIDTH + OBJ_GAP > OBJ_WIDTH)
{
sBFormInit2.x = OBJ_STARTX;
sBFormInit2.majorID += 1;
}
}
}
return true;
}
/* Add the Transporter Contents form */
BOOL intAddTransContentsForm(void)
{
W_FORMINIT sFormInit;
W_FORMINIT sBFormInit;
UDWORD numButtons, i;
SDWORD BufferID;
DROID *psDroid, *psNext;
/* Add the contents form */
memset(&sFormInit, 0, sizeof(W_FORMINIT));
sFormInit.formID = IDTRANS_CONTENTFORM;
sFormInit.id = IDTRANS_CONTABFORM;
sFormInit.style = WFORM_TABBED;
sFormInit.width = TRANSCONT_WIDTH;
sFormInit.height = TRANSCONT_HEIGHT;
sFormInit.x = TRANSCONT_TABX;
sFormInit.y = TRANSCONT_TABY;
sFormInit.majorPos = WFORM_TABTOP;
sFormInit.minorPos = WFORM_TABNONE;
sFormInit.majorSize = OBJ_TABWIDTH;
sFormInit.majorOffset = OBJ_TABOFFSET;
sFormInit.tabVertOffset = (OBJ_TABHEIGHT/2);
sFormInit.tabMajorThickness = OBJ_TABHEIGHT;
numButtons = TRANSPORTER_CAPACITY;
//set the number of tabs required
//sFormInit.numMajor = numForms((OBJ_BUTWIDTH + OBJ_GAP) * numButtons,
// OBJ_WIDTH - OBJ_GAP);
// TABFIXME: Looks like 10 units is max for this?
sFormInit.numMajor = 1;
//set minor tabs to 1
for (i=0; i< sFormInit.numMajor; i++)
{
sFormInit.aNumMinors[i] = 1;
}
sFormInit.pUserData = &StandardTab;
sFormInit.pTabDisplay = intDisplayTab;
if (!widgAddForm(psWScreen, &sFormInit))
{
return false;
}
/* Add the transporter contents buttons */
memset(&sBFormInit, 0, sizeof(W_FORMINIT));
sBFormInit.formID = IDTRANS_CONTABFORM;
sBFormInit.id = IDTRANS_CONTSTART;
sBFormInit.majorID = 0;
sBFormInit.minorID = 0;
sBFormInit.style = WFORM_CLICKABLE;
sBFormInit.x = OBJ_STARTX;
sBFormInit.y = OBJ_STARTY - OBJ_BUTHEIGHT - OBJ_GAP;
sBFormInit.width = OBJ_BUTWIDTH;
sBFormInit.height = OBJ_BUTHEIGHT;
ClearStatBuffers();
//add each button
if (psCurrTransporter != NULL)
{
for (psDroid = psCurrTransporter->psGroup->psList; psDroid != NULL && psDroid !=
psCurrTransporter; psDroid = psNext)
{
psNext = psDroid->psGrpNext;
/* Set the tip and add the button */
sBFormInit.pTip = droidGetName(psDroid);
BufferID = GetStatBuffer();
ASSERT( BufferID >= 0,"Unable to acquire stat buffer." );
RENDERBUTTON_INUSE(&StatBuffers[BufferID]);
StatBuffers[BufferID].Data = (void*)psDroid;
sBFormInit.pUserData = &StatBuffers[BufferID];
sBFormInit.pDisplay = intDisplayTransportButton;
if (!widgAddForm(psWScreen, &sBFormInit))
{
return false;
}
/* Update the init struct for the next button */
sBFormInit.id += 1;
ASSERT( sBFormInit.id < IDTRANS_CONTEND,"Too many Transporter Droid buttons" );
sBFormInit.x += OBJ_BUTWIDTH + OBJ_GAP;
if (sBFormInit.x + OBJ_BUTWIDTH + OBJ_GAP > TRANSCONT_WIDTH)
{
sBFormInit.x = OBJ_STARTX;
sBFormInit.y += OBJ_BUTHEIGHT + OBJ_GAP;
}
if (sBFormInit.y + OBJ_BUTHEIGHT + OBJ_GAP > TRANSCONT_HEIGHT)
{
sBFormInit.y = OBJ_STARTY;
sBFormInit.majorID += 1;
}
}
}
return true;
}
/* Add the Droids back at home form */
BOOL intAddDroidsAvailForm(void)
{
W_FORMINIT sFormInit;
W_BUTINIT sButInit;
W_FORMINIT sBFormInit;
W_BARINIT sBarInit;
UDWORD numButtons, i, butPerForm;
SDWORD BufferID;
DROID *psDroid;
BOOL Animate = true;
// Is the form already up?
if(widgGetFromID(psWScreen,IDTRANS_DROIDS) != NULL)
{
intRemoveTransDroidsAvailNoAnim();
Animate = false;
}
if(intIsRefreshing()) {
Animate = false;
}
/* Add the droids available form */
memset(&sFormInit, 0, sizeof(W_FORMINIT));
sFormInit.formID = 0;
sFormInit.id = IDTRANS_DROIDS;
sFormInit.style = WFORM_PLAIN;
sFormInit.width = TRANSDROID_WIDTH;
sFormInit.height = TRANSDROID_HEIGHT;
sFormInit.x = (SWORD)TRANSDROID_X;
sFormInit.y = (SWORD)TRANSDROID_Y;
// If the window was closed then do open animation.
if(Animate)
{
sFormInit.pDisplay = intOpenPlainForm;
sFormInit.disableChildren = true;
}
else
{
// otherwise just recreate it.
sFormInit.pDisplay = intDisplayPlainForm;
}
if (!widgAddForm(psWScreen, &sFormInit))
{
return false;
}
/* Add the close button */
memset(&sButInit, 0, sizeof(W_BUTINIT));
sButInit.formID = IDTRANS_DROIDS;
sButInit.id = IDTRANS_DROIDCLOSE;
sButInit.style = WBUT_PLAIN;
sButInit.x = TRANSDROID_WIDTH - CLOSE_WIDTH;
sButInit.y = 0;
sButInit.width = CLOSE_WIDTH;
sButInit.height = CLOSE_HEIGHT;
sButInit.pTip = _("Close");
sButInit.FontID = font_regular;
sButInit.pDisplay = intDisplayImageHilight;
sButInit.UserData = PACKDWORD_TRI(0,IMAGE_CLOSEHILIGHT , IMAGE_CLOSE);
if (!widgAddButton(psWScreen, &sButInit))
{
return false;
}
//now add the tabbed droids available form
memset(&sFormInit, 0, sizeof(W_FORMINIT));
sFormInit.formID = IDTRANS_DROIDS;
sFormInit.id = IDTRANS_DROIDTAB;
sFormInit.style = WFORM_TABBED;
sFormInit.width = TRANSDROID_TABWIDTH;
sFormInit.height = TRANSDROID_TABHEIGHT;
sFormInit.x = TRANSDROID_TABX;
sFormInit.y = TRANSDROID_TABY;
sFormInit.majorPos = WFORM_TABTOP;
sFormInit.minorPos = WFORM_TABNONE;
sFormInit.majorSize = (OBJ_TABWIDTH/2);
sFormInit.majorOffset = OBJ_TABOFFSET;
sFormInit.tabVertOffset = (OBJ_TABHEIGHT/2);
sFormInit.tabMajorThickness = OBJ_TABHEIGHT;
sFormInit.tabMajorGap = OBJ_TABOFFSET;
//calc num buttons
numButtons = 0;
//look through the list of droids that were built before the mission
for(psDroid = mission.apsDroidLists[selectedPlayer]; psDroid; psDroid =
psDroid->psNext)
{
//ignore any Transporters!
if (psDroid->droidType != DROID_TRANSPORTER)
{
numButtons++;
}
//quit when reached max can cope with
if (numButtons == MAX_DROIDS)
{
break;
}
}
butPerForm = ((TRANSDROID_TABWIDTH - OBJ_GAP) /
(OBJ_BUTWIDTH + OBJ_GAP)) *
((TRANSDROID_TABHEIGHT - OBJ_GAP) /
(OBJ_BUTHEIGHT + OBJ_GAP));
sFormInit.numMajor = numForms(numButtons, butPerForm);
if (sFormInit.numMajor > MAX_TAB_SMALL_SHOWN)
{ // we DO use smallTab icons here, so be safe and only display max # of
// small sized tab icons. No scrolltabs here.
sFormInit.numMajor = MAX_TAB_SMALL_SHOWN;
}
//set minor tabs to 1
for (i=0; i< sFormInit.numMajor; i++)
{
sFormInit.aNumMinors[i] = 1;
}
sFormInit.pUserData = &SmallTab;
sFormInit.pTabDisplay = intDisplayTab;
if (!widgAddForm(psWScreen, &sFormInit))
{
return false;
}
/* Add the droids available buttons */
memset(&sBFormInit, 0, sizeof(W_FORMINIT));
sBFormInit.formID = IDTRANS_DROIDTAB;
sBFormInit.id = IDTRANS_DROIDSTART;
sBFormInit.majorID = 0;
sBFormInit.minorID = 0;
sBFormInit.style = WFORM_CLICKABLE;
sBFormInit.x = OBJ_STARTX;
sBFormInit.y = AVAIL_STARTY;
sBFormInit.width = OBJ_BUTWIDTH;
sBFormInit.height = OBJ_BUTHEIGHT;
ClearSystem0Buffers();
/* Add the state of repair bar for each droid*/
memset(&sBarInit, 0, sizeof(W_BARINIT));
sBarInit.id = IDTRANS_REPAIRBARSTART;
sBarInit.style = WBAR_PLAIN;
sBarInit.orientation = WBAR_LEFT;
sBarInit.x = STAT_TIMEBARX;
sBarInit.y = STAT_TIMEBARY;
sBarInit.width = STAT_PROGBARWIDTH;
sBarInit.height = STAT_PROGBARHEIGHT;
sBarInit.size = 50;
sBarInit.sCol = WZCOL_ACTION_PROGRESS_BAR_MAJOR;
sBarInit.sMinorCol = WZCOL_ACTION_PROGRESS_BAR_MINOR;
//add droids built before the mission
for (psDroid = mission.apsDroidLists[selectedPlayer]; psDroid != NULL;
psDroid = psDroid->psNext)
{
//stop adding the buttons once MAX_DROIDS has been reached
if (sBFormInit.id == (IDTRANS_DROIDSTART + MAX_DROIDS))
{
break;
}
//don't add Transporter Droids!
if (psDroid->droidType != DROID_TRANSPORTER)
{
/* Set the tip and add the button */
// sBFormInit.pTip = psDroid->pName;
sBFormInit.pTip = droidGetName(psDroid);
BufferID = GetSystem0Buffer();
ASSERT( BufferID >= 0,"Unable to acquire stat buffer." );
RENDERBUTTON_INUSE(&System0Buffers[BufferID]);
System0Buffers[BufferID].Data = (void*)psDroid;
sBFormInit.pUserData = &System0Buffers[BufferID];
sBFormInit.pDisplay = intDisplayTransportButton;
if (!widgAddForm(psWScreen, &sBFormInit))
{
return false;
}
//add bar to indicate stare of repair
sBarInit.size = (UWORD) PERCENT(psDroid->body, psDroid->originalBody);
if(sBarInit.size > 100)
{
sBarInit.size = 100;
}
sBarInit.formID = sBFormInit.id;
//sBarInit.iRange = TBAR_MAX_REPAIR;
if (!widgAddBarGraph(psWScreen, &sBarInit))
{
return false;
}
/* Update the init struct for the next button */
sBFormInit.id += 1;
ASSERT( sBFormInit.id < IDTRANS_DROIDEND,"Too many Droids Built buttons" );
sBFormInit.x += OBJ_BUTWIDTH + OBJ_GAP;
if (sBFormInit.x + OBJ_BUTWIDTH + OBJ_GAP > TRANSDROID_TABWIDTH)
{
sBFormInit.x = OBJ_STARTX;
sBFormInit.y += OBJ_BUTHEIGHT + OBJ_GAP;
}
if (sBFormInit.y + OBJ_BUTHEIGHT + OBJ_GAP > TRANSDROID_TABHEIGHT)
{
sBFormInit.y = AVAIL_STARTY;
sBFormInit.majorID += 1;
}
//and bar
sBarInit.id += 1;
}
}
//reset which tab we were on
if (objMajor > (UWORD)(sFormInit.numMajor - 1))
{
//set to last if have lost a tab
widgSetTabs(psWScreen, IDTRANS_DROIDTAB, (UWORD)(sFormInit.numMajor-1), objMinor);
}
else
{
//set to same tab we were on previously
widgSetTabs(psWScreen, IDTRANS_DROIDTAB, objMajor, objMinor);
}
return true;
}
/*calculates how much space is remaining on the transporter - allows droids to take
up different amount depending on their body size - currently all are set to one!*/
UDWORD calcRemainingCapacity(DROID *psTransporter)
{
SDWORD capacity = TRANSPORTER_CAPACITY;
DROID *psDroid,*psNext;
// If it's dead then just return 0.
if (isDead((BASE_OBJECT *)psTransporter))
{
return 0;
}
for (psDroid = psTransporter->psGroup->psList; psDroid != NULL && psDroid !=
psTransporter; psDroid = psNext)
{
psNext = psDroid->psGrpNext;
capacity -= transporterSpaceRequired(psDroid);
}
if(capacity < 0) capacity = 0;
return (UDWORD)capacity;
}
bool transporterIsEmpty(const DROID* psTransporter)
{
ASSERT(psTransporter->droidType == DROID_TRANSPORTER, "Non-transporter droid given");
// Assume dead droids and non-transporter droids to be empty
return (isDead((const BASE_OBJECT*)psTransporter)
|| psTransporter->droidType != DROID_TRANSPORTER
|| psTransporter->psGroup->psList == NULL
|| psTransporter->psGroup->psList == psTransporter);
}
static void intSetTransCapacityLabel(char *Label)
{
UDWORD capacity = TRANSPORTER_CAPACITY;
if (psCurrTransporter)
{
capacity = calcRemainingCapacity(psCurrTransporter);
//change round the way the remaining capacity is displayed - show 0/10 when empty now
capacity = TRANSPORTER_CAPACITY - capacity;
Label[0] = (UBYTE)('0'+capacity / 10);
Label[1] = (UBYTE)('0'+capacity % 10);
}
}
/*updates the capacity of the current Transporter*/
void intUpdateTransCapacity(WIDGET *psWidget, W_CONTEXT *psContext)
{
W_LABEL *Label = (W_LABEL*)psWidget;
intSetTransCapacityLabel(Label->aText);
}
/* Process return codes from the Transporter Screen*/
void intProcessTransporter(UDWORD id)
{
_intProcessTransporter(id);
}
static void _intProcessTransporter(UDWORD id)
{
if (id >= IDTRANS_START && id <= IDTRANS_END)
{
/* A Transporter button has been pressed */
setCurrentTransporter(id);
/*refresh the Contents list */
intAddTransporterContents();
}
else if (id >= IDTRANS_CONTSTART && id <= IDTRANS_CONTEND)
{
//got to have a current transporter for this to work - and can't be flying
if (psCurrTransporter != NULL && !transporterFlying(psCurrTransporter))
{
transporterRemoveDroid(id);
/*refresh the Contents list */
intAddTransporterContents();
if (onMission)
{
/*refresh the Avail list */
intAddDroidsAvailForm();
}
}
}
else if (id == IDTRANS_CLOSE)
{
intRemoveTransContent();
intRemoveTrans();
psCurrTransporter = NULL;
}
else if (id == IDTRANS_CONTCLOSE)
{
intRemoveTransContent();
}
else if (id == IDTRANS_DROIDCLOSE)
{
intRemoveTransDroidsAvail();
}
else if (id >= IDTRANS_DROIDSTART && id <= IDTRANS_DROIDEND)
{
//got to have a current transporter for this to work - and can't be flying
if (psCurrTransporter != NULL && !transporterFlying(psCurrTransporter))
{
intTransporterAddDroid(id);
/*don't need to explicitly refresh here since intRefreshScreen()
is called by intTransporterAddDroid()*/
/*refresh the Contents list */
//intAddTransporterContents();
/*refresh the Avail list */
//intAddDroidsAvailForm();
}
}
// Process form tab clicks.
else if (id == IDTRANS_TABFORM)
{
//If tab clicked on Transporter screen then refresh rendered buttons.
RefreshObjectButtons();
RefreshTopicButtons();
}
else if (id == IDTRANS_CONTABFORM)
{
//If tab clicked on Transporter Contents screen then refresh rendered buttons.
RefreshStatsButtons();
}
else if (id == IDTRANS_DROIDTAB)
{
//If tab clicked on Droids Available screen then refresh rendered buttons.
RefreshSystem0Buttons();
}
}
/* Remove the Transporter widgets from the screen */
void intRemoveTrans(void)
{
W_TABFORM *Form;
// Start the window close animation.
Form = (W_TABFORM*)widgGetFromID(psWScreen,IDTRANS_FORM);
if (Form)
{
Form->display = intClosePlainForm;
Form->disableChildren = true;
Form->pUserData = NULL; // Used to signal when the close anim has finished.
ClosingTrans = true;
}
intRemoveTransContent();
intRemoveTransDroidsAvail();
intMode = INT_NORMAL;
}
/* Remove the Transporter Content widgets from the screen w/o animation!*/
void intRemoveTransNoAnim(void)
{
//remove main screen
widgDelete(psWScreen, IDTRANS_FORM);
intRemoveTransContentNoAnim();
intRemoveTransDroidsAvailNoAnim();
intMode = INT_NORMAL;
}
/* Remove the Transporter Content widgets from the screen */
void intRemoveTransContent(void)
{
W_TABFORM *Form;
// Start the window close animation.
Form = (W_TABFORM*)widgGetFromID(psWScreen,IDTRANS_CONTENTFORM);
if (Form)
{
Form->display = intClosePlainForm;
Form->disableChildren = true;
Form->pUserData = NULL; // Used to signal when the close anim has finished.
ClosingTransCont = true;
}
}
/* Remove the Transporter Content widgets from the screen w/o animation!*/
void intRemoveTransContentNoAnim(void)
{
//remove main screen
widgDelete(psWScreen, IDTRANS_CONTENTFORM);
}
/* Remove the Transporter Droids Avail widgets from the screen */
void intRemoveTransDroidsAvail(void)
{
W_TABFORM *Form;
// Start the window close animation.
Form = (W_TABFORM*)widgGetFromID(psWScreen,IDTRANS_DROIDS);
if (Form)
{
Form->display = intClosePlainForm;
Form->disableChildren = true;
Form->pUserData = NULL; // Used to signal when the close anim has finished.
ClosingTransDroids = true;
//remember which tab we were on
widgGetTabs(psWScreen, IDTRANS_DROIDTAB, &objMajor, &objMinor);
}
}
/* Remove the Transporter Droids Avail widgets from the screen w/o animation!*/
void intRemoveTransDroidsAvailNoAnim(void)
{
if (widgGetFromID(psWScreen,IDTRANS_DROIDS) != NULL)
{
//remember which tab we were on
widgGetTabs(psWScreen, IDTRANS_DROIDTAB, &objMajor, &objMinor);
//remove main screen
widgDelete(psWScreen, IDTRANS_DROIDS);
}
}
/*sets psCurrTransporter */
void setCurrentTransporter(UDWORD id)
{
DROID *psDroid;
UDWORD currID;
psCurrTransporter = NULL;
currID = IDTRANS_START;
//loop thru all the droids to find the selected one
for (psDroid = transInterfaceDroidList(); psDroid != NULL; psDroid =
psDroid->psNext)
{
if ( psDroid->droidType == DROID_TRANSPORTER &&
(psDroid->action != DACTION_TRANSPORTOUT &&
psDroid->action != DACTION_TRANSPORTIN ) )
{
if (currID == id)
{
break;
}
currID++;
}
}
if (psDroid)
{
psCurrTransporter = psDroid;
//set the data for the transporter timer
widgSetUserData(psWScreen, IDTRANTIMER_DISPLAY, (void*)psCurrTransporter);
}
}
/*removes a droid from the group associated with the transporter*/
void transporterRemoveDroid(UDWORD id)
{
DROID *psDroid, *psNext;
UDWORD currID;
UDWORD droidX, droidY;
DROID_GROUP *psGroup;
ASSERT( psCurrTransporter != NULL, "transporterRemoveUnit:can't remove units" );
if (bMultiMessages)
{
// Not sure how to fix this...
debug(LOG_ERROR, "TODO: Can't unload single droids, sending order to unload all at once, because this code is so convoluted.");
// All at once is better than nothing...
orderDroidLoc(psCurrTransporter, DORDER_DISEMBARK, psCurrTransporter->pos.x, psCurrTransporter->pos.y, ModeQueue);
return;
}
currID = IDTRANS_CONTSTART;
for (psDroid = psCurrTransporter->psGroup->psList; psDroid != NULL && psDroid !=
psCurrTransporter; psDroid = psNext)
{
psNext = psDroid->psGrpNext;
if (currID == id)
{
break;
}
currID++;
}
if (psDroid)
{
/*if we're offWorld we can't pick a tile without swapping the map
pointers - can't be bothered so just do this...*/
if (onMission)
{
psDroid->pos.x = INVALID_XY;
psDroid->pos.y = INVALID_XY;
}
else
{
if (bMultiPlayer)
{
//set the units next to the transporter's current location
droidX = map_coord(psCurrTransporter->pos.x);
droidY = map_coord(psCurrTransporter->pos.y);
}
else
{
//pick a tile because save games won't remember where the droid was when it was loaded
droidX = map_coord(getLandingX(0));
droidY = map_coord(getLandingY(0));
}
if (!pickATileGen(&droidX, &droidY,LOOK_FOR_EMPTY_TILE,zonedPAT))
{
ASSERT( false, "transporterRemoveUnit: Unable to find a valid location" );
}
psDroid->pos.x = (UWORD)world_coord(droidX);
psDroid->pos.y = (UWORD)world_coord(droidY);
psDroid->pos.z = map_Height(psDroid->pos.x, psDroid->pos.y);
}
// remove it from the transporter group
grpLeave(psDroid->psGroup, psDroid);
//add it back into apsDroidLists
if (onMission)
{
//addDroid(psDroid, mission.apsBuiltDroids);
addDroid(psDroid, mission.apsDroidLists);
}
else
{
// add the droid back onto the droid list
addDroid(psDroid, apsDroidLists);
//inform all other players about that
if (bMultiMessages)
{
sendDroidDisEmbark(psDroid,psCurrTransporter);
}
}
if (psDroid->pos.x != INVALID_XY)
{
// We can update the orders now, since everyone has been
// notified of the droid exiting the transporter
updateDroidOrientation(psDroid);
}
//initialise the movement data
initDroidMovement(psDroid);
//reset droid orders
orderDroid(psDroid, DORDER_STOP, ModeImmediate);
psDroid->cluster = 0;
// check if it is a commander
if (psDroid->droidType == DROID_COMMAND)
{
if (grpCreate(&psGroup))
{
grpJoin(psGroup, psDroid);
}
}
psDroid->selected = true;
if (calcRemainingCapacity(psCurrTransporter))
{
//make sure the button isn't flashing
stopMissionButtonFlash(IDTRANS_LAUNCH);
}
}
// we want to sync with all clients *now*.
ForceDroidSync(psDroid);
}
/*adds a droid to the current transporter via the interface*/
void intTransporterAddDroid(UDWORD id)
{
DROID *psDroid, *psNext;
UDWORD currID;
ASSERT( psCurrTransporter != NULL, "intTransporterAddUnit:can't remove units" );
currID = IDTRANS_DROIDSTART;
for (psDroid = transInterfaceDroidList(); psDroid != NULL; psDroid = psNext)
{
psNext = psDroid->psNext;
if (psDroid->droidType != DROID_TRANSPORTER)
{
if (currID == id)
{
break;
}
currID++;
}
}
if (psDroid)
{
transporterAddDroid(psCurrTransporter, psDroid);
}
}
/*Adds a droid to the transporter, removing it from the world */
void transporterAddDroid(DROID *psTransporter, DROID *psDroidToAdd)
{
BOOL bDroidRemoved;
ASSERT( psTransporter != NULL, "Was passed a NULL transporter" );
ASSERT( psDroidToAdd != NULL, "Was passed a NULL droid, can't add to transporter" );
if (!psTransporter || !psDroidToAdd)
{
debug(LOG_ERROR,"We can't add the unit to the transporter!");
return;
}
/* check for space */
if (!checkTransporterSpace(psTransporter, psDroidToAdd))
{
audio_PlayTrack( ID_SOUND_BUILD_FAIL );
addConsoleMessage(_("There is not enough room in the Transport!"), DEFAULT_JUSTIFY, selectedPlayer);
return;
}
if (onMission)
{
// removing from droid mission list
bDroidRemoved = droidRemove(psDroidToAdd, mission.apsDroidLists);
}
else
{
// removing from droid list
bDroidRemoved = droidRemove(psDroidToAdd, apsDroidLists);
}
if (bDroidRemoved)
{
// adding to transporter unit's group list
grpJoin(psTransporter->psGroup, psDroidToAdd);
if (bMultiMessages)
{
//inform all other players to update their local lists
sendDroidEmbark(psDroidToAdd,psTransporter);
}
}
else
{
debug(LOG_ERROR,"droid %d not found, so nothing added to transporter!",psDroidToAdd->id);
}
if (onMission)
{
visRemoveVisibilityOffWorld((BASE_OBJECT *)psDroidToAdd);
}
else
{
visRemoveVisibility((BASE_OBJECT *)psDroidToAdd);
}
//this is called by droidRemove
//intRefreshScreen();
}
/*check to see if the droid can fit on the Transporter - return true if fits*/
BOOL checkTransporterSpace(DROID *psTransporter, DROID *psAssigned)
{
DROID *psDroid, *psNext;
UDWORD capacity;
ASSERT( psTransporter != NULL,
"checkTransporterSpace: Invalid droid pointer" );
ASSERT( psAssigned != NULL,
"checkTransporterSpace: Invalid droid pointer" );
ASSERT( psTransporter->droidType == DROID_TRANSPORTER,
"checkTransporterSpace: Droid is not a Transporter" );
ASSERT( psTransporter->psGroup != NULL,
"checkTransporterSpace: tranporter doesn't have a group" );
//work out how much space is currently left
capacity = TRANSPORTER_CAPACITY;
for (psDroid = psTransporter->psGroup->psList; psDroid != NULL && psDroid !=
psTransporter; psDroid = psNext)
{
psNext = psDroid->psGrpNext;
capacity -= transporterSpaceRequired(psDroid);
}
if (capacity >= transporterSpaceRequired(psAssigned))
{
//when full flash the transporter button
if (capacity - transporterSpaceRequired(psAssigned) == 0)
{
flashMissionButton(IDTRANS_LAUNCH);
}
return true;
}
else
{
return false;
}
}
/*returns the space the droid occupies on a transporter based on the body size*/
UDWORD transporterSpaceRequired(DROID *psDroid)
{
UDWORD size;
if (!bMultiPlayer)
{
size = LIGHT_DROID; // all droids are the same weight for SP games.
}
else
{
switch ((asBodyStats + psDroid->asBits[COMP_BODY].nStat)->size)
{
case SIZE_LIGHT:
size = LIGHT_DROID;
break;
case SIZE_MEDIUM:
size = MEDIUM_DROID;
break;
case SIZE_HEAVY:
case SIZE_SUPER_HEAVY:
size = HEAVY_DROID;
break;
default:
ASSERT( false, "transporterSpaceRequired: Unknown Droid size" );
size = 0;
break;
}
}
return size;
}
/*sets which list of droids to use for the transporter interface*/
DROID * transInterfaceDroidList(void)
{
if (onMission)
{
return mission.apsDroidLists[selectedPlayer];
}
else
{
return apsDroidLists[selectedPlayer];
}
}
UDWORD transporterGetLaunchTime( void )
{
return g_iLaunchTime;
}
void transporterSetLaunchTime(UDWORD time)
{
g_iLaunchTime = time;
}
/*launches the defined transporter to the offworld map*/
BOOL launchTransporter(DROID *psTransporter)
{
UDWORD iX, iY;
//close the interface
intResetScreen(true);
// Hmmm...Only do this if were at our home base about to go off world.
// //deselect all droids/structs etc
// clearSelection();
//this launches the mission if on homebase when the button is pressed
if (!onMission)
{
//tell the transporter to move to the new offworld location
missionGetTransporterExit( psTransporter->player, &iX, &iY );
orderDroidLoc(psTransporter, DORDER_TRANSPORTOUT, iX, iY, ModeQueue);
//g_iLaunchTime = gameTime;
transporterSetLaunchTime(gameTime);
}
//otherwise just launches the Transporter
else
{
if (psTransporter->droidType != DROID_TRANSPORTER)
{
ASSERT( false, "launchTransporter: Invalid Transporter Droid" );
return false;
}
//remove out of stored list and add to current Droid list
//removeDroid(psTransporter, mission.apsDroidLists);
//addDroid(psTransporter, apsDroidLists);
//need to put the Transporter down at a specified location
//psTransporter->pos.x = getLandingX(psTransporter->player);
//psTransporter->pos.y = getLandingY(psTransporter->player);
//unloadTransporter(psTransporter, psTransporter->pos.x, psTransporter->pos.y, false);
orderDroid(psTransporter, DORDER_TRANSPORTIN, ModeImmediate);
/* set action transporter waits for timer */
actionDroid( psTransporter, DACTION_TRANSPORTWAITTOFLYIN );
missionSetReinforcementTime( gameTime );
}
return true;
}
#define TRANSPORTOUT_TIME 4*GAME_TICKS_PER_SEC
/*checks how long the transporter has been travelling to see if it should
have arrived - returns true when there*/
BOOL updateTransporter(DROID *psTransporter)
{
ASSERT( psTransporter != NULL,
"updateTransporter: Invalid droid pointer" );
if (psTransporter->droidType != DROID_TRANSPORTER)
{
ASSERT( false, "updateTransporter: Invalid droid type" );
return true;
}
//if not moving to mission site, exit
if ( psTransporter->action != DACTION_TRANSPORTOUT &&
psTransporter->action != DACTION_TRANSPORTIN )
{
return true;
}
/*if the transporter (selectedPlayer only) is moving droids to safety and
all remaining droids are destoyed then we need to flag the end of mission
as long as we're not flying out*/
if (psTransporter->player == selectedPlayer && getDroidsToSafetyFlag()
&& psTransporter->action != DACTION_TRANSPORTOUT)
{
//if there aren't any droids left...
if (!missionDroidsRemaining(selectedPlayer))
{
// Set the Transporter to have arrived at its destination
psTransporter->action = DACTION_NONE;
//the script can call startMission for this callback for offworld missions
eventFireCallbackTrigger((TRIGGER_TYPE)CALL_START_NEXT_LEVEL);
// clear order
psTransporter->order = DORDER_NONE;
setDroidTarget(psTransporter, NULL);
psTransporter->psTarStats = NULL;
return true;
}
}
// moving to a location
// if we're coming back for more droids then we want the transporter to
// fly to edge of map before turning round again
if ( psTransporter->sMove.Status == MOVEINACTIVE ||
psTransporter->sMove.Status == MOVEHOVER ||
(psTransporter->action == DACTION_TRANSPORTOUT && !missionIsOffworld() &&
(gameTime>transporterGetLaunchTime()+TRANSPORTOUT_TIME) &&
!getDroidsToSafetyFlag() ) )
{
audio_StopObjTrack( psTransporter, ID_SOUND_BLIMP_FLIGHT );
if ( psTransporter->action == DACTION_TRANSPORTIN )
{
/* !!!! GJ Hack - should be landing audio !!!! */
audio_PlayObjDynamicTrack( psTransporter, ID_SOUND_BLIMP_TAKE_OFF, NULL );
}
//DON@T PLAY AUDIO FOR THE FIRST TRANSPORTER LOAD...AB 9/2/99
//show if selectedPlayer's transporter
//if ( onMission && psTransporter->action == DACTION_TRANSPORTIN &&
// psTransporter->player == selectedPlayer)
//changed onMission to missionForReInforcements() to cater for cam2A/cam3A - AB 4/2/99
if (!bFirstTransporter && missionForReInforcements() &&
psTransporter->action == DACTION_TRANSPORTIN &&
psTransporter->player == selectedPlayer)
{
//play reinforcements have arrived message
audio_QueueTrackPos( ID_SOUND_TRANSPORT_LANDING,
psTransporter->pos.x, psTransporter->pos.y, psTransporter->pos.z );
addConsoleMessage(_("Reinforcements landing"),LEFT_JUSTIFY,SYSTEM_MESSAGE);
//reset the data for the transporter timer
widgSetUserData(psWScreen, IDTRANTIMER_DISPLAY, (void*)NULL);
return true;
}
// Got to destination
psTransporter->action = DACTION_NONE;
//reset the flag to trigger the audio message
bFirstTransporter = false;
return true;
}
//not arrived yet...
return false;
}
//process the launch transporter button click
void processLaunchTransporter(void)
{
UDWORD capacity = TRANSPORTER_CAPACITY;
W_CLICKFORM *psForm;
//launch the Transporter
if (psCurrTransporter)
{
//check there is something on the transporter
capacity = calcRemainingCapacity(psCurrTransporter);
if (capacity != TRANSPORTER_CAPACITY)
{
//make sure the button doesn't flash once launched
stopMissionButtonFlash(IDTRANS_LAUNCH);
//disable the form so can't add any more droids into the transporter
psForm = (W_CLICKFORM*)widgGetFromID(psWScreen,IDTRANS_LAUNCH);
if (psForm)
{
formSetClickState(psForm, WBUT_LOCK);
}
//disable the form so can't add any more droids into the transporter
psForm = (W_CLICKFORM*)widgGetFromID(psWScreen,IDTRANTIMER_BUTTON);
if (psForm)
{
formSetClickState(psForm, WBUT_LOCK);
}
launchTransporter(psCurrTransporter);
//set the data for the transporter timer
widgSetUserData(psWScreen, IDTRANTIMER_DISPLAY,
(void*)psCurrTransporter);
eventFireCallbackTrigger((TRIGGER_TYPE)CALL_LAUNCH_TRANSPORTER);
}
}
}
SDWORD bobTransporterHeight( void )
{
// Because 4320/12 = 360 degrees
// this gives us a bob frequency of 4.32 seconds.
// we scale amplitude to 10 (world coordinate metric).
// we need to use 360 degrees and not 180, as otherwise
// it will not 'bounce' off the top _and_ bottom of
// it's movemment arc.
return iSinSR(gameTime, 4320, 10);
}
/*causes one of the mission buttons (Launch Button or Mission Timer) to start flashing*/
void flashMissionButton(UDWORD buttonID)
{
W_TABFORM *psForm;
//get the button from the id
psForm = (W_TABFORM*)widgGetFromID(psWScreen,buttonID);
if (psForm)
{
switch (buttonID)
{
case IDTRANS_LAUNCH:
psForm->UserData = PACKDWORD_TRI(1,IMAGE_LAUNCHDOWN,IMAGE_LAUNCHUP);
break;
case IDTIMER_FORM:
psForm->UserData = PACKDWORD_TRI(1,IMAGE_MISSION_CLOCK,IMAGE_MISSION_CLOCK_UP);
break;
default:
//do nothing other than in debug
ASSERT( false, "flashMissionButton: Unknown button ID" );
break;
}
}
}
/*stops one of the mission buttons (Launch Button or Mission Timer) flashing*/
void stopMissionButtonFlash(UDWORD buttonID)
{
W_TABFORM *psForm;
//get the button from the id
psForm = (W_TABFORM*)widgGetFromID(psWScreen,buttonID);
if (psForm)
{
switch (buttonID)
{
case IDTRANS_LAUNCH:
psForm->UserData = PACKDWORD_TRI(0,IMAGE_LAUNCHDOWN,IMAGE_LAUNCHUP);
break;
case IDTIMER_FORM:
psForm->UserData = PACKDWORD_TRI(0,IMAGE_MISSION_CLOCK,IMAGE_MISSION_CLOCK_UP);
break;
default:
//do nothing other than in debug
ASSERT( false, "stopMissionButtonFlash: Unknown button ID" );
break;
}
}
}
/* set current transporter (for script callbacks) */
void transporterSetScriptCurrent( DROID *psTransporter )
{
g_psCurScriptTransporter = psTransporter;
}
/* get current transporter (for script callbacks) */
DROID * transporterGetScriptCurrent( void )
{
return g_psCurScriptTransporter;
}
/*called when a Transporter has arrived back at the LZ when sending droids to safety*/
void resetTransporter()
{
W_CLICKFORM *psForm;
//enable the form so can add more droids into the transporter
psForm = (W_CLICKFORM*)widgGetFromID(psWScreen,IDTRANS_LAUNCH);
if (psForm)
{
formSetClickState(psForm, 0);
}
}
/*checks the order of the droid to see if its currently flying*/
BOOL transporterFlying(DROID *psTransporter)
{
ASSERT( psTransporter != NULL,
"transporterFlying: Invalid droid pointer" );
ASSERT( psTransporter->droidType == DROID_TRANSPORTER,
"transporterFlying: Droid is not a Transporter" );
if (psTransporter->order == DORDER_TRANSPORTOUT ||
psTransporter->order == DORDER_TRANSPORTIN ||
psTransporter->order == DORDER_TRANSPORTRETURN ||
//in multiPlayer mode the Transporter can be moved around
(bMultiPlayer && psTransporter->order == DORDER_MOVE) ||
//in multiPlayer mode the Transporter can be moved and emptied!
(bMultiPlayer && psTransporter->order == DORDER_DISEMBARK) ||
//in multiPlayer, descending still counts as flying
(bMultiPlayer && psTransporter->order == DORDER_NONE &&
psTransporter->sMove.iVertSpeed != 0))
{
return true;
}
else
{
return false;
}
}
//initialise the flag to indicate the first transporter has arrived - set in startMission()
void initFirstTransporterFlag(void)
{
bFirstTransporter = true;
}