| author | darkvater | 
| Mon, 16 Aug 2004 22:15:44 +0000 | |
| changeset 69 | f8c29cdb388e | 
| parent 39 | d177340ed556 | 
| child 72 | 24bdf837198c | 
| permissions | -rw-r--r-- | 
| 0 | 1 | #include "stdafx.h" | 
| 2 | #include "ttd.h" | |
| 3 | #include "vehicle.h" | |
| 4 | #include "viewport.h" | |
| 5 | #include "command.h" | |
| 6 | #include "player.h" | |
| 7 | #include "town.h" | |
| 8 | #include "gfx.h" | |
| 9 | ||
| 22 | 10 | /* When true, GetTrackStatus for roads will treat roads under reconstruction | 
| 11 | * as normal roads instead of impassable. This is used when detecting whether | |
| 12 | * a road can be removed. This is of course ugly, but I don't know a better | |
| 13 | * solution just like that... */ | |
| 0 | 14 | static bool _road_special_gettrackstatus; | 
| 15 | ||
| 16 | void RoadVehEnterDepot(Vehicle *v); | |
| 17 | ||
| 18 | ||
| 19 | static bool HasTileRoadAt(uint tile, int i) | |
| 20 | {
 | |
| 21 | int mask; | |
| 22 | byte b; | |
| 23 | ||
| 24 | 	switch(GET_TILETYPE(tile)) {
 | |
| 25 | case MP_STREET: | |
| 26 | b = _map5[tile]; | |
| 27 | ||
| 28 | 		if ((b & 0xF0) == 0) {
 | |
| 29 | 		} else if ((b & 0xF0) == 0x10) {
 | |
| 30 | b = (b&8)?5:10; | |
| 31 | 		} else if ((b & 0xF0) == 0x20) {
 | |
| 32 | return (~b & 3) == i; | |
| 33 | } else | |
| 34 | return false; | |
| 35 | break; | |
| 36 | ||
| 37 | case MP_STATION: | |
| 38 | b = _map5[tile]; | |
| 39 | if (!IS_BYTE_INSIDE(b, 0x43, 0x43+8)) | |
| 40 | return false; | |
| 41 | return ((~(b - 0x43) & 3) == i); | |
| 42 | ||
| 43 | case MP_TUNNELBRIDGE: | |
| 44 | mask = GetRoadBitsByTile(tile); | |
| 45 | b = 10; if (mask & 1) break; | |
| 46 | b = 5; if (mask & 2) break; | |
| 47 | return false; | |
| 48 | ||
| 49 | default: | |
| 50 | return false; | |
| 51 | } | |
| 52 | ||
| 53 | return HASBIT(b, i); | |
| 54 | } | |
| 55 | ||
| 56 | static bool CheckAllowRemoveRoad(uint tile, uint br, bool *edge_road) | |
| 57 | {
 | |
| 58 | int blocks; | |
| 59 | byte owner; | |
| 60 | uint n; | |
| 61 | *edge_road = true; | |
| 62 | ||
| 63 | if (_game_mode == GM_EDITOR) | |
| 64 | return true; | |
| 65 | ||
| 66 | blocks = GetRoadBitsByTile(tile); | |
| 67 | if (blocks == 0) | |
| 68 | return true; | |
| 69 | ||
| 70 | // Only do the special processing for actual players. | |
| 71 | if (_current_player >= MAX_PLAYERS) | |
| 72 | return true; | |
| 73 | ||
| 74 | // A railway crossing has the road owner in the map3_lo byte. | |
| 75 | 	if (IS_TILETYPE(tile, MP_STREET) && (_map5[tile] & 0xF0) == 0x10) {
 | |
| 76 | owner = _map3_lo[tile]; | |
| 77 | 	} else {
 | |
| 78 | owner = _map_owner[tile]; | |
| 79 | } | |
| 80 | // Only do the special processing if the road is owned | |
| 81 | // by a town | |
| 82 | 	if (owner != OWNER_TOWN) {
 | |
| 83 | return owner == OWNER_NONE || CheckOwnership(owner); | |
| 84 | } | |
| 85 | ||
| 86 | if (_cheats.magic_bulldozer.value) | |
| 87 | return true; | |
| 88 | ||
| 89 | // Get a bitmask of which neighbouring roads has a tile | |
| 90 | n = 0; | |
| 91 | if (blocks&0x25 && HasTileRoadAt(TILE_ADDXY(tile,-1, 0), 1)) n |= 8; | |
| 92 | if (blocks&0x2A && HasTileRoadAt(TILE_ADDXY(tile, 0, 1), 0)) n |= 4; | |
| 93 | if (blocks&0x19 && HasTileRoadAt(TILE_ADDXY(tile, 1, 0), 3)) n |= 2; | |
| 94 | if (blocks&0x16 && HasTileRoadAt(TILE_ADDXY(tile, 0,-1), 2)) n |= 1; | |
| 95 | ||
| 96 | // If 0 or 1 bits are set in n, or if no bits that match the bits to remove, | |
| 97 | // then allow it | |
| 98 | 	if ((n & (n-1)) != 0 && (n & br) != 0) {
 | |
| 99 | Town *t; | |
| 100 | *edge_road = false; | |
| 101 | // you can remove all kind of roads with extra dynamite | |
| 102 | if (_patches.extra_dynamite) | |
| 103 | return true; | |
| 104 | ||
| 105 | t = ClosestTownFromTile(tile, (uint)-1); | |
| 106 | SET_DPARAM16(0, t->index); | |
| 107 | _error_message = STR_2009_LOCAL_AUTHORITY_REFUSES; | |
| 108 | return false; | |
| 109 | } | |
| 110 | ||
| 111 | return true; | |
| 112 | } | |
| 113 | ||
| 114 | bool IsRoadDepotTile(TileIndex tile) | |
| 115 | {
 | |
| 116 | return IS_TILETYPE(tile, MP_STREET) && | |
| 117 | (_map5[tile] & 0xF0) == 0x20; | |
| 118 | } | |
| 119 | ||
| 120 | uint GetRoadBitsByTile(TileIndex tile) | |
| 121 | {
 | |
| 122 | uint32 r = GetTileTrackStatus(tile, 2); | |
| 123 | return (byte)(r | (r >> 8)); | |
| 124 | } | |
| 125 | ||
| 126 | // cost for removing inner/edge -roads | |
| 127 | static const uint16 _road_remove_cost[2] = {50, 18};
 | |
