/* * 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 . */ /** @file plans_cmd.cpp Handling of plan related commands. */ #include "stdafx.h" #include "command_func.h" #include "command_aux.h" #include "plans_base.h" #include "plans_func.h" #include "window_func.h" #include "company_func.h" #include "company_base.h" #include "string_func.h" #include "window_gui.h" #include "table/strings.h" /** * Create a new plan. * @param tile unused * @param flags type of operation * @param p1 unused * @param p2 unused * @param text unused * @return the cost of this operation or an error */ CommandCost CmdAddPlan(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { if (!Plan::CanAllocateItem()) return_cmd_error(STR_ERROR_TOO_MANY_PLANS); if (flags & DC_EXEC) { _new_plan = new Plan(_current_company); } return CommandCost(); } struct PlanLineCmdData : public CommandAuxiliarySerialisable { TileVector tiles; virtual void Serialise(CommandSerialisationBuffer &buffer) const override { buffer.Send_uint32((uint32_t)this->tiles.size()); for (TileIndex t : this->tiles) { buffer.Send_uint32(t); } } CommandCost Deserialise(CommandDeserialisationBuffer &buffer) { uint32_t size = buffer.Recv_uint32(); if (!buffer.CanRecvBytes(size * 4)) return CMD_ERROR; this->tiles.resize(size); for (uint i = 0; i < size; i++) { this->tiles[i] = buffer.Recv_uint32(); } return CommandCost(); } }; bool AddPlanLine(PlanID plan, TileVector tiles) { PlanLineCmdData data; data.tiles = std::move(tiles); return DoCommandPEx(0, plan, 0, 0, CMD_ADD_PLAN_LINE, nullptr, nullptr, &data); } /** * Create a new line in a plan. * @param tile unused * @param flags type of operation * @param p1 plan id * @param p2 number of nodes * @param text list of tile indexes that compose the line * @param aux_data auxiliary data * @return the cost of this operation or an error */ CommandCost CmdAddPlanLine(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, uint64_t p3, const char *text, const CommandAuxiliaryBase *aux_data) { Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; CommandCost ret = CheckOwnership(p->owner); if (ret.Failed()) return ret; CommandAuxData data; ret = data.Load(aux_data); if (ret.Failed()) return ret; if (data->tiles.size() > (MAX_CMD_TEXT_LENGTH / sizeof(TileIndex))) return_cmd_error(STR_ERROR_TOO_MANY_NODES); if (flags & DC_EXEC) { PlanLine *pl = p->NewLine(); if (!pl) return_cmd_error(STR_ERROR_NO_MORE_SPACE_FOR_LINES); pl->tiles = std::move(data->tiles); pl->UpdateVisualExtents(); if (p->IsListable()) { pl->SetVisibility(p->visible); if (p->visible) pl->MarkDirty(); Window *w = FindWindowById(WC_PLANS, 0); if (w) w->InvalidateData(INVALID_PLAN, false); } } return CommandCost(); } /** * Edit the visibility of a plan. * @param tile unused * @param flags type of operation * @param p1 plan id * @param p2 visibility * @param text unused * @return the cost of this operation or an error */ CommandCost CmdChangePlanVisibility(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; CommandCost ret = CheckOwnership(p->owner); if (ret.Failed()) return ret; if (flags & DC_EXEC) { bool visible = (p2 != 0); if (p->visible_by_all != visible) { p->visible_by_all = visible; Window *w = FindWindowById(WC_PLANS, 0); if (w) w->InvalidateData(INVALID_PLAN, false); if (p->owner != _local_company && p->visible) { for (PlanLine *line : p->lines) { if (line->visible) line->MarkDirty(); } } } } return CommandCost(); } /** * Edit the colour of a plan. * @param tile unused * @param flags type of operation * @param p1 plan id * @param p2 colour * @param text unused * @return the cost of this operation or an error */ CommandCost CmdChangePlanColour(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; if (p2 >= COLOUR_END) return CMD_ERROR; CommandCost ret = CheckOwnership(p->owner); if (ret.Failed()) return ret; if (flags & DC_EXEC) { p->colour = (Colours)p2; _plan_update_counter++; Window *w = FindWindowById(WC_PLANS, 0); if (w) w->InvalidateData(INVALID_PLAN, false); for (const PlanLine *line : p->lines) { if (line->visible) line->MarkDirty(); } if (p->temp_line) p->temp_line->MarkDirty(); } return CommandCost(); } /** * Delete a plan. * @param tile unused * @param flags type of operation * @param p1 plan id * @param p2 unused * @param text unused * @return the cost of this operation or an error */ CommandCost CmdRemovePlan(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; CommandCost ret = CheckOwnership(p->owner); if (ret.Failed()) return ret; if (flags & DC_EXEC) { if (p->IsListable()) { p->SetVisibility(false); Window *w = FindWindowById(WC_PLANS, 0); if (w) w->InvalidateData(p->index, false); } if (p == _current_plan) _current_plan = nullptr; delete p; } return CommandCost(); } /** * Remove a line from a plan. * @param tile unused * @param flags type of operation * @param p1 plan id * @param p2 line id * @param text unused * @return the cost of this operation or an error */ CommandCost CmdRemovePlanLine(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; CommandCost ret = CheckOwnership(p->owner); if (ret.Failed()) return ret; if (p2 >= p->lines.size()) return CMD_ERROR; if (flags & DC_EXEC) { PlanLineVector::iterator it = p->lines.begin(); std::advance(it, p2); (*it)->SetVisibility(false); delete *it; p->lines.erase(it); if (p->IsListable()) { Window *w = FindWindowById(WC_PLANS, 0); if (w) w->InvalidateData(p->index, false); } } return CommandCost(); } /** * Give a custom name to your plan * @param tile unused * @param flags type of operation * @param p1 ID of plan to name * @param p2 unused * @param text the new name * @return the cost of this operation or an error */ CommandCost CmdRenamePlan(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { if (text == nullptr) return CMD_ERROR; Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; CommandCost ret = CheckOwnership(p->owner); if (ret.Failed()) return ret; if (Utf8StringLength(text) >= MAX_LENGTH_PLAN_NAME_CHARS) return CMD_ERROR; if (flags & DC_EXEC) { p->name = text; InvalidateWindowClassesData(WC_PLANS); } return CommandCost(); } /** * Acquire an unowned plan * @param tile unused * @param flags type of operation * @param p1 ID of plan * @param p2 unused * @param text unused * @return the cost of this operation or an error */ CommandCost CmdAcquireUnownedPlan(TileIndex tile, DoCommandFlag flags, uint32_t p1, uint32_t p2, const char *text) { Plan *p = Plan::GetIfValid(p1); if (p == nullptr) return CMD_ERROR; if (Company::IsValidID(p->owner)) return CMD_ERROR; if (!Company::IsValidID(_current_company)) return CMD_ERROR; if (flags & DC_EXEC) { p->owner = _current_company; InvalidateWindowClassesData(WC_PLANS); if (p->visible) { for (PlanLine *line : p->lines) { if (line->visible) line->MarkDirty(); } } } return CommandCost(); }