2010-08-18 15:15:59 +00:00
|
|
|
/* $Id$ */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** @file order_backup.cpp Handling of order backups. */
|
|
|
|
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include "command_func.h"
|
|
|
|
#include "core/pool_func.hpp"
|
2010-08-18 20:48:38 +00:00
|
|
|
#include "network/network.h"
|
|
|
|
#include "network/network_func.h"
|
2010-08-18 15:15:59 +00:00
|
|
|
#include "order_backup.h"
|
|
|
|
#include "vehicle_base.h"
|
|
|
|
|
|
|
|
OrderBackupPool _order_backup_pool("BackupOrder");
|
|
|
|
INSTANTIATE_POOL_METHODS(OrderBackup)
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/** Free everything that is allocated. */
|
2010-08-18 15:15:59 +00:00
|
|
|
OrderBackup::~OrderBackup()
|
|
|
|
{
|
|
|
|
free(this->name);
|
2010-08-18 20:48:38 +00:00
|
|
|
|
|
|
|
if (CleaningPool()) return;
|
|
|
|
|
|
|
|
Order *o = this->orders;
|
|
|
|
while (o != NULL) {
|
|
|
|
Order *next = o->next;
|
|
|
|
delete o;
|
|
|
|
o = next;
|
|
|
|
}
|
2010-08-18 15:15:59 +00:00
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Create an order backup for the given vehicle.
|
|
|
|
* @param v The vehicle to make a backup of.
|
|
|
|
* @param user The user that is requesting the backup.
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
OrderBackup::OrderBackup(const Vehicle *v, uint32 user)
|
2010-08-18 15:15:59 +00:00
|
|
|
{
|
2010-08-18 20:48:38 +00:00
|
|
|
this->user = user;
|
2010-08-18 15:15:59 +00:00
|
|
|
this->tile = v->tile;
|
|
|
|
this->orderindex = v->cur_order_index;
|
|
|
|
this->group = v->group_id;
|
|
|
|
this->service_interval = v->service_interval;
|
|
|
|
|
|
|
|
if (v->name != NULL) this->name = strdup(v->name);
|
|
|
|
|
|
|
|
/* If we have shared orders, store the vehicle we share the order with. */
|
|
|
|
if (v->IsOrderListShared()) {
|
2010-08-18 15:58:30 +00:00
|
|
|
this->clone = (v->FirstShared() == v) ? v->NextShared() : v->FirstShared();
|
2010-08-18 15:15:59 +00:00
|
|
|
} else {
|
|
|
|
/* Else copy the orders */
|
2010-08-18 20:48:38 +00:00
|
|
|
Order **tail = &this->orders;
|
2010-08-18 15:15:59 +00:00
|
|
|
|
|
|
|
/* Count the number of orders */
|
|
|
|
const Order *order;
|
|
|
|
FOR_VEHICLE_ORDERS(v, order) {
|
2010-08-18 20:48:38 +00:00
|
|
|
Order *copy = new Order();
|
|
|
|
copy->AssignOrder(*order);
|
|
|
|
*tail = copy;
|
|
|
|
tail = ©->next;
|
2010-08-18 15:15:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Restore the data of this order to the given vehicle.
|
|
|
|
* @param v The vehicle to restore to.
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
void OrderBackup::DoRestore(Vehicle *v)
|
2010-08-18 15:15:59 +00:00
|
|
|
{
|
|
|
|
/* If we have a custom name, process that */
|
2010-08-18 20:48:38 +00:00
|
|
|
v->name = this->name;
|
|
|
|
this->name = NULL;
|
2010-08-18 15:15:59 +00:00
|
|
|
|
|
|
|
/* If we had shared orders, recover that */
|
2010-08-18 15:58:30 +00:00
|
|
|
if (this->clone != NULL) {
|
2010-08-19 20:39:12 +00:00
|
|
|
DoCommand(0, v->index | CO_SHARE << 30, this->clone->index, DC_EXEC, CMD_CLONE_ORDER);
|
2010-08-18 20:48:38 +00:00
|
|
|
} else if (this->orders != NULL && OrderList::CanAllocateItem()) {
|
|
|
|
v->orders.list = new OrderList(this->orders, v);
|
|
|
|
this->orders = NULL;
|
2010-08-18 15:15:59 +00:00
|
|
|
}
|
|
|
|
|
2010-08-18 20:48:38 +00:00
|
|
|
uint num_orders = v->GetNumOrders();
|
|
|
|
if (num_orders != 0) v->cur_order_index = this->orderindex % num_orders;
|
|
|
|
v->service_interval = this->service_interval;
|
2010-08-18 15:15:59 +00:00
|
|
|
|
|
|
|
/* Restore vehicle group */
|
2010-08-18 20:48:38 +00:00
|
|
|
DoCommand(0, this->group, v->index, DC_EXEC, CMD_ADD_VEHICLE_GROUP);
|
2010-08-18 18:52:16 +00:00
|
|
|
}
|
2010-08-18 15:15:59 +00:00
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Create an order backup for the given vehicle.
|
|
|
|
* @param v The vehicle to make a backup of.
|
|
|
|
* @param user The user that is requesting the backup.
|
|
|
|
* @note Will automatically remove any previous backups of this user.
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
/* static */ void OrderBackup::Backup(const Vehicle *v, uint32 user)
|
2010-08-18 18:52:16 +00:00
|
|
|
{
|
2010-08-18 20:48:38 +00:00
|
|
|
/* Don't use reset as that broadcasts over the network to reset the variable,
|
|
|
|
* which is what we are doing at the moment. */
|
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
|
|
|
if (ob->user == user) delete ob;
|
|
|
|
}
|
|
|
|
new OrderBackup(v, user);
|
2010-08-18 15:15:59 +00:00
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Restore the data of this order to the given vehicle.
|
|
|
|
* @param v The vehicle to restore to.
|
|
|
|
* @param user The user that built the vehicle, thus wants to restore.
|
|
|
|
* @note After restoration the backup will automatically be removed.
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
/* static */ void OrderBackup::Restore(Vehicle *v, uint32 user)
|
2010-08-18 15:15:59 +00:00
|
|
|
{
|
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
2010-08-18 20:48:38 +00:00
|
|
|
if (v->tile != ob->tile || ob->user != user) continue;
|
2010-08-18 18:52:16 +00:00
|
|
|
|
|
|
|
ob->DoRestore(v);
|
|
|
|
delete ob;
|
2010-08-18 15:15:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Reset an OrderBackup given a tile and user.
|
|
|
|
* @param tile The tile associated with the OrderBackup.
|
|
|
|
* @param user The user associated with the OrderBackup.
|
|
|
|
* @note Must not be used from the GUI!
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
/* static */ void OrderBackup::ResetOfUser(TileIndex tile, uint32 user)
|
|
|
|
{
|
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
|
|
|
if (ob->user == user && (ob->tile == tile || tile == INVALID_TILE)) delete ob;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clear an OrderBackup
|
|
|
|
* @param tile Tile related to the to-be-cleared OrderBackup.
|
|
|
|
* @param flags For command.
|
|
|
|
* @param p1 Unused.
|
|
|
|
* @param p2 User that had the OrderBackup.
|
|
|
|
* @param text Unused.
|
|
|
|
* @return The cost of this operation or an error.
|
|
|
|
*/
|
|
|
|
CommandCost CmdClearOrderBackup(TileIndex tile, DoCommandFlag flags, uint32 p1, uint32 p2, const char *text)
|
2010-08-18 15:15:59 +00:00
|
|
|
{
|
2010-08-18 20:48:38 +00:00
|
|
|
/* No need to check anything. If the tile or user don't exist we just ignore it. */
|
|
|
|
if (flags & DC_EXEC) OrderBackup::ResetOfUser(tile == 0 ? INVALID_TILE : tile, p2);
|
|
|
|
|
|
|
|
return CommandCost();
|
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Reset an user's OrderBackup if needed.
|
|
|
|
* @param user The user associated with the OrderBackup.
|
|
|
|
* @pre _network_server.
|
|
|
|
* @note Must not be used from a command.
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
/* static */ void OrderBackup::ResetUser(uint32 user)
|
|
|
|
{
|
|
|
|
assert(_network_server);
|
|
|
|
|
2010-08-18 18:52:16 +00:00
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
2010-08-18 20:48:38 +00:00
|
|
|
/* If it's not an backup of us, so ignore it. */
|
|
|
|
if (ob->user != user) continue;
|
|
|
|
|
|
|
|
DoCommandP(0, 0, user, CMD_CLEAR_ORDER_BACKUP);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Reset the OrderBackups from GUI/game logic.
|
|
|
|
* @param tile The tile of the order backup.
|
|
|
|
* @param from_gui Whether the call came from the GUI, i.e. whether
|
|
|
|
* it must be synced over the network.
|
|
|
|
*/
|
2010-08-18 20:48:38 +00:00
|
|
|
/* static */ void OrderBackup::Reset(TileIndex t, bool from_gui)
|
|
|
|
{
|
|
|
|
/* The user has CLIENT_ID_SERVER as default when network play is not active,
|
|
|
|
* but compiled it. A network client has its own variable for the unique
|
|
|
|
* client/user identifier. Finally if networking isn't compiled in the
|
|
|
|
* default is just plain and simple: 0. */
|
|
|
|
#ifdef ENABLE_NETWORK
|
|
|
|
uint32 user = _networking && !_network_server ? _network_own_client_id : CLIENT_ID_SERVER;
|
|
|
|
#else
|
|
|
|
uint32 user = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
|
|
|
/* If it's not an backup of us, so ignore it. */
|
|
|
|
if (ob->user != user) continue;
|
|
|
|
/* If it's not for our chosen tile either, ignore it. */
|
|
|
|
if (t != INVALID_TILE && t != ob->tile) continue;
|
|
|
|
|
|
|
|
if (from_gui) {
|
2010-12-07 20:45:11 +00:00
|
|
|
/* We need to circumvent the "prevention" from this command being executed
|
|
|
|
* while the game is paused, so use the internal method. Nor do we want
|
|
|
|
* this command to get its cost estimated when shift is pressed. */
|
2010-12-08 18:21:29 +00:00
|
|
|
DoCommandPInternal(ob->tile, 0, user, CMD_CLEAR_ORDER_BACKUP, NULL, NULL, true, false);
|
2010-08-18 20:48:38 +00:00
|
|
|
} else {
|
|
|
|
/* The command came from the game logic, i.e. the clearing of a tile.
|
|
|
|
* In that case we have no need to actually sync this, just do it. */
|
|
|
|
delete ob;
|
|
|
|
}
|
2010-08-18 18:52:16 +00:00
|
|
|
}
|
2010-08-18 15:15:59 +00:00
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Clear the group of all backups having this group ID.
|
|
|
|
* @param group The group to clear.
|
|
|
|
*/
|
2010-08-18 15:15:59 +00:00
|
|
|
/* static */ void OrderBackup::ClearGroup(GroupID group)
|
|
|
|
{
|
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
|
|
|
if (ob->group == group) ob->group = DEFAULT_GROUP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-18 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* Clear/update the (clone) vehicle from an order backup.
|
|
|
|
* @param v The vehicle to clear.
|
|
|
|
* @pre v != NULL
|
|
|
|
* @note If it is not possible to set another vehicle as clone
|
|
|
|
* "example", then this backed up order will be removed.
|
|
|
|
*/
|
2010-08-18 15:58:30 +00:00
|
|
|
/* static */ void OrderBackup::ClearVehicle(const Vehicle *v)
|
|
|
|
{
|
|
|
|
assert(v != NULL);
|
|
|
|
OrderBackup *ob;
|
|
|
|
FOR_ALL_ORDER_BACKUPS(ob) {
|
|
|
|
if (ob->clone == v) {
|
|
|
|
/* Get another item in the shared list. */
|
|
|
|
ob->clone = (v->FirstShared() == v) ? v->NextShared() : v->FirstShared();
|
|
|
|
/* But if that isn't there, remove it. */
|
|
|
|
if (ob->clone == NULL) delete ob;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-18 15:15:59 +00:00
|
|
|
void InitializeOrderBackups()
|
|
|
|
{
|
|
|
|
_order_backup_pool.CleanPool();
|
|
|
|
}
|