2008-05-28 11:56:21 +00:00
|
|
|
/* $Id$ */
|
|
|
|
|
2009-08-21 20:21:05 +00:00
|
|
|
/*
|
|
|
|
* This file is part of OpenTTD.
|
|
|
|
* OpenTTD 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, version 2.
|
|
|
|
* OpenTTD 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 OpenTTD. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2008-10-14 19:27:08 +00:00
|
|
|
/** @file vehiclelist.cpp Lists of vehicles. */
|
2008-05-28 11:56:21 +00:00
|
|
|
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include "train.h"
|
|
|
|
#include "vehiclelist.h"
|
|
|
|
|
2010-09-08 21:37:13 +00:00
|
|
|
/**
|
|
|
|
* Pack a VehicleListIdentifier in a single uint32.
|
|
|
|
* @return The packed identifier.
|
|
|
|
*/
|
|
|
|
uint32 VehicleListIdentifier::Pack()
|
|
|
|
{
|
2012-05-16 22:08:46 +00:00
|
|
|
byte c = this->company == OWNER_NONE ? 0xF : (byte)this->company;
|
|
|
|
assert(c < (1 << 4));
|
2010-09-08 21:37:13 +00:00
|
|
|
assert(this->type < (1 << 3));
|
|
|
|
assert(this->vtype < (1 << 2));
|
|
|
|
assert(this->index < (1 << 20));
|
|
|
|
|
2012-05-16 22:08:46 +00:00
|
|
|
return c << 28 | this->type << 23 | this->vtype << 26 | this->index;
|
2010-09-08 21:37:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unpack a VehicleListIdentifier from a single uint32.
|
|
|
|
* @param data The data to unpack.
|
|
|
|
* @return true iff the data was valid (enough).
|
|
|
|
*/
|
|
|
|
bool VehicleListIdentifier::Unpack(uint32 data)
|
|
|
|
{
|
2012-05-16 22:08:46 +00:00
|
|
|
byte c = GB(data, 28, 4);
|
|
|
|
this->company = c == 0xF ? OWNER_NONE : (CompanyID)c;
|
2010-09-08 21:37:13 +00:00
|
|
|
this->type = (VehicleListType)GB(data, 23, 3);
|
|
|
|
this->vtype = (VehicleType)GB(data, 26, 2);
|
|
|
|
this->index = GB(data, 0, 20);
|
|
|
|
|
|
|
|
return this->type < VLT_END;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decode a packed vehicle list identifier into a new one.
|
|
|
|
* @param data The data to unpack.
|
|
|
|
*/
|
|
|
|
VehicleListIdentifier::VehicleListIdentifier(uint32 data)
|
|
|
|
{
|
|
|
|
bool ret = this->Unpack(data);
|
|
|
|
assert(ret);
|
|
|
|
}
|
|
|
|
|
2008-05-28 11:56:21 +00:00
|
|
|
/**
|
|
|
|
* Generate a list of vehicles inside a depot.
|
|
|
|
* @param type Type of vehicle
|
|
|
|
* @param tile The tile the depot is located on
|
|
|
|
* @param engines Pointer to list to add vehicles to
|
|
|
|
* @param wagons Pointer to list to add wagons to (can be NULL)
|
2008-09-27 14:58:46 +00:00
|
|
|
* @param individual_wagons If true add every wagon to #wagons which is not attached to an engine. If false only add the first wagon of every row.
|
2008-05-28 11:56:21 +00:00
|
|
|
*/
|
2008-09-27 14:58:46 +00:00
|
|
|
void BuildDepotVehicleList(VehicleType type, TileIndex tile, VehicleList *engines, VehicleList *wagons, bool individual_wagons)
|
2008-05-28 11:56:21 +00:00
|
|
|
{
|
|
|
|
engines->Clear();
|
|
|
|
if (wagons != NULL && wagons != engines) wagons->Clear();
|
|
|
|
|
|
|
|
const Vehicle *v;
|
|
|
|
FOR_ALL_VEHICLES(v) {
|
|
|
|
/* General tests for all vehicle types */
|
|
|
|
if (v->type != type) continue;
|
|
|
|
if (v->tile != tile) continue;
|
|
|
|
|
|
|
|
switch (type) {
|
2009-07-01 23:49:13 +00:00
|
|
|
case VEH_TRAIN: {
|
|
|
|
const Train *t = Train::From(v);
|
2009-07-01 23:57:20 +00:00
|
|
|
if (t->IsArticulatedPart() || t->IsRearDualheaded()) continue;
|
2009-07-01 23:49:13 +00:00
|
|
|
if (t->track != TRACK_BIT_DEPOT) continue;
|
|
|
|
if (wagons != NULL && t->First()->IsFreeWagon()) {
|
|
|
|
if (individual_wagons || t->IsFreeWagon()) *wagons->Append() = t;
|
2008-05-28 11:56:21 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
2009-07-01 23:49:13 +00:00
|
|
|
}
|
2008-05-28 11:56:21 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
if (!v->IsInDepot()) continue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!v->IsPrimaryVehicle()) continue;
|
|
|
|
|
|
|
|
*engines->Append() = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ensure the lists are not wasting too much space. If the lists are fresh
|
|
|
|
* (i.e. built within a command) then this will actually do nothing. */
|
|
|
|
engines->Compact();
|
|
|
|
if (wagons != NULL && wagons != engines) wagons->Compact();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate a list of vehicles based on window type.
|
2010-09-08 21:28:50 +00:00
|
|
|
* @param list Pointer to list to add vehicles to
|
|
|
|
* @param vli The identifier of this vehicle list.
|
2010-04-17 14:47:56 +00:00
|
|
|
* @return false if invalid list is requested
|
2008-05-28 11:56:21 +00:00
|
|
|
*/
|
2010-09-08 21:28:50 +00:00
|
|
|
bool GenerateVehicleSortList(VehicleList *list, const VehicleListIdentifier &vli)
|
2008-05-28 11:56:21 +00:00
|
|
|
{
|
|
|
|
list->Clear();
|
|
|
|
|
|
|
|
const Vehicle *v;
|
|
|
|
|
2010-09-08 21:28:50 +00:00
|
|
|
switch (vli.type) {
|
|
|
|
case VL_STATION_LIST:
|
2008-05-28 11:56:21 +00:00
|
|
|
FOR_ALL_VEHICLES(v) {
|
2010-09-08 21:37:13 +00:00
|
|
|
if (v->type == vli.vtype && v->IsPrimaryVehicle()) {
|
2008-05-28 11:56:21 +00:00
|
|
|
const Order *order;
|
|
|
|
|
|
|
|
FOR_VEHICLE_ORDERS(v, order) {
|
2011-05-18 12:19:58 +00:00
|
|
|
if ((order->IsType(OT_GOTO_STATION) || order->IsType(OT_GOTO_WAYPOINT) || order->IsType(OT_IMPLICIT))
|
2010-09-08 21:28:50 +00:00
|
|
|
&& order->GetDestination() == vli.index) {
|
2008-05-28 11:56:21 +00:00
|
|
|
*list->Append() = v;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-09-08 21:28:50 +00:00
|
|
|
case VL_SHARED_ORDERS:
|
2008-08-17 21:07:09 +00:00
|
|
|
/* Add all vehicles from this vehicle's shared order list */
|
2010-09-08 21:28:50 +00:00
|
|
|
v = Vehicle::GetIfValid(vli.index);
|
2010-09-08 21:37:13 +00:00
|
|
|
if (v == NULL || v->type != vli.vtype || !v->IsPrimaryVehicle()) return false;
|
2010-04-17 14:47:56 +00:00
|
|
|
|
|
|
|
for (; v != NULL; v = v->NextShared()) {
|
2008-08-17 21:07:09 +00:00
|
|
|
*list->Append() = v;
|
2008-05-28 11:56:21 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-09-09 14:39:48 +00:00
|
|
|
case VL_GROUP_LIST:
|
|
|
|
if (vli.index != ALL_GROUP) {
|
|
|
|
FOR_ALL_VEHICLES(v) {
|
|
|
|
if (v->type == vli.vtype && v->IsPrimaryVehicle() &&
|
|
|
|
v->owner == vli.company && v->group_id == vli.index) {
|
|
|
|
*list->Append() = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* FALL THROUGH */
|
|
|
|
|
2010-09-08 21:28:50 +00:00
|
|
|
case VL_STANDARD:
|
2008-05-28 11:56:21 +00:00
|
|
|
FOR_ALL_VEHICLES(v) {
|
2010-09-08 21:37:13 +00:00
|
|
|
if (v->type == vli.vtype && v->owner == vli.company && v->IsPrimaryVehicle()) {
|
2008-05-28 11:56:21 +00:00
|
|
|
*list->Append() = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-09-08 21:28:50 +00:00
|
|
|
case VL_DEPOT_LIST:
|
2008-05-28 11:56:21 +00:00
|
|
|
FOR_ALL_VEHICLES(v) {
|
2010-09-08 21:37:13 +00:00
|
|
|
if (v->type == vli.vtype && v->IsPrimaryVehicle()) {
|
2008-05-28 11:56:21 +00:00
|
|
|
const Order *order;
|
|
|
|
|
|
|
|
FOR_VEHICLE_ORDERS(v, order) {
|
2010-09-08 21:28:50 +00:00
|
|
|
if (order->IsType(OT_GOTO_DEPOT) && !(order->GetDepotActionType() & ODATFB_NEAREST_DEPOT) && order->GetDestination() == vli.index) {
|
2008-05-28 11:56:21 +00:00
|
|
|
*list->Append() = v;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-04-17 14:47:56 +00:00
|
|
|
default: return false;
|
2008-05-28 11:56:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
list->Compact();
|
2010-04-17 14:47:56 +00:00
|
|
|
return true;
|
2008-05-28 11:56:21 +00:00
|
|
|
}
|