/** \file ctrain.c * Functions related to running trains */ /* XTrkCad - Model Railroad CAD * Copyright (C) 2005 Dave Bullis * * 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. */ #ifndef WINDOWS #include <errno.h> #endif #include <ctype.h> #include <math.h> #include <string.h> #define PRIVATE_EXTRADATA #include "compound.h" #include "ctrain.h" #include "cundo.h" #include "custom.h" #include "fileio.h" #include "i18n.h" #include "layout.h" #include "messages.h" #include "param.h" #include "track.h" #include "trackx.h" #include "utility.h" long programMode; long maxCouplingSpeed = 100; long hideTrainsInTunnels; extern int doDrawTurnoutPosition; extern void NextTurnoutPosition(track_p); static TRKTYP_T T_CAR = -1; typedef enum { ST_NotOnTrack, ST_StopManual, ST_EndOfTrack, ST_OpenTurnout, ST_NoRoom, ST_Crashed } trainStatus_e; struct extraData { traverseTrack_t trvTrk; long state; carItem_p item; double speed; BOOL_T pencils; BOOL_T direction; BOOL_T autoReverse; trainStatus_e status; DIST_T distance; coOrd couplerPos[2]; unsigned int trkLayer; }; #define NOTALAYER (127) #define CAR_STATE_IGNORED (1L<<17) #define CAR_STATE_PROCESSED (1L<<18) #define CAR_STATE_LOCOISMASTER (1L<<19) #define CAR_STATE_ONHIDENTRACK (1L<<20) #define COUPLERCONNECTIONANGLE 45.0 #define CRASHSPEEDDECAY 5 #define IsOnTrack( XX ) ((XX)->trvTrk.trk!=NULL) #define IsIgnored( XX ) (((XX)->state&CAR_STATE_IGNORED)!=0) #define SetIgnored( XX ) (XX)->state |= CAR_STATE_IGNORED #define ClrIgnored( XX ) (XX)->state &= ~CAR_STATE_IGNORED #define IsLocoMaster( XX ) CarItemIsLocoMaster((XX)->item) #define SetLocoMaster( XX ) CarItemSetLocoMaster((XX)->item,TRUE) #define ClrLocoMaster( XX ) CarItemSetLocoMaster((XX)->item,FALSE) #define IsProcessed( XX ) (((XX)->state&CAR_STATE_PROCESSED)!=0) #define SetProcessed( XX ) (XX)->state |= CAR_STATE_PROCESSED #define ClrProcessed( XX ) (XX)->state &= ~CAR_STATE_PROCESSED static wButton_p newcarB; static void ControllerDialogSyncAll(void); static STATUS_T CmdTrain(wAction_t, coOrd); static wMenu_p trainPopupM; static wMenuPush_p trainPopupMI[10]; static track_p followTrain; static coOrd followCenter; static BOOL_T trainsTimeoutPending; static enum { TRAINS_STOP, TRAINS_RUN, TRAINS_IDLE, TRAINS_PAUSE } trainsState; static wIcon_p stopI, goI; static void RestartTrains(void); static void DrawAllCars(void); static void UncoupleCars(track_p, track_p); static void TrainTimeEndPause(void); static void TrainTimeStartPause(void); static int log_trainMove; static int log_trainPlayback; static void PlaceCar(track_p); #define WALK_CARS_START( CAR, XX, DIR ) \ while (1) { \ (XX) = GetTrkExtraData(CAR);\ { \ #define WALK_CARS_END( CAR, XX, DIR ) \ } \ { \ track_p walk_cars_temp1; \ if ( (walk_cars_temp1=GetTrkEndTrk(CAR,DIR)) == NULL ) break; \ (DIR)=(GetTrkEndTrk(walk_cars_temp1,0)==(CAR)?1:0); \ (CAR)=walk_cars_temp1; \ } \ } /* * Generic Commands */ void CarGetPos( track_p car, coOrd * posR, ANGLE_T * angleR) { struct extraData * xx = GetTrkExtraData(car); if (GetTrkType(car) != T_CAR) { AbortProg("getCarPos"); } *posR = xx->trvTrk.pos; *angleR = xx->trvTrk.angle; } void CarSetVisible( track_p car) { struct extraData * xx; int dir; dir = 0; WALK_CARS_START(car, xx, dir) if (GetTrkType(car) != T_CAR) { AbortProg("carSetVisible"); } WALK_CARS_END(car, xx, dir) dir = 1-dir; WALK_CARS_START(car, xx, dir) { xx->state &= ~(CAR_STATE_ONHIDENTRACK); xx->trkLayer = NOTALAYER; } WALK_CARS_END(car, xx, dir) } static struct { long index; coOrd pos; ANGLE_T angle; DIST_T length; DIST_T width; char desc[STR_SIZE]; char number[STR_SIZE]; } carData; typedef enum { IT, PN, AN, LN, WD, DE, NM } carDesc_e; static descData_t carDesc[] = { /*IT*/ { DESC_LONG, N_("Index"), &carData.index }, /*PN*/ { DESC_POS, N_("Position"), &carData.pos }, /*AN*/ { DESC_ANGLE, N_("Angle"), &carData.angle }, /*LN*/ { DESC_DIM, N_("Length"), &carData.length }, /*WD*/ { DESC_DIM, N_("Width"), &carData.width }, /*DE*/ { DESC_STRING, N_("Description"), &carData.desc, sizeof(carData.desc) }, /*NM*/ { DESC_STRING, N_("Rep Marks"), &carData.number, sizeof(carData.number) }, { DESC_NULL } }; static void UpdateCar( track_p trk, int inx, descData_p descUpd, BOOL_T needUndoStart) { if (inx == -1) { BOOL_T titleChanged; const char * cp; titleChanged = FALSE; cp = wStringGetValue((wString_p)carDesc[NM].control0); unsigned int max_str = sizeof(carData.number); if (max_str && strlen(cp)>max_str) { NoticeMessage2(0, MSG_ENTERED_STRING_TRUNCATED, _("Ok"), NULL, max_str-1); } if (cp && strcmp(carData.number, cp) != 0) { titleChanged = TRUE; carData.number[0] = '\0'; strncat(carData.number, cp, max_str - 1); } if (!titleChanged) { return; } if (needUndoStart) { UndoStart(_("Change Track"), "Change Track"); } UndoModify(trk); UndrawNewTrack(trk); DrawNewTrack(trk); return; } UndrawNewTrack(trk); switch (inx) { case NM: break; default: break; } DrawNewTrack(trk); } static void DescribeCar( track_p trk, char * str, CSIZE_T len) { struct extraData *xx = GetTrkExtraData(trk); char * cp; coOrd size; CarItemSize(xx->item, &size); carData.length = size.x; carData.width = size.y; cp = CarItemDescribe(xx->item, 0, &carData.index); carData.number[0] = '\0'; strncat(carData.number, CarItemNumber(xx->item), sizeof(carData.number) - 1); str[0] = '\0'; strncat(str, cp, len - 1); carData.pos = xx->trvTrk.pos; carData.angle = xx->trvTrk.angle; cp = CarItemDescribe(xx->item, -1, NULL); carData.desc[0] = '\0'; strncat(carData.desc, cp, sizeof(carData.desc) - 1); carDesc[IT].mode = carDesc[PN].mode = carDesc[AN].mode = carDesc[LN].mode = carDesc[WD].mode = DESC_RO; carDesc[DE].mode = carDesc[NM].mode = DESC_RO; DoDescribe(_("Car"), trk, carDesc, UpdateCar); } void FlipTraverseTrack( traverseTrack_p trvTrk) { trvTrk->angle = NormalizeAngle(trvTrk->angle + 180.0); if (trvTrk->length > 0) { trvTrk->dist = trvTrk->length - trvTrk->dist; } } BOOL_T TraverseTrack2( traverseTrack_p trvTrk0, DIST_T dist0) { traverseTrack_t trvTrk = *trvTrk0; DIST_T dist = dist0; if (dist0 < 0) { dist = -dist; FlipTraverseTrack(&trvTrk); } if (trvTrk.trk==NULL || (!TraverseTrack(&trvTrk,&dist)) || trvTrk.trk==NULL || dist!=0.0) { Translate(&trvTrk.pos, trvTrk.pos, trvTrk.angle, dist); } if (dist0 < 0) { FlipTraverseTrack(&trvTrk); } *trvTrk0 = trvTrk; return TRUE; } /*************** * When a track is deleted, cross check that the Traverse Track reference is removed. */ EXPORT void CheckCarTraverse(track_p track) { track_p car; for (car=NULL; TrackIterate(&car);) { if (GetTrkType(car) == T_CAR) { struct extraData * xx = GetTrkExtraData(car); if (xx->trvTrk.trk == track) { xx->trvTrk.trk=NULL; xx->status = ST_NotOnTrack; } } } } static BOOL_T drawCarEnable = TRUE; static BOOL_T noCarDraw = FALSE; static void DrawCar( track_p car, drawCmd_p d, wDrawColor color) { struct extraData * xx = GetTrkExtraData(car); int dir; vector_t coupler[2]; struct extraData * xx1; int dir1; if (drawCarEnable == FALSE) { return; } if (d == &mapD) { return; } if (noCarDraw) { return; } if (hideTrainsInTunnels && ((((xx->state&CAR_STATE_ONHIDENTRACK)!=0) && drawTunnel==0) || (xx->trkLayer!=NOTALAYER && !GetLayerVisible(xx->trkLayer)))) { return; } for (dir=0; dir<2; dir++) { track_p car1; coupler[dir].pos = xx->couplerPos[dir]; if ((car1 = GetTrkEndTrk(car,dir))) { xx1 = GetTrkExtraData(car1); dir1 = (GetTrkEndTrk(car1,0)==car)?0:1; coupler[dir].angle = FindAngle(xx->couplerPos[dir], xx1->couplerPos[dir1]); } else { coupler[dir].angle = NormalizeAngle(xx->trvTrk.angle+(dir==0?0.0:180.0)-15.0); } } CarItemDraw(d, xx->item, color, xx->direction, IsLocoMaster(xx), coupler, xx->pencils, xx->trvTrk.trk); } static DIST_T DistanceCar( track_p trk, coOrd * pos) { struct extraData * xx = GetTrkExtraData(trk); DIST_T dist; coOrd pos1; coOrd size; if (IsIgnored(xx)) { return 10000.0; } CarItemSize(xx->item, &size); /* TODO assumes xx->trvTrk.pos is the car center */ dist = FindDistance(*pos, xx->trvTrk.pos); if (dist < size.x/2.0) { pos1 = *pos; Rotate(&pos1, xx->trvTrk.pos, -xx->trvTrk.angle); pos1.x += -xx->trvTrk.pos.x + size.y/2.0; /* TODO: why not size.x? */ pos1.y += -xx->trvTrk.pos.y + size.x/2.0; if (pos1.x >= 0 && pos1.x <= size.y && pos1.y >= 0 && pos1.y <= size.x) { dist = 0; } } *pos = xx->trvTrk.pos; return dist; } static void SetCarBoundingBox( track_p car) { struct extraData * xx = GetTrkExtraData(car); coOrd lo, hi, p[4]; int inx; coOrd size; /* TODO: should be bounding box of all pieces aligned on track */ CarItemSize(xx->item, &size); /* TODO assumes xx->trvTrk.pos is the car center */ Translate(&p[0], xx->trvTrk.pos, xx->trvTrk.angle, size.x/2.0); Translate(&p[1], p[0], xx->trvTrk.angle+90, size.y/2.0); Translate(&p[0], p[0], xx->trvTrk.angle-90, size.y/2.0); Translate(&p[2], xx->trvTrk.pos, xx->trvTrk.angle+180, size.x/2.0); Translate(&p[3], p[2], xx->trvTrk.angle+90, size.y/2.0); Translate(&p[2], p[2], xx->trvTrk.angle-90, size.y/2.0); lo = hi = p[0]; for (inx = 1; inx < 4; inx++) { if (p[inx].x < lo.x) { lo.x = p[inx].x; } if (p[inx].y < lo.y) { lo.y = p[inx].y; } if (p[inx].x > hi.x) { hi.x = p[inx].x; } if (p[inx].y > hi.y) { hi.y = p[inx].y; } } SetBoundingBox(car, hi, lo); } track_p NewCar( wIndex_t index, carItem_p item, coOrd pos, ANGLE_T angle) { track_p trk; struct extraData * xx; trk = NewTrack(index, T_CAR, 2, sizeof(*xx)); /*SetEndPts( trk, 0 );*/ xx = GetTrkExtraData(trk); /*SetTrkVisible( trk, IsVisible(xx) );*/ xx->item = item; xx->trvTrk.pos = pos; xx->trvTrk.angle = angle; xx->state = 0; SetCarBoundingBox(trk); CarItemSetTrack(item, trk); PlaceCar(trk); return trk; } static void DeleteCar( track_p trk) { struct extraData * xx = GetTrkExtraData(trk); CarItemSetTrack(xx->item, NULL); } static BOOL_T ReadCar( char * line) { return CarItemRead(line); } static BOOL_T WriteCar( track_p trk, FILE * f) { BOOL_T rc = TRUE; return rc; } static void MoveCar( track_p car, coOrd pos) { struct extraData *xx = GetTrkExtraData(car); xx->trvTrk.pos.x += pos.x; xx->trvTrk.pos.y += pos.y; xx->trvTrk.trk = NULL; PlaceCar(car); SetCarBoundingBox(car); } static void RotateCar( track_p car, coOrd pos, ANGLE_T angle) { struct extraData *xx = GetTrkExtraData(car); Rotate(&xx->trvTrk.pos, pos, angle); xx->trvTrk.angle = NormalizeAngle(xx->trvTrk.angle + angle); xx->trvTrk.trk = NULL; PlaceCar(car); SetCarBoundingBox(car); } static BOOL_T QueryCar(track_p trk, int query) { switch (query) { case Q_NODRAWENDPT: return TRUE; default: return FALSE; } } static BOOL_T StoreCar( track_p car, void **data, long * len) { struct extraData *xx = GetTrkExtraData(car); return StoreCarItem(xx->item,data,len); } static BOOL_T ReplayCar (track_p car, void *data,long len) { struct extraData *xx = GetTrkExtraData(car); return ReplayCarItem(xx->item,data,len); } static wBool_t CompareCar( track_cp trk1, track_cp trk2 ) { return TRUE; } static trackCmd_t carCmds = { "CAR ", DrawCar, /* draw */ DistanceCar, /* distance */ DescribeCar, /* describe */ DeleteCar, /* delete */ WriteCar, /* write */ ReadCar, /* read */ MoveCar, /* move */ RotateCar, /* rotate */ NULL, /* rescale */ NULL, /* audit */ NULL, /* getAngle */ NULL, /* split */ NULL, /* traverse */ NULL, /* enumerate */ NULL, /* redraw*/ NULL, /* trim*/ NULL, /* merge*/ NULL, /* modify */ NULL, /* getLength */ NULL, /* getParams */ NULL, /* moveEndPt */ QueryCar, /* query */ NULL, /* ungroup */ NULL, /* flip */ NULL, NULL, NULL, NULL, NULL, NULL, ReplayCar, StoreCar, NULL, /*activate*/ CompareCar }; /* * */ static int numTrainDlg; #define SLIDER_WIDTH (20) #define SLIDER_HEIGHT (200) #define SLIDER_THICKNESS (10) #define MAX_SPEED (100.0) typedef struct { wWin_p win; wIndex_t inx; track_p train; long direction; long followMe; long autoReverse; coOrd pos; char posS[STR_SHORT_SIZE]; DIST_T speed; char speedS[10]; paramGroup_p trainPGp; } trainControlDlg_t, * trainControlDlg_p; static trainControlDlg_t * curTrainDlg; static void SpeedRedraw(wDraw_p, void *, wPos_t, wPos_t); static void SpeedAction(wAction_t, coOrd); static void LocoListChangeEntry(track_p, track_p); static void CmdTrainExit(void *); drawCmd_t speedD = { NULL, &screenDrawFuncs, 0, 1.0, 0.0, { 0.0, 0.0 }, { 0.0, 0.0 }, Pix2CoOrd, CoOrd2Pix }; static paramDrawData_t speedParamData = { SLIDER_WIDTH, SLIDER_HEIGHT, SpeedRedraw, SpeedAction, &speedD }; #ifndef WINDOWS static paramListData_t listData = { 3, 120 }; #endif static char * trainFollowMeLabels[] = { N_("Follow"), NULL }; static char * trainAutoReverseLabels[] = { N_("Auto Reverse"), NULL }; static paramData_t trainPLs[] = { #define I_LIST (0) #ifdef WINDOWS /*0*/ { PD_DROPLIST, NULL, "list", PDO_NOPREF|PDO_NOPSHUPD, (void*)120, NULL, 0 }, #else /*0*/ { PD_LIST, NULL, "list", PDO_NOPREF|PDO_NOPSHUPD, &listData, NULL, 0 }, #endif #define I_STATUS (1) { PD_MESSAGE, NULL, NULL, 0, (void*)120 }, #define I_POS (2) { PD_MESSAGE, NULL, NULL, 0, (void*)120 }, #define I_SLIDER (3) { PD_DRAW, NULL, "speed", PDO_NOPSHUPD|PDO_DLGSETY, &speedParamData }, #define I_DIST (4) { PD_STRING, NULL, "distance", PDO_DLGNEWCOLUMN, (void*)(100-SLIDER_WIDTH), NULL, BO_READONLY }, #define I_ZERO (5) { PD_BUTTON, NULL, "zeroDistance", PDO_NOPSHUPD|PDO_NOPREF|PDO_DLGHORZ, NULL, NULL, BO_ICON }, #define I_GOTO (6) { PD_BUTTON, NULL, "goto", PDO_NOPSHUPD|PDO_NOPREF|PDO_DLGWIDE, NULL, N_("Find") }, #define I_FOLLOW (7) { PD_TOGGLE, NULL, "follow", PDO_NOPREF|PDO_DLGWIDE, trainFollowMeLabels, NULL, BC_HORZ|BC_NOBORDER }, #define I_AUTORVRS (8) { PD_TOGGLE, NULL, "autoreverse", PDO_NOPREF, trainAutoReverseLabels, NULL, BC_HORZ|BC_NOBORDER }, #define I_DIR (9) { PD_BUTTON, NULL, "direction", PDO_NOPREF|PDO_DLGWIDE, NULL, N_("Forward"), 0 }, #define I_STOP (10) { PD_BUTTON, NULL, "stop", PDO_DLGWIDE, NULL, N_("Stop") }, #define I_SPEED (11) { PD_MESSAGE, NULL, NULL, PDO_DLGIGNOREX, (void *)120 } }; static paramGroup_t trainPG = { "train", 0, trainPLs, sizeof trainPLs/sizeof trainPLs[0] }; typedef struct { track_p loco; BOOL_T running; } locoList_t; dynArr_t locoList_da; #define locoList(N) DYNARR_N( locoList_t, locoList_da, N ) static wIndex_t FindLoco( track_p loco) { wIndex_t inx; for (inx = 0; inx<locoList_da.cnt; inx++) { if (locoList(inx).loco == loco) { return inx; } } return -1; } /** * Update the speed display when running trains. Draw the slider in the * correct position and update the odometer. * * \param d IN drawing area for slider * \param d IN the dialog * \param w, h IN unused? * \return describe the return value */ static void SpeedRedraw( wDraw_p d, void * context, wPos_t w, wPos_t h) { wPos_t y, pts[4][2]; trainControlDlg_p dlg = (trainControlDlg_p)context; struct extraData * xx; wDrawColor drawColor; wDrawClear(d); if (dlg == NULL || dlg->train == NULL) { return; } xx = GetTrkExtraData(dlg->train); if (xx->speed > MAX_SPEED) { xx->speed = MAX_SPEED; } if (xx->speed < 0) { xx->speed = 0; } y = (wPos_t)(xx->speed/MAX_SPEED*((SLIDER_HEIGHT-SLIDER_THICKNESS)) +SLIDER_THICKNESS/2); drawColor = wDrawFindColor(wRGB(160, 160, 160)); pts[0][1] = pts[1][1] = y-SLIDER_THICKNESS/2; pts[2][1] = pts[3][1] = y+SLIDER_THICKNESS/2; pts[0][0] = pts[3][0] = 0; pts[1][0] = pts[2][0] = SLIDER_WIDTH; wDrawPolygon(d, pts, NULL, 4, drawColor, 0, 0, 0, 1, 0); drawColor = wDrawFindColor(wRGB(220, 220, 220)); pts[0][1] = pts[1][1] = y+SLIDER_THICKNESS/2; pts[2][1] = pts[3][1] = y; pts[0][0] = pts[3][0] = 0; pts[1][0] = pts[2][0] = SLIDER_WIDTH; wDrawPolygon(d, pts, NULL, 4, drawColor, 0, 0, 0, 1, 0); wDrawLine(d, 0, y, SLIDER_WIDTH, y, 1, wDrawLineSolid, drawColorRed, 0); wDrawLine(d, 0, y+SLIDER_THICKNESS/2, SLIDER_WIDTH, y+SLIDER_THICKNESS/2, 1, wDrawLineSolid, drawColorBlack, 0); wDrawLine(d, 0, y-SLIDER_THICKNESS/2, SLIDER_WIDTH, y-SLIDER_THICKNESS/2, 1, wDrawLineSolid, drawColorBlack, 0); sprintf(dlg->speedS, "%3d %s", (int)(units==UNITS_ENGLISH?xx->speed:xx->speed*1.6), (units==UNITS_ENGLISH?"mph":"km/h")); ParamLoadMessage(dlg->trainPGp, I_SPEED, dlg->speedS); LOG(log_trainPlayback, 3, ("Speed = %d\n", (int)xx->speed)); } static void SpeedAction( wAction_t action, coOrd pos) { trainControlDlg_p dlg = curTrainDlg; struct extraData * xx; FLOAT_T speed; BOOL_T startStop; if (dlg == NULL || dlg->train == NULL) { return; } xx = GetTrkExtraData(dlg->train); switch (action) { case C_DOWN: InfoMessage(""); case C_MOVE: case C_UP: TrainTimeEndPause(); if (IsOnTrack(xx)) { speed = ((FLOAT_T)((pos.y*speedD.dpi)-SLIDER_THICKNESS/2))/ (SLIDER_HEIGHT-SLIDER_THICKNESS)*MAX_SPEED; } else { speed = 0; } if (speed > MAX_SPEED) { speed = MAX_SPEED; } if (speed < 0) { speed = 0; } startStop = (xx->speed == 0) != (speed == 0); xx->speed = speed; SpeedRedraw((wDraw_p)dlg->trainPGp->paramPtr[I_SLIDER].control, dlg, SLIDER_WIDTH, SLIDER_HEIGHT); if (startStop) { if (xx->speed == 0) { xx->status = ST_StopManual; } LocoListChangeEntry(dlg->train, dlg->train); } TrainTimeStartPause(); if (trainsState == TRAINS_IDLE) { RestartTrains(); } break; default: break; } } static void ControllerDialogSync( trainControlDlg_p dlg) { struct extraData * xx=NULL; wIndex_t inx; BOOL_T dir; BOOL_T followMe; BOOL_T autoReverse; coOrd pos; if (dlg == NULL) { return; } inx = wListGetIndex((wList_p)dlg->trainPGp->paramPtr[I_LIST].control); if (dlg->train) { if (inx >= 0 && inx < locoList_da.cnt && dlg->train && dlg->train != locoList(inx).loco) { inx = FindLoco(dlg->train); if (inx >= 0) { wListSetIndex((wList_p)dlg->trainPGp->paramPtr[I_LIST].control, inx); } } } else { wListSetIndex((wList_p)dlg->trainPGp->paramPtr[I_LIST].control, -1); } if (dlg->train) { char * statusMsg; DIST_T speed; xx = GetTrkExtraData(dlg->train); dir = xx->direction==0?0:1; speed = xx->speed; pos = xx->trvTrk.pos; followMe = followTrain == dlg->train; autoReverse = xx->autoReverse; if (xx->trvTrk.trk == NULL) { if (xx->status == ST_Crashed) { statusMsg = _("Crashed"); } else { statusMsg = _("Not on Track"); } } else if (xx->speed > 0) { if (trainsState == TRAINS_STOP) { statusMsg = _("Trains Paused"); } else { statusMsg = _("Running"); } } else { switch (xx->status) { case ST_EndOfTrack: statusMsg = _("End of Track"); break; case ST_OpenTurnout: statusMsg = _("Open Turnout"); break; case ST_StopManual: statusMsg = _("Manual Stop"); break; case ST_NoRoom: statusMsg = _("No Room"); break; case ST_Crashed: statusMsg = _("Crashed"); break; default: statusMsg = _("Unknown Status"); break; } } ParamLoadMessage(dlg->trainPGp, I_STATUS, statusMsg); } else { dir = 0; followMe = FALSE; autoReverse = FALSE; ParamLoadMessage(dlg->trainPGp, I_STATUS, _("No trains")); } if (dlg->followMe != followMe) { dlg->followMe = followMe; ParamLoadControl(dlg->trainPGp, I_FOLLOW); } if (dlg->autoReverse != autoReverse) { dlg->autoReverse = autoReverse; ParamLoadControl(dlg->trainPGp, I_AUTORVRS); } if (dlg->direction != dir) { dlg->direction = dir; wButtonSetLabel((wButton_p)dlg->trainPGp->paramPtr[I_DIR].control, (dlg->direction?_("Reverse"):_("Forward"))); } if (dlg->train) { if (dlg->posS[0] == '\0' || dlg->pos.x != xx->trvTrk.pos.x || dlg->pos.y != xx->trvTrk.pos.y) { long format; dlg->pos = xx->trvTrk.pos; format = GetDistanceFormat(); format &= ~DISTFMT_DECS; sprintf(dlg->posS, "X:%s Y:%s", FormatDistanceEx(xx->trvTrk.pos.x, format), FormatDistanceEx(xx->trvTrk.pos.y, format)); ParamLoadMessage(dlg->trainPGp, I_POS, dlg->posS); } if (dlg->speed != xx->speed) { dlg->speed = xx->speed; sprintf(dlg->speedS, "%3d", (int)(units==UNITS_ENGLISH?xx->speed:xx->speed*1.6)); ParamLoadMessage(dlg->trainPGp, I_SPEED, dlg->speedS); SpeedRedraw((wDraw_p)dlg->trainPGp->paramPtr[I_SLIDER].control, dlg, SLIDER_WIDTH, SLIDER_HEIGHT); } ParamLoadMessage(dlg->trainPGp, I_DIST, FormatDistance(xx->distance)); } else { if (dlg->posS[0] != '\0') { dlg->posS[0] = '\0'; ParamLoadMessage(dlg->trainPGp, I_POS, dlg->posS); } if (dlg->speed >= 0) { dlg->speed = -1; dlg->speedS[0] = '\0'; ParamLoadMessage(dlg->trainPGp, I_SPEED, dlg->speedS); wDrawClear((wDraw_p)dlg->trainPGp->paramPtr[I_SLIDER].control); } ParamLoadMessage(dlg->trainPGp, I_DIST, ""); } } static void ControllerDialogSyncAll(void) { if (curTrainDlg) { ControllerDialogSync(curTrainDlg); } } static void LocoListChangeEntry( track_p oldLoco, track_p newLoco) { wIndex_t inx = -1; struct extraData * xx; if (curTrainDlg == NULL) { return; } if (oldLoco && (inx=FindLoco(oldLoco))>=0) { if (newLoco) { locoList(inx).loco = newLoco; xx = GetTrkExtraData(newLoco); locoList(inx).running = IsOnTrack(xx) && xx->speed > 0; wListSetValues((wList_p)curTrainDlg->trainPGp->paramPtr[I_LIST].control, inx, CarItemNumber(xx->item), locoList(inx).running?goI:stopI, newLoco); } else { wListDelete((wList_p)curTrainDlg->trainPGp->paramPtr[I_LIST].control, inx); for (; inx<locoList_da.cnt-1; inx++) { locoList(inx) = locoList(inx+1); } locoList_da.cnt -= 1; if (inx >= locoList_da.cnt) { inx--; } } } else if (newLoco) { inx = locoList_da.cnt; DYNARR_APPEND(locoList_t, locoList_da, 10); locoList(inx).loco = newLoco; xx = GetTrkExtraData(newLoco); locoList(inx).running = IsOnTrack(xx) && xx->speed > 0; wListAddValue((wList_p)curTrainDlg->trainPGp->paramPtr[I_LIST].control, CarItemNumber(xx->item), locoList(inx).running?goI:stopI, newLoco); } if (curTrainDlg->train == oldLoco) { if (newLoco || locoList_da.cnt <= 0) { curTrainDlg->train = newLoco; } else { curTrainDlg->train = wListGetItemContext((wList_p) curTrainDlg->trainPGp->paramPtr[I_LIST].control, inx); } } ControllerDialogSync(curTrainDlg); } static void LocoListInit(void) { track_p train; struct extraData * xx; locoList_da.cnt = 0; for (train=NULL; TrackIterate(&train);) { if (GetTrkType(train) != T_CAR) { continue; } xx = GetTrkExtraData(train); if (!CarItemIsLoco(xx->item)) { continue; } if (!IsLocoMaster(xx)) { continue; } LocoListChangeEntry(NULL, train); } } static void SetCurTrain( track_p train) { curTrainDlg->train = train; ControllerDialogSync(curTrainDlg); } static void StopTrain( track_p train, trainStatus_e status) { struct extraData * xx; if (train == NULL) { return; } xx = GetTrkExtraData(train); xx->speed = 0; xx->status = status; LocoListChangeEntry(train, train); } static void MoveMainWindow( coOrd pos, ANGLE_T angle) { DIST_T dist; static DIST_T factor = 0.5; ANGLE_T angle1 = angle, angle2; if (angle1 > 180.0) { angle1 = 360.0 - angle1; } if (angle1 > 90.0) { angle1 = 180.0 - angle1; } angle2 = R2D(atan2(mainD.size.x,mainD.size.y)); if (angle1 < angle2) { dist = mainD.size.y/2.0/cos(D2R(angle1)); } else { dist = mainD.size.x/2.0/cos(D2R(90.0-angle1)); } dist *= factor; Translate(&pos, pos, angle, dist); mainD.orig.x = pos.x-mainD.size.x/2;; mainD.orig.y = pos.y-mainD.size.y/2;; panCenter = pos; LOG( log_pan, 2, ( "PanCenter:%d %0.3f %0.3f\n", __LINE__, panCenter.x, panCenter.y ) ); MainLayout( TRUE, TRUE ); // MoveTrainWindow } static void SetTrainDirection( track_p train) { struct extraData *xx, *xx0=GetTrkExtraData(train); int dir0; track_p car; car = train; for (dir0 = 0; dir0 < 2; dir0++) { int dir; dir = dir0; WALK_CARS_START(car, xx, dir) if (car != train) { if (CarItemIsLoco(xx->item)) { xx->direction = (dir==dir0?xx0->direction:!xx0->direction); } } WALK_CARS_END(car, xx, dir) } } static void ControllerDialogUpdate( paramGroup_p pg, int inx, void * valueP) { trainControlDlg_p dlg = curTrainDlg; track_p train; struct extraData * xx; if (dlg == NULL) { return; } TrainTimeEndPause(); switch (inx) { case I_LIST: train = (track_p)wListGetItemContext((wList_p)pg->paramPtr[inx].control, (wIndex_t)*(long*)valueP); if (train == NULL) { return; } dlg->train = train; ControllerDialogSync(dlg); break; case I_ZERO: if (dlg->train == NULL) { return; } TrainTimeEndPause(); xx = GetTrkExtraData(dlg->train); xx->distance = 0.0; ParamLoadMessage(dlg->trainPGp, I_DIST, FormatDistance(xx->distance)); ParamLoadControl(curTrainDlg->trainPGp, I_DIST); TrainTimeStartPause(); break; case I_GOTO: if (dlg->train == NULL) { return; } TrainTimeEndPause(); xx = GetTrkExtraData(dlg->train); followTrain = NULL; dlg->followMe = FALSE; ParamLoadControl(curTrainDlg->trainPGp, I_FOLLOW); CarSetVisible(dlg->train); MoveMainWindow(xx->trvTrk.pos, xx->trvTrk.angle); TrainTimeStartPause(); break; case I_FOLLOW: if (dlg->train == NULL) { return; } if (*(long*)valueP) { followTrain = dlg->train; xx = GetTrkExtraData(dlg->train); if (OFF_MAIND(xx->trvTrk.pos, xx->trvTrk.pos)) { MoveMainWindow(xx->trvTrk.pos, xx->trvTrk.angle); } followCenter = mainCenter; } else { followTrain = NULL; } break; case I_AUTORVRS: if (dlg->train == NULL) { return; } xx = GetTrkExtraData(dlg->train); xx->autoReverse = *(long*)valueP!=0; break; case I_DIR: if (dlg->train == NULL) { return; } xx = GetTrkExtraData(dlg->train); dlg->direction = xx->direction = !xx->direction; wButtonSetLabel((wButton_p)pg->paramPtr[I_DIR].control, (dlg->direction?_("Reverse"):_("Forward"))); SetTrainDirection(dlg->train); TempRedraw(); // ctrain: change direction break; case I_STOP: if (dlg->train == NULL) { return; } TrainTimeEndPause(); StopTrain(dlg->train, ST_StopManual); TrainTimeStartPause(); break; case -1: /* Close window */ CmdTrainExit(NULL); break; } /*ControllerDialogSync( dlg );*/ TrainTimeStartPause(); } static trainControlDlg_p CreateTrainControlDlg(void) { trainControlDlg_p dlg; char * title; paramData_p PLp; dlg = (trainControlDlg_p)MyMalloc(sizeof *dlg); PLp = trainPLs; dlg->posS[0] = '\0'; dlg->speedS[0] = '\0'; PLp[I_LIST].valueP = &dlg->inx; PLp[I_LIST].context = dlg; PLp[I_POS].valueP = &dlg->posS; PLp[I_POS].context = dlg; /*PLp[I_GOTO].valueP = NULL;*/ PLp[I_GOTO].context = dlg; PLp[I_SLIDER].context = dlg; PLp[I_SPEED].valueP = &dlg->speedS; PLp[I_SPEED].context = dlg; PLp[I_DIR].context = dlg; /*PLp[I_STOP].valueP = NULL;*/ PLp[I_STOP].context = dlg; PLp[I_FOLLOW].valueP = &dlg->followMe; PLp[I_FOLLOW].context = dlg; PLp[I_AUTORVRS].valueP = &dlg->autoReverse; PLp[I_AUTORVRS].context = dlg; title = MyStrdup(_("Train Control XXX")); sprintf(title, _("Train Control %d"), ++numTrainDlg); dlg->trainPGp = &trainPG; dlg->win = ParamCreateDialog(dlg->trainPGp, _("Train Control"), NULL, NULL, NULL, FALSE, NULL, 0, ControllerDialogUpdate); return dlg; } /* * STATE INFO */ static struct { STATE_T state; coOrd pos0; } Dtrain; long trainPause = 200; static track_p followTrain = NULL; static void DrawAllCars(void) { track_p car; struct extraData * xx; coOrd size, lo, hi; BOOL_T drawCarEnable1 = drawCarEnable; drawCarEnable = TRUE; wDrawDelayUpdate(mainD.d, TRUE); wDrawRestoreImage(mainD.d); DrawPositionIndicators(); for (car=NULL; TrackIterate(&car);) { if (GetTrkType(car) == T_CAR) { xx = GetTrkExtraData(car); CarItemSize(xx->item, &size); /* TODO assumes xx->trvTrk.pos is the car center */ lo.x = xx->trvTrk.pos.x - size.x/2.0; lo.y = xx->trvTrk.pos.y - size.x/2.0; hi.x = lo.x + size.x; hi.y = lo.y + size.x; if (!OFF_MAIND(lo, hi)) { DrawCar(car, &tempD, wDrawColorBlack); } } } wDrawDelayUpdate(mainD.d, FALSE); drawCarEnable = drawCarEnable1; } static DIST_T GetTrainLength2( track_p * car0, BOOL_T * dir) { DIST_T length = 0, carLength; struct extraData * xx; WALK_CARS_START(*car0, xx, *dir) carLength = CarItemCoupledLength(xx->item); if (length == 0) { length = carLength/2.0; /* TODO assumes xx->trvTrk.pos is the car center */ } else { length += carLength; } WALK_CARS_END(*car0, xx, *dir) return length; } static DIST_T GetTrainLength( track_p car0, BOOL_T dir) { return GetTrainLength2(&car0, &dir); } static void PlaceCar( track_p car) { struct extraData *xx = GetTrkExtraData(car); DIST_T dists[2]; CarItemPlace(xx->item, &xx->trvTrk, dists); CarItemFindCouplerMountPoint(xx->item, xx->trvTrk, xx->couplerPos); car->endPt[0].angle = xx->trvTrk.angle; Translate(&car->endPt[0].pos, xx->trvTrk.pos, car->endPt[0].angle, dists[0]); car->endPt[1].angle = NormalizeAngle(xx->trvTrk.angle + 180.0); Translate(&car->endPt[1].pos, xx->trvTrk.pos, car->endPt[1].angle, dists[1]); LOG(log_trainMove, 4, ("%s @ [%0.3f,%0.3f] A%0.3f\n", CarItemNumber(xx->item), xx->trvTrk.pos.x, xx->trvTrk.pos.y, xx->trvTrk.angle)) SetCarBoundingBox(car); xx->state &= ~(CAR_STATE_ONHIDENTRACK); xx->trkLayer = NOTALAYER; if (xx->trvTrk.trk) { if (!GetTrkVisible(xx->trvTrk.trk)) { xx->state |= CAR_STATE_ONHIDENTRACK; } xx->trkLayer = GetTrkLayer(xx->trvTrk.trk); } } static track_p FindCar( coOrd * pos) { coOrd pos0, pos1; track_p trk, trk1; DIST_T dist1 = 100000, dist; struct extraData * xx; trk1 = NULL; for (trk=NULL; TrackIterate(&trk);) { if (GetTrkType(trk) == T_CAR) { xx = GetTrkExtraData(trk); if (IsIgnored(xx)) { continue; } pos0 = *pos; dist = DistanceCar(trk, &pos0); if (dist < dist1) { dist1 = dist; trk1 = trk; pos1 = pos0; } } } if (dist1 < 10) { *pos = pos1; return trk1; } else { return NULL; } } static track_p FindMasterLoco( track_p train, int * dirR) { struct extraData *xx0; int dir; for (dir = 0; dir<2; dir++) { track_p car0; int dir0; car0 = train; dir0 = dir; WALK_CARS_START(car0, xx0, dir0) if (CarItemIsLoco(xx0->item) && IsLocoMaster(xx0)) { if (dirR) { *dirR = 1-dir0; } return car0; } WALK_CARS_END(car0, xx0, dir0) } return NULL; } static track_p PickMasterLoco( track_p car, int dir) { track_p loco=NULL; struct extraData *xx; WALK_CARS_START(car, xx, dir) if (CarItemIsLoco(xx->item)) { if (IsLocoMaster(xx)) { return car; } if (loco == NULL) { loco = car; } } WALK_CARS_END(car, xx, dir) if (loco == NULL) { return NULL; } xx = GetTrkExtraData(loco); SetLocoMaster(xx); xx->speed = 0; LOG(log_trainMove, 1, ("%s becomes master\n", CarItemNumber(xx->item))) return loco; } static void UncoupleCars( track_p car1, track_p car2) { track_p loco; int dir1, dir2; if (GetTrkEndTrk(car1,0) == car2) { dir1 = 0; } else if (GetTrkEndTrk(car1,1) == car2) { dir1 = 1; } else { ErrorMessage("uncoupleCars - not coupled"); return; } if (GetTrkEndTrk(car2,0) == car1) { dir2 = 0; } else if (GetTrkEndTrk(car2,1) == car1) { dir2 = 1; } else { ErrorMessage("uncoupleCars - not coupled"); return; } loco = FindMasterLoco(car1, NULL); car1->endPt[dir1].track = NULL; car2->endPt[dir2].track = NULL; if (loco) { track_p loco1, loco2; loco1 = PickMasterLoco(car1, 1-dir1); if (loco1 != loco) { LocoListChangeEntry(NULL, loco1); } loco2 = PickMasterLoco(car2, 1-dir2); if (loco2 != loco) { LocoListChangeEntry(NULL, loco2); } } } static void CoupleCars( track_p car1, int dir1, track_p car2, int dir2) { struct extraData * xx1, * xx2; track_p loco1, loco2; track_p car; int dir; xx1 = GetTrkExtraData(car1); xx2 = GetTrkExtraData(car2); if (GetTrkEndTrk(car1,dir1) != NULL || GetTrkEndTrk(car2,dir2) != NULL) { LOG(log_trainMove, 1, ("coupleCars - already coupled\n")) return; } car = car1; dir = 1-dir1; WALK_CARS_START(car, xx1, dir) if (car == car2) { LOG(log_trainMove, 1, ("coupleCars - already coupled\n")) ErrorMessage("Car coupling loop"); return; } WALK_CARS_END(car, xx1, dir) car = car2; dir = 1-dir2; WALK_CARS_START(car, xx2, dir) if (car == car1) { LOG(log_trainMove, 1, ("coupleCars - already coupled\n")) ErrorMessage("Car coupling loop"); return; } WALK_CARS_END(car, xx1, dir) loco1 = FindMasterLoco(car1, NULL); loco2 = FindMasterLoco(car2, NULL); car1->endPt[dir1].track = car2; car2->endPt[dir2].track = car1; /*ConnectTracks( car1, dir1, car2, dir2 );*/ if (logTable(log_trainMove).level >= 2) { LogPrintf("Coupling %s[%d] ", CarItemNumber(xx1->item), dir1); LogPrintf(" and %s[%d]\n", CarItemNumber(xx2->item), dir2); } if ((loco1 != NULL && loco2 != NULL)) { xx1 = GetTrkExtraData(loco1); xx2 = GetTrkExtraData(loco2); if (xx1->speed == 0) { ClrLocoMaster(xx1); LOG(log_trainMove, 2, ("%s loses master\n", CarItemNumber(xx1->item))) if (followTrain == loco1) { followTrain = loco2; } LocoListChangeEntry(loco1, NULL); loco1 = loco2; } else { ClrLocoMaster(xx2); xx1->speed = (xx1->speed + xx2->speed)/2.0; if (xx1->speed < 0) { xx1->speed = 0; } if (xx1->speed > 100) { xx1->speed = 100; } LOG(log_trainMove, 2, ("%s loses master\n", CarItemNumber(xx2->item))) if (followTrain == loco2) { followTrain = loco1; } LocoListChangeEntry(loco2, NULL); } SetTrainDirection(loco1); } } long crashDistFactor=60; static void PlaceCars( track_p car0, int dir0, long crashSpeed, BOOL_T crashFlip) { struct extraData *xx0 = GetTrkExtraData(car0), *xx; int dir; traverseTrack_t trvTrk; DIST_T length; track_p car_curr; DIST_T flipflop = 1; if (crashFlip) { flipflop = -1; } dir = dir0; trvTrk = xx0->trvTrk; if (dir0) { FlipTraverseTrack(&trvTrk); } length = CarItemCoupledLength(xx0->item)/2.0; car_curr = car0; ClrIgnored(xx0); WALK_CARS_START(car_curr, xx, dir) if (car_curr != car0) { DIST_T dist, length1; ClrIgnored(xx); length1 = CarItemCoupledLength(xx->item)/2.0; dist = length + length1; crashSpeed = crashSpeed*CRASHSPEEDDECAY/10; if (crashSpeed > 0) { dist -= dist * crashSpeed/crashDistFactor; } TraverseTrack2(&trvTrk, dist); xx->trvTrk = trvTrk; if (crashSpeed > 0) { xx->trvTrk.angle = NormalizeAngle(xx->trvTrk.angle + flipflop*crashSpeed); xx->trvTrk.trk = NULL; } flipflop = -flipflop; if (dir != 0) { FlipTraverseTrack(&xx->trvTrk); } PlaceCar(car_curr); length = length1; } WALK_CARS_END(car_curr, xx, dir) } static void CrashTrain( track_p car, int dir, traverseTrack_p trvTrkP, long speed, BOOL_T flip) { track_p loco; struct extraData *xx; loco = FindMasterLoco(car,NULL); if (loco != NULL) { StopTrain(loco, ST_Crashed); } xx = GetTrkExtraData(car); xx->trvTrk = *trvTrkP; if (dir) { FlipTraverseTrack(&xx->trvTrk); } PlaceCars(car, 1-dir, speed, flip); if (flip) { speed = - speed; } xx->trvTrk.angle = NormalizeAngle(xx->trvTrk.angle - speed); xx->trvTrk.trk = NULL; PlaceCar(car); } /* * Check if this should couple or optionally if it should crash * It will call couple or crash if needed. * Returns TRUE if we should continue. */ static BOOL_T CheckCoupling( track_p car0, int dir00, BOOL_T doCheckCrash) { track_p car1; struct extraData *xx0, *xx1; coOrd pos1; DIST_T dist0, distc, dist=100000.0; int dir0, dir1, dirl; ANGLE_T angle; traverseTrack_t trvTrk0, trvTrk1; xx0 = xx1 = GetTrkExtraData(car0); /* find length of train from loco to start and end */ dir0 = dir00; dist0 = GetTrainLength2(&car0, &dir0); trvTrk0 = xx0->trvTrk; if (dir00) { FlipTraverseTrack(&trvTrk0); } TraverseTrack2(&trvTrk0, dist0); pos1 = trvTrk0.pos; car1 = FindCar(&pos1); if (!car1) { return TRUE; } xx1 = GetTrkExtraData(car1); if (!IsOnTrack(xx1)) { return TRUE; } /* determine which EP of the found car to couple to */ angle = NormalizeAngle(trvTrk0.angle-xx1->trvTrk.angle); if (angle > 90 && angle < 270) { dir1 = 0; angle = NormalizeAngle(angle+180); } else { dir1 = 1; } /* already coupled? */ if (GetTrkEndTrk(car1,dir1) != NULL) { return TRUE; } /* are we close to aligned? Uses 45 degrees offset today */ /* It assumes that if the cars are aligned they could/should be coupled */ if (angle > COUPLERCONNECTIONANGLE && angle < 360.0-COUPLERCONNECTIONANGLE) { return TRUE; } /* find pos of found end car's coupler, and dist btw couplers */ distc = CarItemCoupledLength(xx1->item); /* pos1 is the end of the xx1 car (end car) */ Translate(&pos1, xx1->trvTrk.pos, xx1->trvTrk.angle+(dir1?180.0:0.0), distc/2.0); dist = FindDistance(trvTrk0.pos, pos1); /* How far away are the two ends?*/ if (dist < trackGauge/10) { return TRUE; } /* not real close: are we overlapped? */ angle = FindAngle(trvTrk0.pos, pos1); angle = NormalizeAngle(angle - trvTrk0.angle); if (angle < 90 || angle > 270) { return TRUE; } /* are we beyond the end of the found car? */ if (dist > distc) { return TRUE; } /* are we on the same track? */ trvTrk1 = xx1->trvTrk; if (dir1) { FlipTraverseTrack(&trvTrk1); } /* Move second train back along track half a car length */ TraverseTrack2(&trvTrk1, distc/2.0-dist); if ( trvTrk0.trk == NULL || trvTrk1.trk == NULL ) // fell off the end of track return FALSE; /* If tracks are not the same - dont couple */ if (trvTrk1.trk != trvTrk0.trk) { return TRUE; } /* If this is further apart than 2 track gauges on a turnout, dont couple */ if (GetTrkType(trvTrk0.trk) == T_TURNOUT) { if (dist > GetTrkGauge(trvTrk0.trk)*2) { return TRUE; } } /* Concluded we are hitting each other */ if (doCheckCrash) { track_p loco1; long speed, speed0, speed1; speed0 = (long)xx0->speed; if ((xx0->direction==0) != (dir00==0)) { speed0 = - speed0; } loco1 = FindMasterLoco(car1, &dirl); xx1 = NULL; if (loco1) { xx1 = GetTrkExtraData(loco1); speed1 = (long)xx1->speed; if (car1 == loco1) { dirl = IsAligned(xx1->trvTrk.angle, FindAngle(trvTrk0.pos, xx1->trvTrk.pos))?1:0; } if ((xx1->direction==1) != (dirl==1)) { speed1 = -speed1; } } else { speed1 = 0; } speed = labs(speed0 + speed1); LOG(log_trainMove, 2, ("coupling speed=%ld\n", speed)) if (speed > maxCouplingSpeed) { CrashTrain(car0, dir0, &trvTrk0, speed, FALSE); CrashTrain(car1, dir1, &trvTrk1, speed, TRUE); return FALSE; } } if (dir00) { dist = -dist; } TraverseTrack2(&xx0->trvTrk, dist); CoupleCars(car0, dir0, car1, dir1); LOG(log_trainMove, 3, (" -> %0.3f\n", dist)) return TRUE; } static void PlaceTrain( track_p car0, BOOL_T doCheckCrash, BOOL_T doCheckCoupling) { track_p car_curr; struct extraData *xx0; int dir0; xx0 = GetTrkExtraData(car0); LOG(log_trainMove, 2, (" placeTrain: %s [%0.3f %0.3f] A%0.3f", CarItemNumber(xx0->item), xx0->trvTrk.pos.x, xx0->trvTrk.pos.y, xx0->trvTrk.angle)) car_curr = car0; for (dir0=0; dir0<2; dir0++) { int dir; struct extraData *xx; car_curr = car0; dir = dir0; xx = xx0; WALK_CARS_START(car_curr, xx, dir) SetIgnored(xx); WALK_CARS_END(car_curr, xx, dir); } /* check for coupling to other cars */ if (doCheckCoupling) { if (xx0->trvTrk.trk) if (!CheckCoupling(car0, 0, doCheckCrash)) { return; } if (xx0->trvTrk.trk) if (!CheckCoupling(car0, 1, doCheckCrash)) { return; } } PlaceCar(car0); for (dir0=0; dir0<2; dir0++) { PlaceCars(car0, dir0, 0, FALSE); } } static void PlaceTrainInit( track_p car0, track_p trk0, coOrd pos0, ANGLE_T angle0, BOOL_T doCheckCoupling) { struct extraData * xx = GetTrkExtraData(car0); xx->trvTrk.trk = trk0; xx->trvTrk.dist = xx->trvTrk.length = -1; xx->trvTrk.pos = pos0; xx->trvTrk.angle = angle0; PlaceTrain(car0, FALSE, doCheckCoupling); } static void FlipTrain( track_p train) { DIST_T d0, d1; struct extraData * xx; if (train == NULL) { return; } d0 = GetTrainLength(train, 0); d1 = GetTrainLength(train, 1); xx = GetTrkExtraData(train); TraverseTrack2(&xx->trvTrk, d0-d1); FlipTraverseTrack(&xx->trvTrk); xx->trvTrk.length = -1; PlaceTrain(train, FALSE, TRUE); } static BOOL_T MoveTrain( track_p train, long timeD) { DIST_T ips, dist0, dist1; struct extraData *xx, *xx1; traverseTrack_t trvTrk; DIST_T length; track_p car1; int dir1; int measured; /* make sure the distance is only measured once per train */ if (train == NULL) { return FALSE; } xx = GetTrkExtraData(train); if (xx->speed <= 0) { return FALSE; } ips = ((xx->speed*5280.0*12.0)/(60.0*60.0*GetScaleRatio(GetLayoutCurScale()))); dist0 = ips * timeD/1000.0; length = GetTrainLength(train, xx->direction); dist1 = length + dist0; trvTrk = xx->trvTrk; if (trvTrk.trk == NULL) { return FALSE; } LOG(log_trainMove, 1, ("moveTrain: %s t%ld->%0.3f S%0.3f D%d [%0.3f %0.3f] A%0.3f T%d\n", CarItemNumber(xx->item), timeD, dist0, xx->speed, xx->direction, xx->trvTrk.pos.x, xx->trvTrk.pos.y, xx->trvTrk.angle, xx->trvTrk.trk?GetTrkIndex(xx->trvTrk.trk):-1)) if (xx->direction) { FlipTraverseTrack(&trvTrk); } TraverseTrack(&trvTrk, &dist1); if (dist1 > 0.0) { if (dist1 > dist0) { /*ErrorMessage( "%s no room: L%0.3f D%0.3f", CarItemNumber(xx->item), length, dist1 );*/ StopTrain(train, ST_NoRoom); return FALSE; } else { dist0 -= dist1; LOG(log_trainMove, 1, (" %s STOP D%d [%0.3f %0.3f] A%0.3f D%0.3f\n", CarItemNumber(xx->item), xx->direction, xx->trvTrk.pos.x, xx->trvTrk.pos.y, xx->trvTrk.angle, dist0)) } /*ErrorMessage( "%s stopped at End Of Track", CarItemNumber(xx->item) );*/ if (xx->autoReverse) { xx->direction = !xx->direction; SetTrainDirection(train); } else { if (xx->speed > maxCouplingSpeed) { car1 = train; dir1 = xx->direction; GetTrainLength2(&car1, &dir1); CrashTrain(car1, dir1, &trvTrk, (long)xx->speed, FALSE); return TRUE; } else { if (trvTrk.trk && trvTrk.trk->endCnt > 1) //Test for null track after Traverse StopTrain(train, ST_OpenTurnout ); else StopTrain(train, ST_EndOfTrack); return (FALSE); } } } trvTrk = xx->trvTrk; TraverseTrack2(&xx->trvTrk, xx->direction==0?dist0:-dist0); car1 = train; dir1 = 0; GetTrainLength2(&car1, &dir1); dir1 = 1-dir1; measured = FALSE; WALK_CARS_START(car1, xx1, dir1); if (CarItemIsLoco(xx1->item) && !measured) { xx->distance += dist0; measured = TRUE; } WALK_CARS_END(car1, xx1, dir1); if (train == followTrain) { if (followCenter.x != mainCenter.x || followCenter.y != mainCenter.y) { if (curTrainDlg->train == followTrain) { curTrainDlg->followMe = FALSE; ParamLoadControl(curTrainDlg->trainPGp, I_FOLLOW); } followTrain = NULL; } else if (OFF_MAIND(xx->trvTrk.pos, xx->trvTrk.pos)) { MoveMainWindow(xx->trvTrk.pos, NormalizeAngle(xx->trvTrk.angle+(xx->direction?180.0:0.0))); followCenter = mainCenter; } } PlaceTrain(train, TRUE, TRUE); return TRUE; } static BOOL_T MoveTrains(long timeD) { BOOL_T trains_moved = FALSE; track_p train; struct extraData * xx; for (train=NULL; TrackIterate(&train);) { if (GetTrkType(train) != T_CAR) { continue; } xx = GetTrkExtraData(train); if (!CarItemIsLoco(xx->item)) { continue; } if (!IsLocoMaster(xx)) { continue; } if (xx->speed == 0) { continue; } trains_moved |= MoveTrain(train, timeD); } ControllerDialogSyncAll(); TempRedraw(); // MoveTrains return trains_moved; } static void MoveTrainsLoop(void) { long time1, timeD; static long time0 = 0; trainsTimeoutPending = FALSE; if (trainsState != TRAINS_RUN) { time0 = 0; return; } if (time0 == 0) { time0 = wGetTimer(); } time1 = wGetTimer(); timeD = time1-time0; time0 = time1; if (timeD > 1000) { timeD = 1000; } if (MoveTrains(timeD)) { wAlarm(trainPause, MoveTrainsLoop); trainsTimeoutPending = TRUE; } else { time0 = 0; trainsState = TRAINS_IDLE; TrainTimeEndPause(); } } static void RestartTrains(void) { if (trainsState != TRAINS_RUN) { TrainTimeStartPause(); } trainsState = TRAINS_RUN; if (!trainsTimeoutPending) { MoveTrainsLoop(); } } static long trainTime0 = 0; static long playbackTrainPause = 0; static drawCmd_t trainMovieD = { NULL, &screenDrawFuncs, 0, 16.0, 0, {0,0}, {1,1}, Pix2CoOrd, CoOrd2Pix }; static long trainMovieFrameDelay; static long trainMovieFrameNext; static void TrainTimeEndPause(void) { if (recordF) { if (trainTime0 != 0) { long delay; delay = wGetTimer()-trainTime0; if (delay > 0) { fprintf(recordF, "TRAINPAUSE %ld\n", delay); } } trainTime0 = 0; } } static void TrainTimeStartPause(void) { if (trainTime0 == 0) { trainTime0 = wGetTimer(); } } static BOOL_T TrainTimeDoPause(char * line) { BOOL_T drawCarEnable2; playbackTrainPause = atol(line); LOG(log_trainPlayback, 1, ("DoPause %ld\n", playbackTrainPause)); trainsState = TRAINS_RUN; if (trainMovieFrameDelay > 0) { drawCarEnable2 = drawCarEnable; drawCarEnable = TRUE; TakeSnapshot(&trainMovieD); drawCarEnable = drawCarEnable2; LOG(log_trainPlayback, 1, ("SNAP 0\n")); trainMovieFrameNext = trainMovieFrameDelay; } /*MoveTrains();*/ while (playbackTrainPause > 0) { if (playbackTrainPause > trainPause) { wPause(trainPause); MoveTrains(trainPause); playbackTrainPause -= trainPause; if (trainMovieFrameDelay > 0) { trainMovieFrameNext -= trainPause; } } else { wPause(playbackTrainPause); MoveTrains(playbackTrainPause); if (trainMovieFrameDelay > 0) { trainMovieFrameNext -= playbackTrainPause; } playbackTrainPause = 0; } if (trainMovieFrameDelay > 0 && trainMovieFrameNext <= 0) { drawCarEnable2 = drawCarEnable; drawCarEnable = TRUE; TakeSnapshot(&trainMovieD); drawCarEnable = drawCarEnable2; LOG(log_trainPlayback, 1, ("SNAP %ld\n", trainMovieFrameNext)); trainMovieFrameNext = trainMovieFrameDelay; } } return TRUE; } static BOOL_T TrainDoMovie(char * line) { /* on/off, scale, orig, size */ long fps; if (trainMovieD.dpi == 0) { trainMovieD.dpi = mainD.dpi; } if (!GetArgs(line, "lfpp", &fps, &trainMovieD.scale, &trainMovieD.orig, &trainMovieD.size)) { return FALSE; } if (fps > 0) { trainMovieFrameDelay = 1000/fps; } else { trainMovieFrameDelay = 0; } trainMovieFrameNext = 0; return TRUE; } void AttachTrains(void) { track_p car; track_p loco; struct extraData * xx; coOrd pos; track_p trk; ANGLE_T angle; EPINX_T ep0, ep1; int dir; for (car=NULL; TrackIterate(&car);) { ClrTrkBits(car, TB_CARATTACHED); if (GetTrkType(car) != T_CAR) { continue; } xx = GetTrkExtraData(car); ClrProcessed(xx); } for (car=NULL; TrackIterate(&car);) { if (GetTrkType(car) != T_CAR) { continue; } xx = GetTrkExtraData(car); if (IsProcessed(xx)) { continue; } loco = FindMasterLoco(car, NULL); if (loco != NULL) { xx = GetTrkExtraData(loco); } else { loco = car; } pos = xx->trvTrk.pos; if (xx->status == ST_Crashed) { continue; } TRK_ITERATE(trk) { if (trk == xx->trvTrk.trk) { break; } } if (trk!=NULL && !QueryTrack(trk, Q_ISTRACK)) { trk = NULL; } if (trk==NULL || GetTrkDistance(trk,&pos)>trackGauge*2.0) { trk = OnTrack2(&pos, FALSE, TRUE, FALSE, NULL); } if (trk!=NULL) { /*if ( trk == xx->trvTrk.trk ) continue;*/ angle = GetAngleAtPoint(trk, pos, &ep0, &ep1); if (NormalizeAngle(xx->trvTrk.angle-angle+90) > 180) { angle = NormalizeAngle(angle+180); } PlaceTrainInit(loco, trk, pos, angle, TRUE); } else { PlaceTrainInit(loco, NULL, xx->trvTrk.pos, xx->trvTrk.angle, FALSE); } dir = 0; WALK_CARS_START(loco, xx, dir) WALK_CARS_END(loco, xx, dir) dir = 1-dir; WALK_CARS_START(loco, xx, dir) SetProcessed(xx); if (xx->trvTrk.trk) { SetTrkBits(xx->trvTrk.trk, TB_CARATTACHED); xx->status = ST_StopManual; } else { xx->status = ST_NotOnTrack; } WALK_CARS_END(loco, xx, dir) } for (car=NULL; TrackIterate(&car);) { if (GetTrkType(car) != T_CAR) { continue; } xx = GetTrkExtraData(car); ClrProcessed(xx); } } static void UpdateTrainAttachment(void) { track_p trk; struct extraData * xx; for (trk=NULL; TrackIterate(&trk);) { ClrTrkBits(trk, TB_CARATTACHED); } for (trk=NULL; TrackIterate(&trk);) { if (GetTrkType(trk) == T_CAR) { xx = GetTrkExtraData(trk); if (xx->trvTrk.trk != NULL) { SetTrkBits(xx->trvTrk.trk, TB_CARATTACHED); } } } } static BOOL_T TrainOnMovableTrack( track_p trk, track_p *trainR) { track_p train; struct extraData * xx; int dir; for (train=NULL; TrackIterate(&train);) { if (GetTrkType(train) != T_CAR) { continue; } xx = GetTrkExtraData(train); if (IsOnTrack(xx)) { if (xx->trvTrk.trk == trk) { break; } } } *trainR = train; if (train == NULL) { return TRUE; } dir = 0; WALK_CARS_START(train, xx, dir) WALK_CARS_END(train, xx, dir) dir = 1-dir; WALK_CARS_START(train, xx, dir) if (xx->trvTrk.trk != trk) { ErrorMessage(MSG_CANT_MOVE_UNDER_TRAIN); return FALSE; } WALK_CARS_END(train, xx, dir) train = FindMasterLoco(train, NULL); if (train != NULL) { *trainR = train; } return TRUE; } /* * */ #define DO_UNCOUPLE (0) #define DO_FLIPCAR (1) #define DO_FLIPTRAIN (2) #define DO_DELCAR (3) #define DO_DELTRAIN (4) #define DO_MUMASTER (5) #define DO_CHANGEDIR (6) #define DO_STOP (7) #define DO_PENCILS_ON (8) #define DO_PENCILS_OFF (9) static track_p trainFuncCar; static coOrd trainFuncPos; static wButton_p trainPauseB; static STATUS_T CmdTrain(wAction_t action, coOrd pos) { track_p trk0, trk1; static track_p currCar; coOrd pos0, pos1; static coOrd delta; ANGLE_T angle1; EPINX_T ep0, ep1; int dir; struct extraData * xx=NULL; wPos_t w, h; switch (action) { case C_START: /*UndoStart( "Trains", "Trains" );*/ UndoSuspend(); programMode = MODE_TRAIN; drawCarEnable = FALSE; doDrawTurnoutPosition = 1; DoChangeNotification(CHANGE_PARAMS|CHANGE_TOOLBAR); if (CarAvailableCount() <= 0) { if (NoticeMessage(MSG_NO_CARS, _("Yes"), _("No")) > 0) { DoCarDlg(); DoChangeNotification(CHANGE_PARAMS); } } EnableCommands(); if (curTrainDlg == NULL) { curTrainDlg = CreateTrainControlDlg(); } curTrainDlg->train = NULL; wListClear((wList_p)curTrainDlg->trainPGp->paramPtr[I_LIST].control); Dtrain.state = 0; trk0 = NULL; tempSegs_da.cnt = 0; DYNARR_SET(trkSeg_t, tempSegs_da, 8); RestartTrains(); wButtonSetLabel(trainPauseB, (char*)goI); trainTime0 = 0; AttachTrains(); curTrainDlg->train = NULL; curTrainDlg->speed = -1; wDrawClear((wDraw_p)curTrainDlg->trainPGp->paramPtr[I_SLIDER].control); LocoListInit(); ControllerDialogSync(curTrainDlg); wShow(curTrainDlg->win); wControlShow((wControl_p)newcarB, (toolbarSet&(1<<BG_HOTBAR)) == 0); currCarItemPtr = NULL; TempRedraw(); // CmdTrain C_START return C_CONTINUE; case C_TEXT: if (Dtrain.state == 0) { return C_CONTINUE; } else { return C_CONTINUE; } case C_DOWN: /*trainEnable = FALSE;*/ InfoMessage(""); if (trainsState == TRAINS_RUN) { trainsState = TRAINS_PAUSE; TrainTimeEndPause(); } pos0 = pos; if (currCarItemPtr != NULL) { DIST_T dist; currCar = NewCar(-1, currCarItemPtr, zero, 0.0); CarItemUpdate(currCarItemPtr); HotBarCancel(); if (currCar == NULL) { LOG1(log_error, ("Train: currCar became NULL 1\n")) return C_CONTINUE; } xx = GetTrkExtraData(currCar); xx->pencils = FALSE; dist = CarItemCoupledLength(xx->item)/2.0; Translate(&pos, xx->trvTrk.pos, xx->trvTrk.angle, dist); SetTrkEndPoint(currCar, 0, pos, xx->trvTrk.angle); Translate(&pos, xx->trvTrk.pos, xx->trvTrk.angle+180.0, dist); SetTrkEndPoint(currCar, 1, pos, NormalizeAngle(xx->trvTrk.angle+180.0)); /*xx->state |= (xx->item->options&CAR_DESC_BITS);*/ ClrLocoMaster(xx); if (CarItemIsLoco(xx->item)) { SetLocoMaster(xx); LocoListChangeEntry(NULL, currCar); } if ((trk0 = OnTrack(&pos0, FALSE, TRUE))) { xx->trvTrk.angle = GetAngleAtPoint(trk0, pos0, &ep0, &ep1); if (NormalizeAngle(FindAngle(pos, pos0) - xx->trvTrk.angle) > 180.0) { xx->trvTrk.angle = NormalizeAngle(xx->trvTrk.angle + 180); } xx->status = ST_StopManual; } else { xx->trvTrk.angle = 90; } PlaceTrainInit(currCar, trk0, pos0, xx->trvTrk.angle, (MyGetKeyState()&WKEY_SHIFT) == 0); /*DrawCars( &tempD, currCar, TRUE );*/ } else { currCar = FindCar(&pos); delta.x = pos.x - pos0.x; delta.y = pos.y - pos0.y; if (logTable(log_trainMove).level >= 1) { if (currCar) { xx = GetTrkExtraData(currCar); LogPrintf("selected %s\n", CarItemNumber(xx->item)); for (dir=0; dir<2; dir++) { int dir1 = dir; track_p car1 = currCar; struct extraData * xx1 = GetTrkExtraData(car1); LogPrintf("dir=%d\n", dir1); WALK_CARS_START(car1, xx1, dir1) LogPrintf(" %s [%0.3f,%d]\n", CarItemNumber(xx1->item), xx1->trvTrk.angle, dir1); WALK_CARS_END(car1, xx1, dir1) } } } } if (currCar == NULL) { return C_CONTINUE; } trk0 = FindMasterLoco(currCar, NULL); if (trk0) { SetCurTrain(trk0); } return C_CONTINUE; case C_MOVE: if (currCar == NULL) { return C_CONTINUE; } pos.x += delta.x; pos.y += delta.y; pos0 = pos; xx = GetTrkExtraData(currCar); trk0 = OnTrack(&pos0, FALSE, TRUE); if (/*currCarItemPtr != NULL &&*/ trk0) { angle1 = GetAngleAtPoint(trk0, pos0, &ep0, &ep1); if (currCarItemPtr != NULL) { if (NormalizeAngle(FindAngle(pos, pos0) - angle1) > 180.0) { angle1 = NormalizeAngle(angle1 + 180); } } else { if (NormalizeAngle(xx->trvTrk.angle - angle1 + 90.0) > 180.0) { angle1 = NormalizeAngle(angle1 + 180); } } xx->trvTrk.angle = angle1; } tempSegs_da.cnt = 1; PlaceTrainInit(currCar, trk0, pos0, xx->trvTrk.angle, (MyGetKeyState()&WKEY_SHIFT) == 0); ControllerDialogSync(curTrainDlg); return C_CONTINUE; case C_UP: if (currCar != NULL) { trk0 = FindMasterLoco(currCar, NULL); if (trk0) { xx = GetTrkExtraData(trk0); if (!IsOnTrack(xx) || xx->speed <= 0) { StopTrain(trk0, ST_StopManual); } } Dtrain.state = 1; ControllerDialogSync(curTrainDlg); } InfoSubstituteControls(NULL, NULL); currCar = trk0 = NULL; currCarItemPtr = NULL; /*trainEnable = TRUE;*/ if (trainsState == TRAINS_PAUSE) { RestartTrains(); } return C_CONTINUE; case C_LCLICK: if (MyGetKeyState() & WKEY_SHIFT) { pos0 = pos; programMode = MODE_DESIGN; if ((trk0=OnTrack(&pos,FALSE,TRUE)) && QueryTrack(trk0, Q_CAN_NEXT_POSITION) && TrainOnMovableTrack(trk0, &trk1)) { if (trk1) { xx = GetTrkExtraData(trk1); pos1 = xx->trvTrk.pos; angle1 = xx->trvTrk.angle; } else { pos1 = pos0; angle1 = 0; } AdvancePositionIndicator(trk0, pos0, &pos1, &angle1); if (trk1) { xx->trvTrk.pos = pos1; xx->trvTrk.angle = angle1; PlaceTrain(trk1, FALSE, TRUE); } } programMode = MODE_TRAIN; trk0 = NULL; MainRedraw(); //CmdTrain: Make sure track is redrawn after switch thrown } else { trk0 = FindCar(&pos); if (trk0 == NULL) { return C_CONTINUE; } trk0 = FindMasterLoco(trk0, NULL); if (trk0 == NULL) { return C_CONTINUE; } SetCurTrain(trk0); } return C_CONTINUE; case C_RCLICK: trainFuncPos = pos; trainFuncCar = FindCar(&pos); if (trainFuncCar == NULL || GetTrkType(trainFuncCar) != T_CAR) { return C_CONTINUE; } xx = GetTrkExtraData(trainFuncCar); if (xx->pencils) { wMenuPushEnable(trainPopupMI[DO_PENCILS_OFF], TRUE); wMenuPushEnable(trainPopupMI[DO_PENCILS_ON], FALSE); } else { wMenuPushEnable(trainPopupMI[DO_PENCILS_OFF], FALSE); wMenuPushEnable(trainPopupMI[DO_PENCILS_ON], TRUE); } trk0 = FindMasterLoco(trainFuncCar,NULL); dir = IsAligned(xx->trvTrk.angle, FindAngle(xx->trvTrk.pos, trainFuncPos)) ? 0 : 1; wMenuPushEnable(trainPopupMI[DO_UNCOUPLE], GetTrkEndTrk(trainFuncCar, dir)!=NULL); wMenuPushEnable(trainPopupMI[DO_MUMASTER], CarItemIsLoco(xx->item) && !IsLocoMaster(xx)); if (trk0) { xx = GetTrkExtraData(trk0); } wMenuPushEnable(trainPopupMI[DO_CHANGEDIR], trk0!=NULL); wMenuPushEnable(trainPopupMI[DO_STOP], trk0!=NULL && xx->speed>0); /*trainEnable = FALSE;*/ trk0 = FindMasterLoco(trainFuncCar, NULL); if (trk0) { SetCurTrain(trk0); } if (!inPlayback) { wMenuPopupShow(trainPopupM); } return C_CONTINUE; case C_REDRAW: wDrawSaveImage(mainD.d); DrawAllCars(); wWinGetSize(mainW, &w, &h); w -= wControlGetPosX(newCarControls[0]) + 4; if (w > 20) { wListSetSize((wList_p)newCarControls[0], w, wControlGetHeight(newCarControls[0])); } return C_CONTINUE; case C_CANCEL: /*trainEnable = FALSE;*/ trainsState = TRAINS_STOP; TrainTimeEndPause(); LOG(log_trainMove, 1, ("Train Cancel\n")) Dtrain.state = 0; doDrawTurnoutPosition = 0; drawCarEnable = TRUE; programMode = MODE_DESIGN; UpdateTrainAttachment(); UndoResume(); DoChangeNotification(CHANGE_PARAMS|CHANGE_TOOLBAR); if (curTrainDlg->win) { wHide(curTrainDlg->win); } MainRedraw(); // CmdTrain: Exit curTrainDlg->train = NULL; return C_CONTINUE; case C_CONFIRM: /*trainEnable = FALSE;*/ if (trainsState != TRAINS_STOP) { trainsState = TRAINS_STOP; wButtonSetLabel(trainPauseB, (char*)stopI); TrainTimeEndPause(); } currCar = NULL; currCarItemPtr = NULL; HotBarCancel(); InfoSubstituteControls(NULL, NULL); return C_TERMINATE; } return C_CONTINUE; } /* * */ STATUS_T CmdCarDescAction( wAction_t action, coOrd pos) { return CmdTrain(action, pos); } #include "bitmaps/train.xpm" #include "bitmaps/exit.xpm" #include "bitmaps/newcar.xpm" #include "bitmaps/zero.xpm" #include "bitmaps/ballgreen.xpm" #include "bitmaps/ballred.xpm" static void CmdTrainStopGo(void * junk) { wIcon_p icon; if (trainsState == TRAINS_STOP) { icon = goI; RestartTrains(); } else { trainsState = TRAINS_STOP; icon = stopI; TrainTimeEndPause(); } ControllerDialogSync(curTrainDlg); wButtonSetLabel(trainPauseB, (char*)icon); if (recordF) { fprintf(recordF, "TRAINSTOPGO %s\n", trainsState==TRAINS_STOP?"STOP":"GO"); } } static BOOL_T TrainStopGoPlayback(char * line) { while (*line && isspace((unsigned char)*line)) { line++; } if ((strcasecmp(line, "STOP") == 0) != (trainsState == TRAINS_STOP)) { CmdTrainStopGo(NULL); } return TRUE; } static void CmdTrainExit(void * junk) { Reset(); InfoSubstituteControls(NULL, NULL); } static void TrainFunc( void * action) { struct extraData * xx, *xx1; ANGLE_T angle; int dir; track_p loco; track_p temp0, temp1; coOrd pos0, pos1; ANGLE_T angle0, angle1; EPINX_T ep0=-1, ep1=-1; if (trainFuncCar == NULL) { fprintf(stderr, "trainFunc: trainFuncCar==NULL\n"); return; } xx = GetTrkExtraData(trainFuncCar); angle = FindAngle(xx->trvTrk.pos, trainFuncPos); angle = NormalizeAngle(angle-xx->trvTrk.angle); dir = (angle>90&&angle<270); switch ((int)(long)action) { case DO_UNCOUPLE: if (GetTrkEndTrk(trainFuncCar,dir)) { UncoupleCars(trainFuncCar, GetTrkEndTrk(trainFuncCar,dir)); } break; case DO_PENCILS_ON: xx->pencils = TRUE; break; case DO_PENCILS_OFF: xx->pencils = FALSE; break; case DO_FLIPCAR: temp0 = GetTrkEndTrk(trainFuncCar,0); pos0 = GetTrkEndPos(trainFuncCar,0); angle0 = GetTrkEndAngle(trainFuncCar,0); temp1 = GetTrkEndTrk(trainFuncCar,1); pos1 = GetTrkEndPos(trainFuncCar,1); angle1 = GetTrkEndAngle(trainFuncCar,1); if (temp0) { ep0 = GetEndPtConnectedToMe(temp0,trainFuncCar); trainFuncCar->endPt[0].track = NULL; temp0->endPt[ep0].track = NULL; } if (temp1) { ep1 = GetEndPtConnectedToMe(temp1,trainFuncCar); trainFuncCar->endPt[1].track = NULL; temp1->endPt[ep1].track = NULL; } xx->direction = !xx->direction; FlipTraverseTrack(&xx->trvTrk); SetTrkEndPoint(trainFuncCar, 0, pos1, angle1); SetTrkEndPoint(trainFuncCar, 1, pos0, angle0); if (temp0) { trainFuncCar->endPt[1].track = temp0; temp0->endPt[ep0].track = trainFuncCar; } if (temp1) { trainFuncCar->endPt[0].track = temp1; temp1->endPt[ep1].track = trainFuncCar; } ControllerDialogSync(curTrainDlg); PlaceCar(trainFuncCar); break; case DO_FLIPTRAIN: FlipTrain(trainFuncCar); /*PlaceTrain( trainFuncCar, xx->trk, xx->trvTrk.pos, xx->trvTrk.angle );*/ break; case DO_DELCAR: for (dir=0; dir<2; dir++) if (GetTrkEndTrk(trainFuncCar,dir)) { UncoupleCars(trainFuncCar, GetTrkEndTrk(trainFuncCar,dir)); } if (CarItemIsLoco(xx->item)) { LocoListChangeEntry(trainFuncCar, NULL); } trainFuncCar->deleted = TRUE; /*DeleteTrack( trainFuncCar, FALSE );*/ CarItemUpdate(xx->item); HotBarCancel(); InfoSubstituteControls(NULL, NULL); break; case DO_DELTRAIN: dir = 0; loco = FindMasterLoco(trainFuncCar, NULL); WALK_CARS_START(trainFuncCar, xx, dir) WALK_CARS_END(trainFuncCar, xx, dir) dir = 1-dir; temp0 = NULL; WALK_CARS_START(trainFuncCar, xx, dir) if (temp0) { xx1 = GetTrkExtraData(temp0); temp0->deleted = TRUE; /*DeleteTrack( temp0, FALSE );*/ CarItemUpdate(xx1->item); } temp0 = trainFuncCar; WALK_CARS_END(trainFuncCar, xx, dir) if (temp0) { xx1 = GetTrkExtraData(temp0); temp0->deleted = TRUE; /*DeleteTrack( temp0, FALSE );*/ CarItemUpdate(xx1->item); } if (loco) { LocoListChangeEntry(loco, NULL); } HotBarCancel(); InfoSubstituteControls(NULL, NULL); break; case DO_MUMASTER: if (CarItemIsLoco(xx->item)) { loco = FindMasterLoco(trainFuncCar, NULL); if (loco != trainFuncCar) { SetLocoMaster(xx); LOG(log_trainMove, 1, ("%s gets master\n", CarItemNumber(xx->item))) if (loco) { xx1 = GetTrkExtraData(loco); ClrLocoMaster(xx1); LOG(log_trainMove, 1, ("%s looses master\n", CarItemNumber(xx1->item))) xx->speed = xx1->speed; xx1->speed = 0; } LocoListChangeEntry(loco, trainFuncCar); } } break; case DO_CHANGEDIR: loco = FindMasterLoco(trainFuncCar, NULL); if (loco) { xx = GetTrkExtraData(loco); xx->direction = !xx->direction; SetTrainDirection(loco); ControllerDialogSync(curTrainDlg); } break; case DO_STOP: loco = FindMasterLoco(trainFuncCar, NULL); if (loco) { StopTrain(loco, ST_StopManual); ControllerDialogSync(curTrainDlg); } break; } MainRedraw(); //TrainFunc: Redraw if Train altered if (trainsState == TRAINS_PAUSE) { RestartTrains(); } else { } } EXPORT wIndex_t trainCmdInx; void InitCmdTrain(wMenu_p menu) { log_trainMove = LogFindIndex("trainMove"); log_trainPlayback = LogFindIndex("trainPlayback"); trainPLs[I_ZERO].winLabel = (char*)wIconCreatePixMap(zero_xpm); ParamRegister(&trainPG); trainCmdInx = AddMenuButton(menu, CmdTrain, "cmdTrain", _("Train"), wIconCreatePixMap(train_xpm), LEVEL0_50, IC_POPUP3|IC_LCLICK|IC_RCLICK, 0, NULL); stopI = wIconCreatePixMap(ballred); goI = wIconCreatePixMap(ballgreen); trainPauseB = AddToolbarButton("cmdTrainPause", stopI, IC_MODETRAIN_ONLY, CmdTrainStopGo, NULL); AddToolbarButton("cmdTrainExit", wIconCreatePixMap(exit_xpm), IC_MODETRAIN_ONLY, CmdTrainExit, NULL); newcarB = AddToolbarButton("cmdTrainNewCar", wIconCreatePixMap(newcar_xpm), IC_MODETRAIN_ONLY, CarItemLoadList, NULL); T_CAR = InitObject(&carCmds); trainPopupM = MenuRegister("Train Commands"); trainPopupMI[DO_UNCOUPLE] = wMenuPushCreate(trainPopupM, "", _("Uncouple"), 0, TrainFunc, (void*)DO_UNCOUPLE); trainPopupMI[DO_FLIPCAR] = wMenuPushCreate(trainPopupM, "", _("Flip Car"), 0, TrainFunc, (void*)DO_FLIPCAR); trainPopupMI[DO_PENCILS_ON] = wMenuPushCreate(trainPopupM, "", _("Clearance Lines On"), 0, TrainFunc, (void*)DO_PENCILS_ON); trainPopupMI[DO_PENCILS_OFF] = wMenuPushCreate(trainPopupM, "", _("Clearance Lines Off"), 0, TrainFunc, (void*)DO_PENCILS_OFF); trainPopupMI[DO_FLIPTRAIN] = wMenuPushCreate(trainPopupM, "", _("Flip Train"), 0, TrainFunc, (void*)DO_FLIPTRAIN); trainPopupMI[DO_MUMASTER] = wMenuPushCreate(trainPopupM, "", _("MU Master"), 0, TrainFunc, (void*)DO_MUMASTER); trainPopupMI[DO_CHANGEDIR] = wMenuPushCreate(trainPopupM, "", _("Change Direction"), 0, TrainFunc, (void*)DO_CHANGEDIR); trainPopupMI[DO_STOP] = wMenuPushCreate(trainPopupM, "", _("Stop"), 0, TrainFunc, (void*)DO_STOP); wMenuSeparatorCreate(trainPopupM); trainPopupMI[DO_DELCAR] = wMenuPushCreate(trainPopupM, "", _("Remove Car"), 0, TrainFunc, (void*)DO_DELCAR); trainPopupMI[DO_DELTRAIN] = wMenuPushCreate(trainPopupM, "", _("Remove Train"), 0, TrainFunc, (void*)DO_DELTRAIN); AddPlaybackProc("TRAINSTOPGO", (playbackProc_p)TrainStopGoPlayback, NULL); AddPlaybackProc("TRAINPAUSE", (playbackProc_p)TrainTimeDoPause, NULL); AddPlaybackProc("TRAINMOVIE", (playbackProc_p)TrainDoMovie, NULL); }