| 128 | ||
| 129 | /* Delete a piece of road | |
| 130 | * p1 = piece type | |
| 131 | */ | |
| 132 | int32 CmdRemoveRoad(int x, int y, uint32 flags, uint32 p1, uint32 p2) | |
| 133 | {
 | |
| 134 | TileInfo ti; | |
| 135 | int32 cost; | |
| 136 | uint tile; | |
| 137 | Town *t; | |
| 138 | /* true if the roadpiece was always removeable, | |
| 139 | false if it was a center piece. Affects town ratings drop | |
| 140 | */ | |
| 141 | bool edge_road; | |
| 142 | ||
| 143 | SET_EXPENSES_TYPE(EXPENSES_CONSTRUCTION); | |
| 144 | ||
| 145 | FindLandscapeHeight(&ti, x, y); | |
| 146 | tile = ti.tile; | |
| 147 | t = ClosestTownFromTile(tile, (uint)-1); // needed for town rating penalty | |
| 148 | ||
| 149 | // allow deleting road under bridge | |
| 150 | if (ti.type != MP_TUNNELBRIDGE && !EnsureNoVehicle(tile)) | |
| 151 | return CMD_ERROR; | |
| 152 | ||
| 153 | 	{
 | |
| 154 | bool b; | |
| 155 | _road_special_gettrackstatus = true; | |
| 156 | b = CheckAllowRemoveRoad(tile, p1, &edge_road); | |
| 157 | _road_special_gettrackstatus = false; | |
| 158 | if (!b) | |
| 159 | return CMD_ERROR; | |
| 160 | } | |
| 161 | ||
| 162 | 	if (ti.type == MP_TUNNELBRIDGE) {
 | |
| 163 | if (!EnsureNoVehicleZ(tile, GET_TILEHEIGHT(tile))) | |
| 164 | return CMD_ERROR; | |
| 165 | ||
| 166 | 		if ((ti.map5 & 0xE9) == 0xE8) {
 | |
| 167 | if (p1 & 10) goto return_error; | |
| 168 | 		} else if ((ti.map5 & 0xE9) == 0xE9) {
 | |
| 169 | if (p1 & 5) goto return_error; | |
| 170 | } else | |
| 171 | goto return_error; | |
| 172 | ||
| 173 | cost = _price.remove_road * 2; | |
| 174 | ||
| 175 | 		if (flags & DC_EXEC) {
 | |
| 176 | _map5[tile] = ti.map5 & 0xC7; | |
| 177 | _map_owner[tile] = OWNER_NONE; | |
| 178 | MarkTileDirtyByTile(tile); | |
| 179 | } | |
| 180 | return cost; | |
| 181 | 	} else if (ti.type == MP_STREET) {
 | |
| 182 | 		if (!(ti.map5 & 0xF0)) {
 | |
| 183 | uint c = p1, t2; | |
| 184 | ||
| 185 | 			if (ti.tileh != 0  && (ti.map5 == 5 || ti.map5 == 10)) {
 | |
| 186 | c |= (c & 0xC) >> 2; | |
| 187 | c |= (c & 0x3) << 2; | |
| 188 | } | |
| 189 | ||
| 190 | // limit the bits to delete to the existing bits. | |
| 191 | if ((c &= ti.map5) == 0) goto return_error; | |
| 192 | ||
| 193 | // calculate the cost | |
| 194 | t2 = c; | |
| 195 | cost = 0; | |
| 196 | 			do {
 | |
| 197 | if (t2&1) cost += _price.remove_road; | |
| 198 | } while(t2>>=1); | |
| 199 | ||
| 200 | // check if you're allowed to remove the street owned by a town | |
| 201 | // removal allowance depends on difficulty setting | |
| 202 | 			if(_map_owner[tile] == OWNER_TOWN && _game_mode != GM_EDITOR) {
 | |
| 203 | if (!CheckforTownRating(tile, flags, t, ROAD_REMOVE)) | |
| 204 | return CMD_ERROR; | |
| 205 | } | |
| 206 | ||
| 207 | 			if (flags & DC_EXEC) {
 | |
| 208 | // checks if the owner is town than decrease town rating by 50 until you have | |
| 209 | // a "Poor" town rating | |
| 210 | if(_map_owner[tile] == OWNER_TOWN && _game_mode != GM_EDITOR) | |
| 211 | ChangeTownRating(t, -_road_remove_cost[(byte)edge_road], -100); | |
| 212 | ||
| 213 | _map5[tile] ^= c; | |
| 214 | if ((_map5[tile]&0xF) == 0) | |
| 215 | DoClearSquare(tile); | |
| 216 | else | |
| 217 | MarkTileDirtyByTile(tile); | |
| 218 | ||
| 219 | } | |
| 220 | return cost; | |
| 221 | 		} else if (!(ti.map5 & 0xE0)) {
 | |
| 222 | byte c; | |
| 223 | ||
| 224 | 			if (!(ti.map5 & 8)) {
 | |
| 225 | c = 2; | |
| 226 | if (p1 & 5) goto return_error; | |
| 227 | 			} else {
 | |
| 228 | c = 1; | |
| 229 | if (p1 & 10) goto return_error; | |
| 230 | } | |
| 231 | ||
| 232 | cost = _price.remove_road * 2; | |
| 233 | 			if (flags & DC_EXEC) {
 | |
| 234 | ModifyTile(tile, | |
| 235 | MP_SETTYPE(MP_RAILWAY) | | |
| 236 | MP_MAP2_CLEAR | MP_MAP3LO | MP_MAP3HI_CLEAR | MP_MAP5, | |
| 237 | _map3_hi[tile] & 0xF, /* map3_lo */ | |
| 238 | c /* map5 */ | |
| 239 | ); | |
| 240 | } | |
| 241 | return cost; | |
| 242 | } else | |
| 243 | goto return_error; | |
| 244 | ||
| 245 | 	} else {
 | |
| 246 | return_error:; | |
| 247 | return_cmd_error(INVALID_STRING_ID); | |
| 248 | } | |
| 249 | } | |
| 250 | ||
| 251 | ||
| 252 | enum {
 | |
| 253 | ROAD_NW = 1, // NW road track | |
| 254 | ROAD_SW = 2, // SW road track | |
| 255 | ROAD_SE = 4, // SE road track | |
| 256 | ROAD_NE = 8, // NE road track | |
| 257 | ROAD_ALL = (ROAD_NW | ROAD_SW | ROAD_SE | ROAD_NE) | |
| 258 | }; | |
| 259 | ||
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 260 | static const byte _valid_tileh_slopes_road[3][15] = {
 | 
| 0 | 261 | // set of normal ones | 
| 262 | 	{
 | |
| 263 | ROAD_ALL, 0, 0, | |
| 264 | ROAD_SW | ROAD_NE, 0, 0, // 3, 4, 5 | |
| 265 | ROAD_NW | ROAD_SE, 0, 0, | |
| 266 | ROAD_NW | ROAD_SE, 0, 0, // 9, 10, 11 | |
| 267 | ROAD_SW | ROAD_NE, 0, 0 | |
| 268 | }, | |
| 269 | // allowed road for an evenly raised platform | |
| 270 | 	{ 
 | |
| 271 | 0, | |
| 272 | ROAD_SW | ROAD_NW, | |
| 273 | ROAD_SW | ROAD_SE, | |
| 274 | ROAD_NW | ROAD_SE | ROAD_SW, | |
| 275 | ||
| 276 | ROAD_SE | ROAD_NE, // 4 | |
| 277 | ROAD_ALL, | |
| 278 | ROAD_SW | ROAD_NE | ROAD_SE, | |
| 279 | ROAD_ALL, | |
| 280 | ||
| 281 | ROAD_NW | ROAD_NE, // 8 | |
| 282 | ROAD_SW | ROAD_NE | ROAD_NW, | |
| 283 | ROAD_ALL, | |
| 284 | ROAD_ALL, | |
| 285 | ||
| 286 | ROAD_NW | ROAD_SE | ROAD_NE, // 12 | |
| 287 | ROAD_ALL, | |
| 288 | ROAD_ALL | |
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 289 | }, | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 290 | // valid railway crossings on slopes | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 291 | 	{
 | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 292 | 1, 0, 0, // 0, 1, 2 | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 293 | 0, 0, 1, // 3, 4, 5 | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 294 | 0, 1, 0, // 6, 7, 8 | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 295 | 0, 1, 1, // 9, 10, 11 | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 296 | 0, 1, 1, // 12, 13, 14 | 
| 0 | 297 | } | 
| 298 | }; | |
| 299 | ||
| 300 | ||
| 301 | static uint32 CheckRoadSlope(int tileh, byte *pieces, byte existing) | |
| 302 | {
 | |
| 303 | 	if (!(tileh & 0x10)) {
 | |
| 304 | byte road_bits = *pieces | existing; | |
| 305 | ||
| 306 | // no special foundation | |
| 307 | 		if ((~_valid_tileh_slopes_road[0][tileh] & road_bits) == 0) {
 | |
| 308 | // force that all bits are set when we have slopes | |
| 309 | if (tileh != 0) *pieces |= _valid_tileh_slopes_road[0][tileh]; | |
| 310 | return 0; // no extra cost | |
| 311 | } | |
| 312 | ||
| 313 | // foundation is used. Whole tile is leveled up | |
| 314 | 		if ((~_valid_tileh_slopes_road[1][tileh] & road_bits) == 0) {
 | |
| 315 | return existing ? 0 : _price.terraform; | |
| 316 | } | |
| 317 | ||
| 318 | // partly leveled up tile, only if there's no road on that tile | |
| 319 | 		if ( !existing && (tileh == 1 || tileh == 2 || tileh == 4 || tileh == 8) ) {
 | |
| 320 | // force full pieces. | |
| 321 | *pieces |= (*pieces & 0xC) >> 2; | |
| 322 | *pieces |= (*pieces & 0x3) << 2; | |
| 323 | return existing ? 0 : _price.terraform; | |
| 324 | } | |
| 325 | } | |
| 326 | return CMD_ERROR; | |
| 327 | } | |
| 328 | ||
| 329 | /* Build a piece of road | |
| 330 | * p1 = piece flags | |
| 331 | */ | |
| 332 | ||
| 333 | int32 CmdBuildRoad(int x, int y, uint32 flags, uint32 p1, uint32 p2) | |
| 334 | {
 | |
| 335 | TileInfo ti; | |
| 336 | int32 cost; | |
| 337 | byte pieces = (byte)p1, existing = 0; | |
| 338 | uint tile; | |
| 339 | ||
| 340 | SET_EXPENSES_TYPE(EXPENSES_CONSTRUCTION); | |
| 341 | ||
| 342 | FindLandscapeHeight(&ti, x, y); | |
| 343 | tile = ti.tile; | |
| 344 | ||
| 345 | // allow building road under bridge | |
| 346 | if (ti.type != MP_TUNNELBRIDGE && !EnsureNoVehicle(tile)) | |
| 347 | return CMD_ERROR; | |
| 348 | ||
| 349 | 	if (ti.type == MP_STREET) {
 | |
| 350 | 		if (!(ti.map5 & 0xF0)) {
 | |
| 351 | if ( ((pieces) & (byte)(ti.map5)) == (pieces)) | |
| 352 | return_cmd_error(STR_1007_ALREADY_BUILT); | |
| 353 | existing = ti.map5; | |
| 354 | 		} else {
 | |
| 355 | if (!(ti.map5 & 0xE0) && pieces != ((ti.map5 & 8) ? 5 : 10)) | |
| 356 | return_cmd_error(STR_1007_ALREADY_BUILT); | |
| 357 | goto do_clear; | |
| 358 | } | |
| 359 | 	} else if (ti.type == MP_RAILWAY) {
 | |
| 360 | byte m5; | |
| 361 | ||
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 362 | if (ti.tileh & 0x10) // very steep tile | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 363 | return_cmd_error(STR_1000_LAND_SLOPED_IN_WRONG_DIRECTION); | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 364 | |
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 365 | if(!_valid_tileh_slopes_road[2][ti.tileh]) // prevent certain slopes | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 366 | return_cmd_error(STR_1000_LAND_SLOPED_IN_WRONG_DIRECTION); | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 367 | |
| 0 | 368 | 		if (ti.map5 == 2) {
 | 
| 369 | if (pieces & 5) goto do_clear; | |
| 370 | m5 = 0x10; | |
| 371 | 		} else if (ti.map5 == 1) {
 | |
| 372 | if (pieces & 10) goto do_clear; | |
| 373 | m5 = 0x18; | |
| 374 | } else | |
| 375 | goto do_clear; | |
| 376 | ||
| 377 | 		if (flags & DC_EXEC) {
 | |
| 378 | ModifyTile(tile, | |
| 379 | MP_SETTYPE(MP_STREET) | | |
| 380 | MP_MAP2_CLEAR | MP_MAP3LO | MP_MAP3HI | MP_MAP5, | |
| 381 | _current_player, /* map3_lo */ | |
| 382 | _map3_lo[tile] & 0xF, /* map3_hi */ | |
| 383 | m5 /* map5 */ | |
| 384 | ); | |
| 385 | } | |
| 386 | return _price.build_road * 2; | |
| 387 | 	} else if (ti.type == MP_TUNNELBRIDGE) {
 | |
| 388 | ||
| 389 | /* check for flat land */ | |
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 390 | if (ti.tileh & 0x10) // very steep tile | 
| 0 | 391 | return_cmd_error(STR_1000_LAND_SLOPED_IN_WRONG_DIRECTION); | 
| 392 | ||
| 393 | /* is this middle part of a bridge? */ | |
| 394 | if ((ti.map5 & 0xC0) != 0xC0) | |
| 395 | goto do_clear; | |
| 396 | ||
| 397 | /* only allow roads pertendicular to bridge */ | |
| 398 | if ((pieces & 5) == (ti.map5 & 0x01)) | |
| 399 | goto do_clear; | |
| 400 | ||
| 401 | /* check if clear land under bridge */ | |
| 402 | if ((ti.map5 & 0xF8) == 0xE8) /* road under bridge */ | |
| 403 | return_cmd_error(STR_1007_ALREADY_BUILT); | |
| 404 | else if ((ti.map5 & 0xE0) == 0xE0) /* other transport route under bridge */ | |
| 405 | return_cmd_error(STR_1008_MUST_REMOVE_RAILROAD_TRACK); | |
| 406 | else if ((ti.map5 & 0xF8) == 0xC8) /* water under bridge */ | |
| 407 | return_cmd_error(STR_3807_CAN_T_BUILD_ON_WATER); | |
| 408 | ||
| 409 | /* all checked, can build road now! */ | |
| 410 | cost = _price.build_road * 2; | |
| 411 | 		if (flags & DC_EXEC) {
 | |
| 412 | ModifyTile(tile, | |
| 413 | MP_MAPOWNER_CURRENT | MP_MAP5, | |
| 414 | (ti.map5 & 0xC7) | 0x28 // map5 | |
| 415 | ); | |
| 416 | } | |
| 417 | return cost; | |
| 418 | 	} else {
 | |
| 419 | do_clear:; | |
| 420 | if (DoCommandByTile(tile, 0, 0, flags & ~DC_EXEC, CMD_LANDSCAPE_CLEAR) == CMD_ERROR) | |
| 421 | return CMD_ERROR; | |
| 422 | } | |
| 423 | ||
| 424 | cost = CheckRoadSlope(ti.tileh, &pieces, existing); | |
| 425 | if (cost == CMD_ERROR) return_cmd_error(STR_1800_LAND_SLOPED_IN_WRONG_DIRECTION); | |
| 426 | ||
| 427 | // the AI is not allowed to used foundationed tiles. | |
| 428 | if (cost && (_is_ai_player || !_patches.build_on_slopes)) | |
| 429 | return CMD_ERROR; | |
| 430 | ||
| 431 | 	if (!(ti.type == MP_STREET && (ti.map5 & 0xF0) == 0)) {
 | |
| 432 | cost += DoCommandByTile(tile, 0, 0, flags, CMD_LANDSCAPE_CLEAR); | |
| 433 | 	} else {
 | |
| 434 | // Don't put the pieces that already exist | |
| 435 | pieces &= ~ti.map5; | |
| 436 | } | |
| 437 | ||
| 438 | 	{
 | |
| 439 | byte t = pieces; | |
| 440 | 		while (t) {
 | |
| 441 | if (t & 1) cost += _price.build_road; | |
| 442 | t >>= 1; | |
| 443 | } | |
| 444 | } | |
| 445 | ||
| 446 | 	if (flags & DC_EXEC) {
 | |
| 447 | 		if (ti.type != MP_STREET) {
 | |
| 448 | _map_type_and_height[tile] &= 0xF; | |
| 449 | _map_type_and_height[tile] |= MP_STREET << 4; | |
| 450 | _map5[tile] = 0; | |
| 451 | _map_owner[tile] = _current_player; | |
| 452 | } | |
| 453 | ||
| 454 | _map5[tile] |= (byte)pieces; | |
| 455 | ||
| 456 | MarkTileDirtyByTile(tile); | |
| 457 | } | |
| 458 | return cost; | |
| 459 | } | |
| 460 | ||
| 461 | int32 DoConvertStreetRail(uint tile, uint totype, bool exec) | |
| 462 | {
 | |
| 463 | // not a railroad crossing? | |
| 464 | if ((_map5[tile] & 0xF0) != 0x10) return CMD_ERROR; | |
| 465 | ||
| 466 | // not owned by me? | |
| 467 | if (!CheckTileOwnership(tile) || !EnsureNoVehicle(tile)) return CMD_ERROR; | |
| 468 | ||
| 469 | // tile is already of requested type? | |
| 470 | if ( (uint)(_map3_hi[tile] & 0xF) == totype) return CMD_ERROR; | |
| 471 | ||
| 472 | 	if (exec) {
 | |
| 473 | // change type. | |
| 474 | _map3_hi[tile] = (_map3_hi[tile] & 0xF0) + totype; | |
| 475 | MarkTileDirtyByTile(tile); | |
| 476 | } | |
| 477 | ||
| 478 | return _price.build_rail >> 1; | |
| 479 | } | |
| 480 | ||
| 481 | ||
| 482 | // Build a long piece of road. | |
| 483 | // x,y = end tile | |
| 484 | // p1 = start tile | |
| 485 | // p2&1 = start tile starts in the 2nd half | |
| 486 | // p2&2 = end tile starts in the 2nd half | |
| 487 | // p2&4 = direction (0 = along x, 1=along y) | |
| 488 | int32 CmdBuildLongRoad(int x, int y, uint32 flags, uint32 p1, uint32 p2) | |
| 489 | {
 | |
| 490 | uint start_tile, end_tile, tile; | |
| 491 | int mode; | |
| 492 | int32 cost,ret; | |
| 493 | ||
| 494 | start_tile = p1; | |
| 495 | end_tile = TILE_FROM_XY(x, y); | |
| 496 | ||
| 497 | 	if (start_tile > end_tile || (start_tile == end_tile && (p2&1))) {
 | |
| 498 | uint t = start_tile; start_tile = end_tile; end_tile = t; | |
| 499 | p2 ^= IS_INT_INSIDE(p2&3, 1, 3) ? 3 : 0; | |
| 500 | } | |
| 501 | ||
| 502 | cost = 0; | |
| 503 | tile = start_tile; | |
| 504 | // Start tile is the small number. | |
| 505 | 	for(;;) {
 | |
| 506 | mode = (p2&4) ? 5 : 10; | |
| 507 | ||
| 508 | if (tile == start_tile && (p2&1)) | |
| 509 | mode &= (4+2); | |
| 510 | else if (tile == end_tile && !(p2&2)) | |
| 511 | mode &= (1+8); | |
| 512 | ||
| 513 | ret = DoCommandByTile(tile, mode, 0, flags, CMD_BUILD_ROAD); | |
| 514 | 		if (ret == CMD_ERROR) {
 | |
| 515 | if (_error_message != STR_1007_ALREADY_BUILT) | |
| 516 | return CMD_ERROR; | |
| 517 | 		} else {
 | |
| 518 | cost += ret; | |
| 519 | } | |
| 520 | ||
| 521 | if (tile == end_tile) | |
| 522 | break; | |
| 523 | ||
| 524 | tile += (p2&4)?TILE_XY(0,1):TILE_XY(1,0); | |
| 525 | } | |
| 526 | ||
| 527 | // already built? | |
| 528 | if (cost == 0) | |
| 529 | return CMD_ERROR; | |
| 530 | ||
| 531 | return cost; | |
| 532 | } | |
| 533 | ||
| 534 | // Remove a long piece of road. | |
| 535 | // x,y = end tile | |
| 536 | // p1 = start tile | |
| 537 | // p2&1 = start tile starts in the 2nd half | |
| 538 | // p2&2 = end tile starts in the 2nd half | |
| 539 | // p2&4 = direction (0 = along x, 1=along y) | |
| 540 | int32 CmdRemoveLongRoad(int x, int y, uint32 flags, uint32 p1, uint32 p2) | |
| 541 | {
 | |
| 542 | uint start_tile, end_tile, tile; | |
| 543 | int32 cost,ret; | |
| 544 | ||
| 545 | start_tile = p1; | |
| 546 | end_tile = TILE_FROM_XY(x, y); | |
| 547 | ||
| 548 | 	if (start_tile > end_tile || (start_tile == end_tile && (p2&1))) {
 | |
| 549 | uint t = start_tile; start_tile = end_tile; end_tile = t; | |
| 550 | p2 ^= IS_INT_INSIDE(p2&3, 1, 3) ? 3 : 0; | |
| 551 | } | |
| 552 | ||
| 553 | cost = 0; | |
| 554 | tile = start_tile; | |
| 555 | // Start tile is the small number. | |
| 556 | 	for(;;) {
 | |
| 557 | uint bits = (p2 & 4) ? ROAD_SE | ROAD_NW : ROAD_SW | ROAD_NE; | |
| 558 | if (tile == end_tile && !(p2&2)) bits &= ROAD_NW | ROAD_NE; | |
| 559 | if (tile == start_tile && (p2&1)) bits &= ROAD_SE | ROAD_SW; | |
| 560 | ||
| 561 | // try to remove the halves. | |
| 562 | 		if (bits) {
 | |
| 563 | ret = DoCommandByTile(tile, bits, 0, flags, CMD_REMOVE_ROAD); | |
| 564 | if (ret != CMD_ERROR) | |
| 565 | cost += ret; | |
| 566 | } | |
| 567 | ||
| 568 | if (tile == end_tile) | |
| 569 | break; | |
| 570 | ||
| 571 | tile += (p2&4)?TILE_XY(0,1):TILE_XY(1,0); | |
| 572 | } | |
| 573 | ||
| 574 | // already built? | |
| 575 | if (cost == 0) | |
| 576 | return CMD_ERROR; | |
| 577 | ||
| 578 | return cost; | |
| 579 | } | |
| 580 | ||
| 581 | /* Build a road depot | |
| 582 | * p1 - direction (0-3) | |
| 583 | * p2 - unused | |
| 584 | */ | |
| 585 | ||
| 586 | int32 CmdBuildRoadDepot(int x, int y, uint32 flags, uint32 p1, uint32 p2) | |
| 587 | {
 | |
| 588 | TileInfo ti; | |
| 589 | int32 cost; | |
| 590 | Depot *dep; | |
| 591 | uint tile; | |
| 592 | ||
| 593 | SET_EXPENSES_TYPE(EXPENSES_CONSTRUCTION); | |
| 594 | ||
| 595 | FindLandscapeHeight(&ti, x, y); | |
| 596 | ||
| 597 | tile = ti.tile; | |
| 598 | ||
| 599 | if (!EnsureNoVehicle(tile)) | |
| 600 | return CMD_ERROR; | |
| 601 | ||
| 602 | 	if (ti.tileh != 0) {
 | |
| 603 | if (_is_ai_player || !_patches.build_on_slopes || (ti.tileh & 0x10 || !((0x4C >> p1) & ti.tileh) )) | |
| 604 | return_cmd_error(STR_0007_FLAT_LAND_REQUIRED); | |
| 605 | } | |
| 606 | ||
| 607 | cost = DoCommandByTile(tile, 0, 0, flags, CMD_LANDSCAPE_CLEAR); | |
| 608 | if (cost == CMD_ERROR) | |
| 609 | return CMD_ERROR; | |
| 610 | ||
| 611 | dep = AllocateDepot(); | |
| 612 | if (dep == NULL) | |
| 613 | return CMD_ERROR; | |
| 614 | ||
| 615 | 	if (flags & DC_EXEC) {	
 | |
| 616 | if (_current_player == _local_player) | |
| 617 | _last_built_road_depot_tile = (TileIndex)tile; | |
| 618 | ||
| 619 | dep->xy = tile; | |
| 620 | dep->town_index = ClosestTownFromTile(tile, (uint)-1)->index; | |
| 621 | ||
| 622 | ModifyTile(tile, | |
| 623 | MP_SETTYPE(MP_STREET) | | |
| 624 | MP_MAPOWNER_CURRENT | MP_MAP5, | |
| 625 | (p1 | 0x20) /* map5 */ | |
| 626 | ); | |
| 627 | ||
| 628 | } | |
| 629 | return cost + _price.build_road_depot; | |
| 630 | } | |
| 631 | ||
| 632 | static int32 RemoveRoadDepot(uint tile, uint32 flags) | |
| 633 | {
 | |
| 634 | if (!CheckTileOwnership(tile)) | |
| 635 | return CMD_ERROR; | |
| 636 | ||
| 637 | if (!EnsureNoVehicle(tile)) | |
| 638 | return CMD_ERROR; | |
| 639 | ||
| 640 | 	if (flags & DC_EXEC) {
 | |
| 641 | DoDeleteDepot(tile); | |
| 642 | } | |
| 643 | ||
| 644 | return _price.remove_road_depot; | |
| 645 | } | |
| 646 | ||
| 647 | #define M(x) (1<<(x)) | |
| 648 | ||
| 649 | static int32 ClearTile_Road(uint tile, byte flags) {
 | |
| 650 | int32 ret; | |
| 651 | byte m5 = _map5[tile]; | |
| 652 | ||
| 653 | 	if ( (m5 & 0xF0) == 0) {
 | |
| 654 | byte b = m5 & 0xF; | |
| 655 | ||
| 656 | 		if (! ((1 << b) & (M(1)|M(2)|M(4)|M(8))) ) {
 | |
| 657 | if ( (!(flags & DC_AI_BUILDING) || _map_owner[tile]!=OWNER_TOWN) && flags&DC_AUTO) | |
| 658 | return_cmd_error(STR_1801_MUST_REMOVE_ROAD_FIRST); | |
| 659 | } | |
| 660 | return DoCommandByTile(tile, b, 0, flags, CMD_REMOVE_ROAD); | |
| 661 | 	} else if ( (m5 & 0xE0) == 0) {
 | |
| 662 | if (flags & DC_AUTO) | |
| 663 | return_cmd_error(STR_1801_MUST_REMOVE_ROAD_FIRST); | |
| 664 | ||
| 665 | ret = DoCommandByTile(tile, (m5&8)?5:10, 0, flags, CMD_REMOVE_ROAD); | |
| 666 | if (ret == CMD_ERROR) | |
| 667 | return CMD_ERROR; | |
| 668 | ||
| 669 | 		if (flags & DC_EXEC) {
 | |
| 670 | DoCommandByTile(tile, 0, 0, flags, CMD_LANDSCAPE_CLEAR); | |
| 671 | } | |
| 672 | ||
| 673 | return ret; | |
| 674 | 	} else {
 | |
| 675 | if (flags & DC_AUTO) | |
| 676 | return_cmd_error(STR_2004_BUILDING_MUST_BE_DEMOLISHED); | |
| 677 | return RemoveRoadDepot(tile,flags); | |
| 678 | } | |
| 679 | } | |
| 680 | ||
| 681 | ||
| 682 | typedef struct DrawRoadTileStruct {
 | |
| 683 | uint16 image; | |
| 684 | byte subcoord_x; | |
| 685 | byte subcoord_y; | |
| 686 | } DrawRoadTileStruct; | |
| 687 | ||
| 688 | typedef struct DrawRoadSeqStruct {
 | |
| 689 | uint32 image; | |
| 690 | byte subcoord_x; | |
| 691 | byte subcoord_y; | |
| 692 | byte width; | |
| 693 | byte height; | |
| 694 | } DrawRoadSeqStruct; | |
| 695 | ||
| 696 | #include "table/road_land.h" | |
| 697 | ||
| 698 | ||
| 699 | static uint GetRoadFoundation(uint tileh, uint bits) {
 | |
| 700 | int i; | |
| 701 | // normal level sloped building | |
| 702 | if ((~_valid_tileh_slopes_road[1][tileh] & bits) == 0) | |
| 703 | return tileh; | |
| 704 | ||
| 705 | // inclined sloped building | |
| 706 | if ( ((i=0, tileh == 1) || (i+=2, tileh == 2) || (i+=2, tileh == 4) || (i+=2, tileh == 8)) && | |
| 707 | ((bits == (ROAD_SW | ROAD_NE)) || (i++, bits == (ROAD_NW | ROAD_SE)))) | |
| 708 | return i + 15; | |
| 709 | ||
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 710 | // rail crossing | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 711 | if ((bits & 0x10) && _valid_tileh_slopes_road[2][tileh]) | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 712 | return tileh; | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 713 | |
| 0 | 714 | return 0; | 
| 715 | } | |
| 716 | ||
| 717 | const byte _road_sloped_sprites[14] = {
 | |
| 718 | 0, 0, 2, 0, | |
| 719 | 0, 1, 0, 0, | |
| 720 | 3, 0, 0, 0, | |
| 721 | 0, 0 | |
| 722 | }; | |
| 723 | ||
| 724 | static void DrawTile_Road(TileInfo *ti) | |
| 725 | {
 | |
| 726 | uint32 image; | |
| 727 | byte m2; | |
| 728 | const byte *s; | |
| 729 | ||
| 730 | 	if ( (ti->map5 & 0xF0) == 0) { // if it is a road the upper 4 bits are 0
 | |
| 731 | const DrawRoadTileStruct *drts; | |
| 732 | ||
| 733 | 		if (ti->tileh != 0) {
 | |
| 734 | int f = GetRoadFoundation(ti->tileh, ti->map5 & 0xF); | |
| 735 | if (f) DrawFoundation(ti, f); | |
| 736 | ||
| 737 | // default sloped sprites.. | |
| 738 | 			if (ti->tileh != 0) {
 | |
| 739 | image = _road_sloped_sprites[ti->tileh - 1] + 0x53F; | |
| 740 | 			} else  {
 | |
| 741 | image = _road_tile_sprites_1[ti->map5 & 0xF]; | |
| 742 | } | |
| 743 | 		} else {
 | |
| 744 | image = _road_tile_sprites_1[ti->map5 & 0xF]; | |
| 745 | } | |
| 746 | ||
| 747 | m2 = _map2[ti->tile] & 7; | |
| 748 | ||
| 749 | if (m2 == 0) image |= 0x3178000; | |
| 750 | ||
| 751 | 		if (_map3_hi[ti->tile] & 0x80) {
 | |
| 752 | image += 19; | |
| 753 | 		} else if (m2 > 1 && m2 != 6) {
 | |
| 754 | image -= 19; /* pavement along the road? */ | |
| 755 | } | |
| 756 | ||
| 757 | DrawGroundSprite(image); | |
| 758 | ||
| 759 | if (!(_display_opt & DO_FULL_DETAIL) || _cur_dpi->zoom == 2) | |
| 760 | return; | |
| 761 | ||
| 762 | 		if (m2 >= 6) {
 | |
| 763 | // roadwork | |
| 764 | DrawGroundSprite(0x586 + ((ti->map5&8)!=0 ? 0 : 1)); | |
| 765 | return; | |
| 766 | } | |
| 767 | ||
| 768 | drts = (const DrawRoadTileStruct*)_road_display_table[m2][ti->map5 & 0xF]; | |
| 769 | ||
| 770 | 		while ((image = drts->image) != 0) {
 | |
| 771 | int x = ti->x | drts->subcoord_x; | |
| 772 | int y = ti->y | drts->subcoord_y; | |
| 773 | byte z = ti->z; | |
| 774 | if (ti->tileh != 0) z = GetSlopeZ(x, y); | |
| 775 | AddSortableSpriteToDraw(image, x, y, 2, 2, 0x10, z); | |
| 776 | drts++; | |
| 777 | } | |
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 778 | 	} else if ( (ti->map5 & 0xE0) == 0) { // railroad crossing
 | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 779 | int f = GetRoadFoundation(ti->tileh, ti->map5 & 0xF); | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 780 | if (f) DrawFoundation(ti, f); | 
| 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 781 | |
| 0 | 782 | image = 0x55B; | 
| 783 | ||
| 784 | if ( (ti->map5 & 8) != 0) | |
| 785 | image--; | |
| 786 | ||
| 787 | if ( (ti->map5 & 4) != 0) | |
| 788 | image += 2; | |
| 789 | ||
| 790 | 		if ( _map3_hi[ti->tile] & 0x80) {
 | |
| 791 | image += 8; | |
| 792 | 		} else {
 | |
| 793 | m2 = _map2[ti->tile] & 7; | |
| 794 | if (m2 == 0) image |= 0x3178000; | |
| 795 | if (m2 > 1) image += 4; | |
| 796 | } | |
| 797 | ||
| 798 | DrawGroundSprite(image + (_map3_hi[ti->tile] & 0xF) * 12); | |
| 799 | 	} else {
 | |
| 800 | uint32 ormod; | |
| 801 | int player; | |
| 802 | const DrawRoadSeqStruct *drss; | |
| 803 | ||
| 804 | 		if (ti->tileh != 0) { DrawFoundation(ti, ti->tileh); }
 | |
| 805 | ||
| 806 | ormod = 0x315; | |
| 807 | player = _map_owner[ti->tile]; | |
| 808 | if (player < MAX_PLAYERS) | |
| 809 | ormod = PLAYER_SPRITE_COLOR(player); | |
| 810 | ||
| 811 | s = _road_display_datas[ti->map5 & 0xF]; | |
| 812 | ||
| 813 | DrawGroundSprite(*(uint32*)s); | |
| 814 | s += sizeof(uint32); | |
| 815 | drss = (DrawRoadSeqStruct*)s; | |
| 816 | ||
| 817 | 		while ((image=drss->image) != 0) {
 | |
| 818 | if (image & 0x8000) | |
| 819 | image |= ormod; | |
| 820 | ||
| 821 | AddSortableSpriteToDraw(image, ti->x | drss->subcoord_x, | |
| 822 | ti->y | drss->subcoord_y, drss->width, drss->height, 0x14, ti->z); | |
| 823 | drss++; | |
| 824 | } | |
| 825 | } | |
| 826 | } | |
| 827 | ||
| 828 | void DrawRoadDepotSprite(int x, int y, int image) | |
| 829 | {
 | |
| 830 | uint32 ormod; | |
| 831 | const DrawRoadSeqStruct *dtss; | |
| 832 | const byte *t; | |
| 833 | ||
| 834 | ormod = PLAYER_SPRITE_COLOR(_local_player); | |
| 835 | ||
| 836 | t = _road_display_datas[image]; | |
| 837 | ||
| 838 | x+=33; | |
| 839 | y+=17; | |
| 840 | ||
| 841 | DrawSprite(*(uint32*)t, x, y); | |
| 842 | t += sizeof(uint32); | |
| 843 | ||
| 844 | 	for(dtss = (DrawRoadSeqStruct *)t; dtss->image != 0; dtss++) {
 | |
| 845 | Point pt = RemapCoords(dtss->subcoord_x, dtss->subcoord_y, 0); | |
| 846 | ||
| 847 | image = dtss->image; | |
| 848 | if (image & 0x8000) | |
| 849 | image |= ormod; | |
| 850 | ||
| 851 | DrawSprite(image, x + pt.x, y + pt.y); | |
| 852 | } | |
| 853 | } | |
| 854 | ||
| 855 | uint GetSlopeZ_Road(TileInfo *ti) | |
| 856 | {
 | |
| 857 | uint z = ti->z; | |
| 858 | int th = ti->tileh; | |
| 859 | ||
| 860 | // check if it's a foundation | |
| 861 | 	if (ti->tileh != 0) {
 | |
| 13 
3e7c1f74a996
(svn r14) Fix: railroad crossings on slopes are now possible
 dominik parents: 
0diff
changeset | 862 | 		if ((ti->map5 & 0xE0) == 0) { /* road or crossing */
 | 
| 0 | 863 | uint f = GetRoadFoundation(ti->tileh, ti->map5 & 0x3F); | 
| 864 | 			if (f != 0) {
 | |
| 865 | 				if (f < 15) {
 | |
| 866 | // leveled foundation | |
| 867 | return z + 8; | |
| 868 | } | |
| 869 | // inclined foundation | |
| 870 | th = _inclined_tileh[f - 15]; | |
| 871 | } | |
| 872 | 		} else if ((ti->map5 & 0xF0) == 0x20) {
 | |
| 873 | // depot | |
| 874 | return z + 8; | |
| 875 | } | |
| 876 | return GetPartialZ(ti->x&0xF, ti->y&0xF, th) + z; | |
| 877 | } | |
| 39 | 878 | return z; // normal Z if no slope | 
| 879 | } | |
| 880 | ||
| 881 | uint GetSlopeTileh_Road(TileInfo *ti) | |
| 882 | {
 | |
| 883 | // check if it's a foundation | |
| 884 | 	if (ti->tileh != 0) {
 | |
| 885 | 		if ((ti->map5 & 0xE0) == 0) { /* road or crossing */
 | |
| 886 | uint f = GetRoadFoundation(ti->tileh, ti->map5 & 0x3F); | |
| 887 | 			if (f != 0) {
 | |
| 888 | 				if (f < 15) {
 | |
| 889 | // leveled foundation | |
| 890 | return 0; | |
| 891 | } | |
| 892 | // inclined foundation | |
| 893 | return _inclined_tileh[f - 15]; | |
| 894 | } | |
| 895 | 		} else if ((ti->map5 & 0xF0) == 0x20) {
 | |
| 896 | // depot | |
| 897 | return 0; | |
| 898 | } | |
| 899 | } | |
| 900 | return ti->tileh; | |
| 0 | 901 | } | 
| 902 | ||
| 903 | static void GetAcceptedCargo_Road(uint tile, AcceptedCargo *ac) | |
| 904 | {
 | |
| 905 | /* not used */ | |
| 906 | } | |
| 907 | ||
| 908 | static void AnimateTile_Road(uint tile) | |
| 909 | {
 | |
| 910 | 	if ((_map5[tile] & 0xF0) == 0x10) {
 | |
| 911 | MarkTileDirtyByTile(tile); | |
| 912 | } | |
| 913 | } | |
| 914 | ||
| 915 | static const byte _town_road_types[5][2] = {
 | |
| 916 | 	{1,1},
 | |
| 917 | 	{2,2},
 | |
| 918 | 	{2,2},
 | |
| 919 | 	{5,5},
 | |
| 920 | 	{3,2},
 | |
| 921 | }; | |
| 922 | ||
| 923 | static const byte _town_road_types_2[5][2] = {
 | |
| 924 | 	{1,1},
 | |
| 925 | 	{2,2},
 | |
| 926 | 	{3,2},
 | |
| 927 | 	{3,2},
 | |
| 928 | 	{3,2},
 | |
| 929 | }; | |
| 930 | ||
| 931 | ||
| 932 | static void TileLoop_Road(uint tile) | |
| 933 | {
 | |
| 934 | Town *t; | |
| 935 | int grp; | |
| 936 | ||
| 937 | 	if (_opt.landscape == LT_HILLY) {
 | |
| 938 | // Fix snow style if the road is above the snowline | |
| 939 | 		if ((_map3_hi[tile] & 0x80) != ((GetTileZ(tile) > _opt.snow_line) ? 0x80 : 0x00)) {
 | |
| 940 | _map3_hi[tile] ^= 0x80; | |
| 941 | MarkTileDirtyByTile(tile); | |
| 942 | } | |
| 943 | 	} else if (_opt.landscape == LT_DESERT) {
 | |
| 944 | // Fix desert style | |
| 945 | 		if (GetMapExtraBits(tile) == 1 && !(_map3_hi[tile] & 0x80)) {
 | |
| 946 | _map3_hi[tile] |= 0x80; | |
| 947 | MarkTileDirtyByTile(tile); | |
| 948 | } | |
| 949 | } | |
| 950 | ||
| 951 | if (_map5[tile] & 0xE0) | |
| 952 | return; | |
| 953 | ||
| 954 | 	if ((_map2[tile] & 7) < 6) {
 | |
| 955 | t = ClosestTownFromTile(tile, (uint)-1); | |
| 956 | grp = 0; | |
| 957 | 		if (t != NULL) {
 | |
| 958 | // If in the scenario editor, set the owner to a town. | |
| 959 | 			if (_game_mode == GM_EDITOR) {
 | |
| 960 | _map_owner[tile] = OWNER_TOWN; | |
| 961 | } | |
| 962 | ||
| 963 | grp = GetTownRadiusGroup(t, tile); | |
| 964 | ||
| 965 | // Show an animation to indicate road work | |
| 966 | if (t->road_build_months != 0 && | |
| 967 | !(GetTileDist(t->xy, tile) >= 8 && grp==0) && | |
| 968 | 					(_map5[tile]==5 || _map5[tile]==10)) {
 | |
| 969 | 				if (GetTileSlope(tile, NULL) == 0 && EnsureNoVehicle(tile) && CHANCE16(1,20)) {
 | |
| 970 | _map2[tile] = ((_map2[tile]&7) <= 1) ? 6 : 7; | |
| 971 | ||
| 972 | SndPlayTileFx(0x1F,tile); | |
| 973 | CreateEffectVehicleAbove( | |
| 974 | GET_TILE_X(tile) * 16 + 7, | |
| 975 | GET_TILE_Y(tile) * 16 + 7, | |
| 976 | 0, | |
| 977 | EV_ROADWORK); | |
| 978 | MarkTileDirtyByTile(tile); | |
| 979 | return; | |
| 980 | } | |
| 981 | } | |
| 982 | } | |
| 983 | ||
| 984 | 		{
 | |
| 985 | const byte *p = (_opt.landscape == LT_CANDY) ? _town_road_types_2[grp] : _town_road_types[grp]; | |
| 986 | byte b = _map2[tile] & 7; | |
| 987 | ||
| 988 | if (b == p[0]) | |
| 989 | return; | |
| 990 | ||
| 991 | 			if (b == p[1]) {
 | |
| 992 | b = p[0]; | |
| 993 | 			} else if (b == 0) {
 | |
| 994 | b = p[1]; | |
| 995 | 			} else {
 | |
| 996 | b = 0; | |
| 997 | } | |
| 998 | _map2[tile] = (_map2[tile] & ~7) | b; | |
| 999 | MarkTileDirtyByTile(tile); | |
| 1000 | } | |
| 1001 | 	} else {
 | |
| 1002 | // Handle road work | |
| 1003 | ||
| 1004 | byte b = _map2[tile]; | |
| 1005 | 		if (b < 0x80) {
 | |
| 1006 | _map2[tile] = b + 8; | |
| 1007 | return; | |
| 1008 | } | |
| 1009 | _map2[tile] = ((b&7) == 6) ? 1 : 2; | |
| 1010 | MarkTileDirtyByTile(tile); | |
| 1011 | } | |
| 1012 | } | |
| 1013 | ||
| 1014 | void ShowRoadDepotWindow(uint tile); | |
| 1015 | ||
| 1016 | static void ClickTile_Road(uint tile) | |
| 1017 | {
 | |
| 1018 | 	if ((_map5[tile] & 0xF0) == 0x20) {
 | |
| 1019 | ShowRoadDepotWindow(tile); | |
| 1020 | } | |
| 1021 | } | |
| 1022 | ||
| 1023 | static const byte _road_trackbits[16] = {
 | |
| 1024 | 0x0, 0x0, 0x0, 0x10, 0x0, 0x2, 0x8, 0x1A, 0x0, 0x4, 0x1, 0x15, 0x20, 0x26, 0x29, 0x3F, | |
| 1025 | }; | |
| 1026 | ||
| 1027 | static uint32 GetTileTrackStatus_Road(uint tile, int mode)	{
 | |
| 1028 | 	if (mode == 0) {
 | |
| 1029 | if ((_map5[tile] & 0xF0) != 0x10) | |
| 1030 | return 0; | |
| 1031 | return _map5[tile] & 8 ? 0x101 : 0x202; | |
| 1032 | 	} else if  (mode == 2) {
 | |
| 1033 | byte b = _map5[tile]; | |
| 1034 | 		if ((b & 0xF0) == 0) {
 | |
| 1035 | if (!_road_special_gettrackstatus && (_map2[tile]&7) >= 6) | |
| 1036 | return 0; | |
| 1037 | return _road_trackbits[b&0xF] * 0x101; | |
| 1038 | 		} else if ((b&0xE0) == 0) {
 | |
| 1039 | uint32 r = 0x101; | |
| 1040 | if (b&8) r <<= 1; | |
| 1041 | ||
| 1042 | 			if (b&4) {
 | |
| 1043 | r *= 0x10001; | |
| 1044 | } | |
| 1045 | return r; | |
| 1046 | } | |
| 1047 | } | |
| 1048 | return 0; | |
| 1049 | } | |
| 1050 | ||
| 1051 | static const StringID _road_tile_strings[] = {
 | |
| 1052 | STR_1818_ROAD_RAIL_LEVEL_CROSSING, | |
| 1053 | STR_1817_ROAD_VEHICLE_DEPOT, | |
| 1054 | ||
| 1055 | STR_1814_ROAD, | |
| 1056 | STR_1814_ROAD, | |
| 1057 | STR_1814_ROAD, | |
| 1058 | STR_1815_ROAD_WITH_STREETLIGHTS, | |
| 1059 | STR_1814_ROAD, | |
| 1060 | STR_1816_TREE_LINED_ROAD, | |
| 1061 | STR_1814_ROAD, | |
| 1062 | STR_1814_ROAD, | |
| 1063 | }; | |
| 1064 | ||
| 1065 | static void GetTileDesc_Road(uint tile, TileDesc *td) | |
| 1066 | {
 | |
| 1067 | int i = (_map5[tile] >> 4); | |
| 1068 | if (i == 0) | |
| 1069 | i = (_map2[tile] & 7) + 3; | |
| 1070 | td->str = _road_tile_strings[i - 1]; | |
| 1071 | td->owner = _map_owner[tile]; | |
| 1072 | } | |
| 1073 | ||
| 1074 | static const byte _roadveh_enter_depot_unk0[4] = {
 | |
| 1075 | 8, 9, 0, 1 | |
| 1076 | }; | |
| 1077 | ||
| 1078 | static uint32 VehicleEnter_Road(Vehicle *v, uint tile, int x, int y) | |
| 1079 | {
 | |
| 1080 | 	if ((_map5[tile] & 0xF0) == 0x10)	{
 | |
| 1081 | 		if (v->type == VEH_Train && (_map5[tile] & 4) == 0) {
 | |
| 1082 | /* train crossing a road */ | |
| 1083 | SndPlayVehicleFx(12, v); | |
| 1084 | _map5[tile] |= 4; | |
| 1085 | MarkTileDirtyByTile(tile); | |
| 1086 | } | |
| 1087 | 	} else if ((_map5[tile]&0xF0) ==  0x20){
 | |
| 1088 | 		if (v->type == VEH_Road && v->u.road.frame == 11) {
 | |
| 1089 | 			if (_roadveh_enter_depot_unk0[_map5[tile]&3] == v->u.road.state) {
 | |
| 1090 | RoadVehEnterDepot(v); | |
| 1091 | return 4; | |
| 1092 | } | |
| 1093 | } | |
| 1094 | } | |
| 1095 | return 0; | |
| 1096 | } | |
| 1097 | ||
| 1098 | static void VehicleLeave_Road(Vehicle *v, uint tile, int x, int y) | |
| 1099 | {
 | |
| 1100 | 	if ((_map5[tile] & 0xF0) == 0x10 && v->type == VEH_Train && v->next == NULL) {
 | |
| 1101 | _map5[tile] &= ~4; | |
| 1102 | MarkTileDirtyByTile(tile); | |
| 1103 | } | |
| 1104 | } | |
| 1105 | ||
| 1106 | static void ChangeTileOwner_Road(uint tile, byte old_player, byte new_player) | |
| 1107 | {
 | |
| 1108 | byte b; | |
| 1109 | ||
| 1110 | // road/rail crossing where the road is owned by the current player? | |
| 1111 | 	if (old_player == _map3_lo[tile] && (_map5[tile]&0xF0) == 0x10) {
 | |
| 1112 | _map3_lo[tile] = (new_player == 0xFF) ? OWNER_NONE : new_player; | |
| 1113 | } | |
| 1114 | ||
| 1115 | if (_map_owner[tile] != old_player) | |
| 1116 | return; | |
| 1117 | ||
| 1118 | 	if (new_player != 255) {
 | |
| 1119 | _map_owner[tile] = new_player; | |
| 1120 | 	}	else {
 | |
| 1121 | b = _map5[tile]&0xF0; | |
| 1122 | 		if (b == 0) {
 | |
| 1123 | _map_owner[tile] = OWNER_NONE; | |
| 1124 | 		} else if (b == 0x10) {
 | |
| 1125 | _map5[tile] = (_map5[tile]&8) ? 0x5 : 0xA; | |
| 1126 | _map_owner[tile] = _map3_lo[tile]; | |
| 1127 | _map3_lo[tile] = 0; | |
| 1128 | _map3_hi[tile] &= 0x80; | |
| 1129 | 		} else {
 | |
| 1130 | DoCommandByTile(tile, 0, 0, DC_EXEC, CMD_LANDSCAPE_CLEAR); | |
| 1131 | } | |
| 1132 | } | |
| 1133 | } | |
| 1134 | ||
| 1135 | void InitializeRoad() | |
| 1136 | {
 | |
| 1137 | _last_built_road_depot_tile = 0; | |
| 1138 | } | |
| 1139 | ||
| 1140 | const TileTypeProcs _tile_type_road_procs = {
 | |
| 1141 | DrawTile_Road, /* draw_tile_proc */ | |
| 1142 | GetSlopeZ_Road, /* get_slope_z_proc */ | |
| 1143 | ClearTile_Road, /* clear_tile_proc */ | |
| 1144 | GetAcceptedCargo_Road, /* get_accepted_cargo_proc */ | |
| 1145 | GetTileDesc_Road, /* get_tile_desc_proc */ | |
| 1146 | GetTileTrackStatus_Road, /* get_tile_track_status_proc */ | |
| 1147 | ClickTile_Road, /* click_tile_proc */ | |
| 1148 | AnimateTile_Road, /* animate_tile_proc */ | |
| 1149 | TileLoop_Road, /* tile_loop_clear */ | |
| 1150 | ChangeTileOwner_Road, /* change_tile_owner_clear */ | |
| 1151 | NULL, /* get_produced_cargo_proc */ | |
| 1152 | VehicleEnter_Road, /* vehicle_enter_tile_proc */ | |
| 1153 | VehicleLeave_Road, /* vehicle_leave_tile_proc */ | |
| 39 | 1154 | GetSlopeTileh_Road, /* get_slope_tileh_proc */ | 
| 0 | 1155 | }; |