diff --git a/megamek/data/images/misc/cargo.png b/megamek/data/images/misc/cargo.png new file mode 100644 index 00000000000..1a7b17f8274 Binary files /dev/null and b/megamek/data/images/misc/cargo.png differ diff --git a/megamek/i18n/megamek/client/messages.properties b/megamek/i18n/megamek/client/messages.properties index 0328aa11b82..84dbd79fd54 100644 --- a/megamek/i18n/megamek/client/messages.properties +++ b/megamek/i18n/megamek/client/messages.properties @@ -1701,6 +1701,8 @@ DeployMinefieldDisplay.its_your_turn=It's your turn to deploy minefields. DeployMinefieldDisplay.removeMines=Remove DeployMinefieldDisplay.waitingForDeploymentPhase=Waiting to begin Deployment phase... DeployMinefieldDisplay.waitingForDeployMinefieldPhase=Waiting to begin Deploy minefield phase... +DeployMinefieldDisplay.deployCarriable=Cargo({0}) +DeployMinefieldDisplay.deployCarriableDialogHeader=Deploy Ground Object #Expand Map Dialog ExpandMapDialog.title=Expand map settings @@ -2431,6 +2433,8 @@ MovementDisplay.butLoad=Load MovementDisplay.butLand=Land MovementDisplay.butJoin=Join MovementDisplay.butManeuver=Maneuver +MovementDisplay.moveDropCargo=Drop Cargo +MovementDisplay.movePickupCargo=Pick Up Cargo MovementDisplay.moveModeConvert=Convert Mode MovementDisplay.moveModeLeg=Walk MovementDisplay.moveModeTrack=Engage tracks @@ -2875,6 +2879,7 @@ PlayerSettingsDialog.header.initMod=Initiative Modifier PlayerSettingsDialog.header.minefields=Minefields PlayerSettingsDialog.header.skills=Method for Rolling Pilot Skills PlayerSettingsDialog.header.email=Round Report Email +PlayerSettingsDialog.header.GroundObjects=Carryable Ground Objects PlayerSettingsDialog.botSettings=Bot Settings... PlayerSettingsDialog.autoConfigFaction=Faction: PlayerSettingsDialog.autoConfig=Autoconfig @@ -3934,6 +3939,8 @@ WeaponAttackAction.BPodAtInf=B-Pod firing at infantry WeaponAttackAction.BPodOnlyAtInf=B-Pods can't target non-infantry. WeaponAttackAction.CantAimAndCallShots=you can't combine aimed shots and called shots. WeaponAttackAction.CantClearMines=Weapon can't clear minefields. +WeaponAttackAction.CantFireWhileCarryingCargo=Carrying cargo prevents arm/torso weapon firing. +WeaponAttackAction.CantFireWhileLoadingUnloadingCargo=Cannot fire weapons while loading/unloading cargo. WeaponAttackAction.CantFireWhileGrappled=Can only fire head and front torso weapons when grappled. WeaponAttackAction.CantFireWithOtherWeapons=Already firing a weapon that can only be fired by itself! (%s) WeaponAttackAction.CantFireArmsAndMainGun=Can't fire arm-mounted weapons and the main gun in the same turn. diff --git a/megamek/i18n/megamek/common/report-messages.properties b/megamek/i18n/megamek/common/report-messages.properties index 6a9c54306bc..a724b92ed6a 100755 --- a/megamek/i18n/megamek/common/report-messages.properties +++ b/megamek/i18n/megamek/common/report-messages.properties @@ -73,6 +73,7 @@ 2005= () flees the battlefield. 2010=It carries () with it. 2015=It takes () with it. +2016=It carries with it. 2020= ejects from a (). 2025= () is abandoned by its crew. 2026= () is given the order to abandon ship. @@ -254,6 +255,11 @@ 2510=, Starts to flip. 2511=, Capsizes and sinks. 2512= deploys a chaff pod. +2513= loads from . +2514= unloads in +2515= is dropped to the ground. Needs , rolls : +2516=success - cargo remains intact. +2517=failure - cargo destroyed! 3000=Weapon Attack Phase------------------- 3003=Inferno fire (bombs) started in hex . @@ -980,6 +986,11 @@ 6700=Limb Blow off avoided due to armored actuator. 6710=Critical hit to avoided due to armored component. +#cargo reports. +6720= damaged by incoming attack. tons remaining. +6721= damaged by incoming attack and is completely destroyed. +6722= is dropped to the ground due to the carrying unit falling. + 6800= () was not properly secured prior to takeoff. 7000=Victory!------------------- diff --git a/megamek/src/megamek/client/Client.java b/megamek/src/megamek/client/Client.java index 4ecbb99feca..716b9bbb6b2 100644 --- a/megamek/src/megamek/client/Client.java +++ b/megamek/src/megamek/client/Client.java @@ -385,6 +385,13 @@ public void sendAddSquadron(FighterSquadron fs, Collection fighterIds) public void sendDeployMinefields(Vector minefields) { send(new Packet(PacketCommand.DEPLOY_MINEFIELDS, minefields)); } + + /** + * Sends an updated state of ground objects (i.e. cargo etc) + */ + public void sendDeployGroundObjects(Map> groundObjects) { + send(new Packet(PacketCommand.UPDATE_GROUND_OBJECTS, groundObjects)); + } /** * Sends a "set Artillery Autohit Hexes" packet @@ -576,6 +583,12 @@ protected void receiveEntityVisibilityIndicator(Packet packet) { game.processGameEvent(new GameEntityChangeEvent(this, e)); } } + + @SuppressWarnings("unchecked") + protected void receiveUpdateGroundObjects(Packet packet) { + game.setGroundObjects((Map>) packet.getObject(0)); + game.processGameEvent(new GameBoardChangeEvent(this)); + } @SuppressWarnings("unchecked") protected void receiveDeployMinefields(Packet packet) { @@ -885,6 +898,9 @@ protected boolean handleGameSpecificPacket(Packet packet) { case REMOVE_MINEFIELD: receiveRemoveMinefield(packet); break; + case UPDATE_GROUND_OBJECTS: + receiveUpdateGroundObjects(packet); + break; case ADD_SMOKE_CLOUD: SmokeCloud cloud = (SmokeCloud) packet.getObject(0); game.addSmokeCloud(cloud); diff --git a/megamek/src/megamek/client/ui/swing/ClientGUI.java b/megamek/src/megamek/client/ui/swing/ClientGUI.java index b2f87f92c0d..2db706e1b7b 100644 --- a/megamek/src/megamek/client/ui/swing/ClientGUI.java +++ b/megamek/src/megamek/client/ui/swing/ClientGUI.java @@ -233,6 +233,7 @@ public class ClientGUI extends AbstractClientGUI implements BoardViewListener, private MovementModifierSpriteHandler movementModifierSpriteHandler; private SensorRangeSpriteHandler sensorRangeSpriteHandler; private CollapseWarningSpriteHandler collapseWarningSpriteHandler; + private GroundObjectSpriteHandler groundObjectSpriteHandler; private FiringSolutionSpriteHandler firingSolutionSpriteHandler; private FiringArcSpriteHandler firingArcSpriteHandler; @@ -490,12 +491,13 @@ private void initializeSpriteHandlers() { FlareSpritesHandler flareSpritesHandler = new FlareSpritesHandler(bv, client.getGame()); sensorRangeSpriteHandler = new SensorRangeSpriteHandler(bv, client.getGame()); collapseWarningSpriteHandler = new CollapseWarningSpriteHandler(bv); + groundObjectSpriteHandler = new GroundObjectSpriteHandler(bv, client.getGame()); firingSolutionSpriteHandler = new FiringSolutionSpriteHandler(bv, client); firingArcSpriteHandler = new FiringArcSpriteHandler(bv, this); spriteHandlers.addAll(List.of(movementEnvelopeHandler, movementModifierSpriteHandler, sensorRangeSpriteHandler, flareSpritesHandler, collapseWarningSpriteHandler, - firingSolutionSpriteHandler, firingArcSpriteHandler)); + groundObjectSpriteHandler, firingSolutionSpriteHandler, firingArcSpriteHandler)); spriteHandlers.forEach(BoardViewSpriteHandler::initialize); } @@ -1177,7 +1179,7 @@ void switchPanel(GamePhase phase) { // otherwise, hide the panel panSecondary.setVisible(false); } - + // Set the new panel's listeners if (curPanel instanceof BoardViewListener) { bv.addBoardViewListener((BoardViewListener) curPanel); @@ -2912,11 +2914,20 @@ public void showSensorRanges(Entity entity, Coords assumedPosition) { /** * Shows collapse warnings in the given list of Coords in the BoardView * - * @param warnList The Coords to show the warning on + * @param warnList The list of coordinates to show the warning on */ public void showCollapseWarning(List warnList) { collapseWarningSpriteHandler.setCFWarningSprites(warnList); } + + /** + * Shows ground object icons in the given list of Coords in the BoardView + * + * @param groundObjectList The list of coordinates to show + */ + public void showGroundObjects(Map> groundObjectList) { + groundObjectSpriteHandler.setGroundObjectSprites(groundObjectList); + } /** * Shows firing solutions from the viewpoint of the given entity on targets diff --git a/megamek/src/megamek/client/ui/swing/DeployMinefieldDisplay.java b/megamek/src/megamek/client/ui/swing/DeployMinefieldDisplay.java index 1d8dd208f26..232c4bd03bd 100644 --- a/megamek/src/megamek/client/ui/swing/DeployMinefieldDisplay.java +++ b/megamek/src/megamek/client/ui/swing/DeployMinefieldDisplay.java @@ -15,7 +15,6 @@ import megamek.client.event.BoardViewEvent; import megamek.client.ui.Messages; -import megamek.client.ui.swing.util.KeyCommandBind; import megamek.client.ui.swing.widget.MegamekButton; import megamek.common.*; import megamek.common.event.GamePhaseChangeEvent; @@ -25,8 +24,7 @@ import java.awt.event.MouseEvent; import java.util.*; -import static megamek.client.ui.swing.util.UIUtil.guiScaledFontHTML; -import static megamek.client.ui.swing.util.UIUtil.uiLightViolet; +import javax.swing.JOptionPane; public class DeployMinefieldDisplay extends StatusBarPhaseDisplay { private static final long serialVersionUID = -1243277953037374936L; @@ -34,17 +32,22 @@ public class DeployMinefieldDisplay extends StatusBarPhaseDisplay { /** * This enumeration lists all of the possible ActionCommands that can be * carried out during the deploy minefield phase. Each command has a string - * for the command plus a flag that determines what unit type it is - * appropriate for. + * for the command. * @author arlith */ public static enum DeployMinefieldCommand implements PhaseCommand { + COMMAND_NONE("noCommand"), DEPLOY_MINE_CONV("deployMineConv"), DEPLOY_MINE_COM("deployMineCom"), DEPLOY_MINE_VIBRA("deployMineVibra"), DEPLOY_MINE_ACTIVE("deployMineActive"), DEPLOY_MINE_INFERNO("deployMineInferno"), + DEPLOY_CARRYABLE("deployCarriable"), REMOVE_MINES("removeMines"); + + private static DeployMinefieldCommand[] actualCommands = + { DEPLOY_MINE_CONV, DEPLOY_MINE_COM, DEPLOY_MINE_VIBRA, DEPLOY_MINE_ACTIVE, + DEPLOY_MINE_INFERNO, DEPLOY_CARRYABLE, REMOVE_MINES }; /** * Priority that determines this buttons order @@ -56,6 +59,26 @@ public static enum DeployMinefieldCommand implements PhaseCommand { cmd = c; } + /** + * Given a string, figure out the command value + */ + public static DeployMinefieldCommand fromString(String command) { + for (DeployMinefieldCommand value : values()) { + if (value.getCmd().equals(command)) { + return value; + } + } + + return null; + } + + /** + * Get all the commands that aren't NO-OP + */ + public static DeployMinefieldCommand[] getActualCommands() { + return actualCommands; + } + @Override public String getCmd() { return cmd; @@ -82,13 +105,28 @@ public String getHotKeyDesc() { // buttons protected Map buttons; - private boolean deployM = false; - private boolean deployC = false; - private boolean deployV = false; - private boolean deployA = false; - private boolean deployI = false; - private boolean remove = false; + DeployMinefieldCommand currentCommand = DeployMinefieldCommand.COMMAND_NONE; + private boolean deployingConventionalMinefields() { + return currentCommand.equals(DeployMinefieldCommand.DEPLOY_MINE_CONV); + } + + private boolean deployingActiveMinefields() { + return currentCommand.equals(DeployMinefieldCommand.DEPLOY_MINE_ACTIVE); + } + + private boolean deployingInfernoMinefields() { + return currentCommand.equals(DeployMinefieldCommand.DEPLOY_MINE_INFERNO); + } + + private boolean deployingCommandMinefields() { + return currentCommand.equals(DeployMinefieldCommand.DEPLOY_MINE_COM); + } + + private boolean deployingVibrabombMinefields() { + return currentCommand.equals(DeployMinefieldCommand.DEPLOY_MINE_VIBRA); + } + private Player p; private Vector deployedMinefields = new Vector<>(); @@ -119,16 +157,16 @@ public DeployMinefieldDisplay(ClientGUI clientgui) { @Override protected void setButtons() { - buttons = new HashMap<>((int) (DeployMinefieldCommand.values().length * 1.25 + 0.5)); - for (DeployMinefieldCommand cmd : DeployMinefieldCommand.values()) { - buttons.put(cmd, createButton(cmd.getCmd(), "DeployMinefieldDisplay.")); + buttons = new HashMap<>((int) (DeployMinefieldCommand.getActualCommands().length * 1.25 + 0.5)); + for (DeployMinefieldCommand cmd : DeployMinefieldCommand.getActualCommands()) { + buttons.put(cmd, createButton(cmd.getCmd(), "DeployMinefieldDisplay.")); } numButtonGroups = (int) Math.ceil((buttons.size()+0.0) / buttonsPerGroup); } @Override protected void setButtonsTooltips() { - for (DeployMinefieldCommand cmd : DeployMinefieldCommand.values()) { + for (DeployMinefieldCommand cmd : DeployMinefieldCommand.getActualCommands()) { String tt = createToolTip(cmd.getCmd(), "DeployMinefieldDisplay.", cmd.getHotKeyDesc()); buttons.get(cmd).setToolTipText(tt); } @@ -137,7 +175,7 @@ protected void setButtonsTooltips() { @Override protected ArrayList getButtonList() { ArrayList buttonList = new ArrayList<>(); - for (DeployMinefieldCommand cmd : DeployMinefieldCommand.values()) { + for (DeployMinefieldCommand cmd : DeployMinefieldCommand.getActualCommands()) { buttonList.add(buttons.get(cmd)); } return buttonList; @@ -153,6 +191,7 @@ private void beginMyTurn() { setVibrabombEnabled(p.getNbrMFVibra()); setActiveEnabled(p.getNbrMFActive()); setInfernoEnabled(p.getNbrMFInferno()); + setCarryableEnabled(p.getGroundObjectsToPlace().size()); setRemoveMineEnabled(true); butDone.setEnabled(true); @@ -182,28 +221,32 @@ private void disableButtons() { setVibrabombEnabled(0); setActiveEnabled(0); setInfernoEnabled(0); + setCarryableEnabled(0); setRemoveMineEnabled(false); butDone.setEnabled(false); } private void deployMinefield(Coords coords) { - if (!clientgui.getClient().getGame().getBoard().contains(coords)) { + Game game = clientgui.getClient().getGame(); + + if (!game.getBoard().contains(coords)) { return; } // check if this is a water hex boolean sea = false; - Hex hex = clientgui.getClient().getGame().getBoard().getHex(coords); + Hex hex = game.getBoard().getHex(coords); if (hex.containsTerrain(Terrains.WATER)) { sea = true; } - if (remove) { - if (!clientgui.getClient().getGame().containsMinefield(coords)) { + if (currentCommand == DeployMinefieldCommand.REMOVE_MINES) { + if (!game.containsMinefield(coords) && + game.getGroundObjects(coords).size() == 0) { return; } - Enumeration mfs = clientgui.getClient().getGame().getMinefields(coords).elements(); + Enumeration mfs = game.getMinefields(coords).elements(); ArrayList mfRemoved = new ArrayList<>(); while (mfs.hasMoreElements()) { Minefield mf = (Minefield) mfs.nextElement(); @@ -225,19 +268,50 @@ private void deployMinefield(Coords coords) { } for (Minefield mf : mfRemoved) { - clientgui.getClient().getGame().removeMinefield(mf); + game.removeMinefield(mf); + } + + // remove all carryables here as well and put them back to the player + for (ICarryable carryable : game.getGroundObjects(coords)) { + p.getGroundObjectsToPlace().add(carryable); } + + game.getGroundObjects().remove(coords); + + clientgui.showGroundObjects(game.getGroundObjects()); + + } else if (currentCommand == DeployMinefieldCommand.DEPLOY_CARRYABLE) { + List groundObjects = p.getGroundObjectsToPlace(); + + ICarryable toDeploy = groundObjects.get(0); + + if (groundObjects.size() > 1) { + String title = "Choose Cargo to Place"; + String body = "Choose the cargo to place:"; + toDeploy = (ICarryable) JOptionPane.showInputDialog(clientgui.getFrame(), + body, title, JOptionPane.QUESTION_MESSAGE, null, + groundObjects.toArray(), groundObjects.get(0)); + } + + game.placeGroundObject(coords, toDeploy); + groundObjects.remove(toDeploy); + + if (groundObjects.size() <= 0) { + currentCommand = DeployMinefieldCommand.COMMAND_NONE; + } + + clientgui.showGroundObjects(game.getGroundObjects()); } else { - // first check that there is not already a mine of this type + // first check that there is not already a mine of this type // deployed - Enumeration mfs = clientgui.getClient().getGame().getMinefields(coords).elements(); + Enumeration mfs = game.getMinefields(coords).elements(); while (mfs.hasMoreElements()) { Minefield mf = (Minefield) mfs.nextElement(); - if ((deployM && (mf.getType() == Minefield.TYPE_CONVENTIONAL)) - || (deployC && (mf.getType() == Minefield.TYPE_COMMAND_DETONATED)) - || (deployV && (mf.getType() == Minefield.TYPE_VIBRABOMB)) - || (deployA && (mf.getType() == Minefield.TYPE_ACTIVE)) - || (deployI && (mf.getType() == Minefield.TYPE_INFERNO))) { + if ((deployingConventionalMinefields() && (mf.getType() == Minefield.TYPE_CONVENTIONAL)) + || (deployingCommandMinefields() && (mf.getType() == Minefield.TYPE_COMMAND_DETONATED)) + || (deployingVibrabombMinefields() && (mf.getType() == Minefield.TYPE_VIBRABOMB)) + || (deployingActiveMinefields() && (mf.getType() == Minefield.TYPE_ACTIVE)) + || (deployingInfernoMinefields() && (mf.getType() == Minefield.TYPE_INFERNO))) { clientgui.doAlertDialog(Messages.getString("DeployMinefieldDisplay.IllegalPlacement"), Messages.getString("DeployMinefieldDisplay.DuplicateMinefield")); return; @@ -245,13 +319,13 @@ private void deployMinefield(Coords coords) { } Minefield mf = null; - if (sea && !(deployM || deployI)) { + if (sea && !(deployingConventionalMinefields() || deployingInfernoMinefields())) { clientgui.doAlertDialog(Messages.getString("DeployMinefieldDisplay.IllegalPlacement"), Messages.getString("DeployMinefieldDisplay.WaterPlacement")); return; } int depth = 0; - if (deployM) { + if (deployingConventionalMinefields()) { if (sea) { SeaMineDepthDialog smd = new SeaMineDepthDialog( clientgui.frame, hex.depth()); @@ -267,8 +341,12 @@ private void deployMinefield(Coords coords) { Minefield.TYPE_CONVENTIONAL, mfd.getDensity(), sea, depth); p.setNbrMFConventional(p.getNbrMFConventional() - 1); + + if (p.getNbrMFConventional() <= 0) { + currentCommand = DeployMinefieldCommand.COMMAND_NONE; + } } - } else if (deployC) { + } else if (deployingCommandMinefields()) { MineDensityDialog mfd = new MineDensityDialog(clientgui.frame); mfd.setVisible(true); @@ -277,8 +355,12 @@ private void deployMinefield(Coords coords) { Minefield.TYPE_COMMAND_DETONATED, mfd.getDensity(), sea, depth); p.setNbrMFCommand(p.getNbrMFCommand() - 1); + + if (p.getNbrMFCommand() <= 0) { + currentCommand = DeployMinefieldCommand.COMMAND_NONE; + } } - } else if (deployA) { + } else if (deployingActiveMinefields()) { MineDensityDialog mfd = new MineDensityDialog(clientgui.frame); mfd.setVisible(true); @@ -286,8 +368,12 @@ private void deployMinefield(Coords coords) { mf = Minefield.createMinefield(coords, p.getId(), Minefield.TYPE_ACTIVE, mfd.getDensity()); p.setNbrMFActive(p.getNbrMFActive() - 1); + + if (p.getNbrMFActive() <= 0) { + currentCommand = DeployMinefieldCommand.COMMAND_NONE; + } } - } else if (deployI) { + } else if (deployingInfernoMinefields()) { MineDensityDialog mfd = new MineDensityDialog(clientgui.frame); mfd.setVisible(true); @@ -296,8 +382,12 @@ private void deployMinefield(Coords coords) { Minefield.TYPE_INFERNO, mfd.getDensity(), sea, depth); p.setNbrMFInferno(p.getNbrMFInferno() - 1); + + if (p.getNbrMFInferno() <= 0) { + currentCommand = DeployMinefieldCommand.COMMAND_NONE; + } } - } else if (deployV) { + } else if (deployingVibrabombMinefields()) { MineDensityDialog mfd = new MineDensityDialog(clientgui.frame); mfd.setVisible(true); @@ -310,13 +400,17 @@ private void deployMinefield(Coords coords) { Minefield.TYPE_VIBRABOMB, mfd.getDensity(), vsd.getSetting()); p.setNbrMFVibra(p.getNbrMFVibra() - 1); + + if (p.getNbrMFVibra() <= 0) { + currentCommand = DeployMinefieldCommand.COMMAND_NONE; + } } } else { return; } if (mf != null) { mf.setWeaponDelivered(false); - clientgui.getClient().getGame().addMinefield(mf); + game.addMinefield(mf); deployedMinefields.addElement(mf); } clientgui.getBoardView().refreshDisplayables(); @@ -327,28 +421,12 @@ private void deployMinefield(Coords coords) { setVibrabombEnabled(p.getNbrMFVibra()); setActiveEnabled(p.getNbrMFActive()); setInfernoEnabled(p.getNbrMFInferno()); - - if (p.getNbrMFConventional() == 0) { - deployM = false; - } - if (p.getNbrMFCommand() == 0) { - deployC = false; - } - if (p.getNbrMFVibra() == 0) { - deployV = false; - } - if (p.getNbrMFActive() == 0) { - deployA = false; - } - if (p.getNbrMFInferno() == 0) { - deployI = false; - } - + setCarryableEnabled(p.getGroundObjectsToPlace().size()); } @Override public void clear() { - //TODO: undefined for now + } // @@ -435,54 +513,15 @@ public void actionPerformed(ActionEvent ev) { if (!clientgui.getClient().isMyTurn()) { // odd... return; - } else if (ev.getActionCommand().equals(DeployMinefieldCommand.DEPLOY_MINE_CONV.getCmd())) { - deployM = true; - deployC = false; - deployV = false; - deployA = false; - deployI = false; - remove = false; - } else if (ev.getActionCommand().equals(DeployMinefieldCommand.DEPLOY_MINE_COM.getCmd())) { - deployM = false; - deployC = true; - deployV = false; - deployA = false; - deployI = false; - remove = false; - } else if (ev.getActionCommand().equals(DeployMinefieldCommand.DEPLOY_MINE_VIBRA.getCmd())) { - deployM = false; - deployC = false; - deployV = true; - deployA = false; - deployI = false; - remove = false; - } else if (ev.getActionCommand().equals(DeployMinefieldCommand.DEPLOY_MINE_ACTIVE.getCmd())) { - deployM = false; - deployC = false; - deployV = false; - deployA = true; - deployI = false; - remove = false; - } else if (ev.getActionCommand().equals(DeployMinefieldCommand.DEPLOY_MINE_INFERNO.getCmd())) { - deployM = false; - deployC = false; - deployV = false; - deployA = false; - deployI = true; - remove = false; - } else if (ev.getActionCommand().equals(DeployMinefieldCommand.REMOVE_MINES.getCmd())) { - deployM = false; - deployC = false; - deployV = false; - deployA = false; - deployI = false; - remove = true; } + + currentCommand = DeployMinefieldCommand.fromString(ev.getActionCommand()); } // End public void actionPerformed(ActionEvent ev) @Override public void ready() { endMyTurn(); + clientgui.getClient().sendDeployGroundObjects(clientgui.getClient().getGame().getGroundObjects()); clientgui.getClient().sendDeployMinefields(deployedMinefields); clientgui.getClient().sendPlayerInfo(); } @@ -516,6 +555,12 @@ private void setInfernoEnabled(int nbr) { "DeployMinefieldDisplay." + DeployMinefieldCommand.DEPLOY_MINE_INFERNO.getCmd(), nbr)); buttons.get(DeployMinefieldCommand.DEPLOY_MINE_INFERNO).setEnabled(nbr > 0); } + + private void setCarryableEnabled(int nbr) { + buttons.get(DeployMinefieldCommand.DEPLOY_CARRYABLE).setText(Messages.getString( + "DeployMinefieldDisplay." + DeployMinefieldCommand.DEPLOY_CARRYABLE.getCmd(), nbr)); + buttons.get(DeployMinefieldCommand.DEPLOY_CARRYABLE).setEnabled(nbr > 0); + } private void setRemoveMineEnabled(boolean enable) { buttons.get(DeployMinefieldCommand.REMOVE_MINES).setEnabled(enable); diff --git a/megamek/src/megamek/client/ui/swing/DeploymentDisplay.java b/megamek/src/megamek/client/ui/swing/DeploymentDisplay.java index 40955580048..44b529e14e4 100644 --- a/megamek/src/megamek/client/ui/swing/DeploymentDisplay.java +++ b/megamek/src/megamek/client/ui/swing/DeploymentDisplay.java @@ -228,7 +228,7 @@ public void selectEntity(int en) { clientgui.getUnitDisplay().showPanel("movement"); clientgui.updateFiringArc(ce()); clientgui.showSensorRanges(ce()); - computeCFWarningHexes(ce()); + computeCFWarningHexes(ce()); } else { disableButtons(); setNextEnabled(true); diff --git a/megamek/src/megamek/client/ui/swing/MovementDisplay.java b/megamek/src/megamek/client/ui/swing/MovementDisplay.java index 614d303fb17..cd356bb1e26 100644 --- a/megamek/src/megamek/client/ui/swing/MovementDisplay.java +++ b/megamek/src/megamek/client/ui/swing/MovementDisplay.java @@ -77,12 +77,13 @@ public class MovementDisplay extends ActionPhaseDisplay { public static final int CMD_AIRMECH = 1 << 7; // Command used only in menus and has no associated button public static final int CMD_NO_BUTTON = 1 << 8; + public static final int CMD_PROTOMECH = 1 << 9; // Convenience defines for common combinations public static final int CMD_AERO_BOTH = CMD_AERO | CMD_AERO_VECTORED; - public static final int CMD_GROUND = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_INF; - public static final int CMD_NON_VECTORED = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_INF | CMD_AERO; - public static final int CMD_ALL = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_INF | CMD_AERO | CMD_AERO_VECTORED; - public static final int CMD_NON_INF = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_AERO | CMD_AERO_VECTORED; + public static final int CMD_GROUND = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_INF | CMD_PROTOMECH; + public static final int CMD_NON_VECTORED = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_INF | CMD_AERO | CMD_PROTOMECH; + public static final int CMD_ALL = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_INF | CMD_AERO | CMD_AERO_VECTORED | CMD_PROTOMECH; + public static final int CMD_NON_INF = CMD_MECH | CMD_TANK | CMD_VTOL | CMD_AERO | CMD_AERO_VECTORED | CMD_PROTOMECH; private boolean isUnJammingRAC; private boolean isUsingChaff; @@ -179,6 +180,8 @@ public enum MoveCommand implements PhaseCommand { MOVE_LONGEST_WALK("MoveLongestWalk", CMD_NONE), // Traitor MOVE_TRAITOR("Traitor", CMD_NONE), + MOVE_PICKUP_CARGO("movePickupCargo", CMD_MECH | CMD_PROTOMECH), + MOVE_DROP_CARGO("moveDropCargo", CMD_MECH | CMD_PROTOMECH), MOVE_MORE("MoveMore", CMD_NONE); /** @@ -566,7 +569,9 @@ protected ArrayList getButtonList() { } else if ((ce instanceof Mech) && ((Mech) ce).hasTracks()) { flag = CMD_MECH | CMD_CONVERTER; } else if ((ce instanceof Protomech) && ce.getMovementMode().isWiGE()) { - flag = CMD_MECH | CMD_AIRMECH; + flag = CMD_PROTOMECH | CMD_MECH | CMD_AIRMECH; + } else if (ce instanceof Protomech) { + flag = CMD_PROTOMECH; } } return getButtonList(flag); @@ -944,6 +949,11 @@ private void addStepToMovePath(MoveStepType moveStep, int recover, int mineToLay cmd.addStep(moveStep, recover, mineToLay); updateMove(); } + + private void addStepToMovePath(MoveStepType moveStep, Map additionalIntData) { + cmd.addStep(moveStep, additionalIntData); + updateMove(); + } private void updateMove() { updateMove(true); @@ -981,6 +991,11 @@ private void updateAeroButtons() { */ @Override protected void updateDonePanel() { + // we don't need to be doing all this stuff if we're not showing this + if (!getClientgui().getClient().getGame().getPhase().isMovement()) { + return; + } + if (cmd == null || cmd.length() == 0) { updateDonePanelButtons(Messages.getString("MovementDisplay.Move"), Messages.getString("MovementDisplay.Skip"), false, null); return; @@ -1118,6 +1133,7 @@ private synchronized void endMyTurn() { clientgui.getBoardView().clearMovementData(); clientgui.clearFieldOfFire(); clientgui.clearTemporarySprites(); + cmd = null; } /** @@ -1188,6 +1204,8 @@ private void disableButtons() { setManeuverEnabled(false); setStrafeEnabled(false); setBombEnabled(false); + setPickupCargoEnabled(false); + setDropCargoEnabled(false); getBtn(MoveCommand.MOVE_CLIMB_MODE).setEnabled(false); getBtn(MoveCommand.MOVE_DIG_IN).setEnabled(false); @@ -2799,8 +2817,38 @@ private synchronized void updateLoadButtons() { updateUnloadButton(); updateTowingButtons(); updateMountButton(); - } - + updatePickupCargoButton(); + updateDropCargoButton(); + } + + /** Updates the status of the "pickup cargo" button */ + private void updatePickupCargoButton() { + final Entity ce = ce(); + // there has to be an entity, objects are on the ground, + // the entity can pick them up + if ((ce == null) || (game().getGroundObjects(finalPosition(), ce).size() <= 0) || + ((cmd.getLastStep() != null) && + (cmd.getLastStep().getType() == MoveStepType.PICKUP_CARGO))) { + setPickupCargoEnabled(false); + return; + } + + setPickupCargoEnabled(true); + } + + /** Updates the status of the "drop cargo" button */ + private void updateDropCargoButton() { + final Entity ce = ce(); + // there has to be an entity, objects are on the ground, + // the entity can pick them up + if ((ce == null) || ce.getCarriedObjects().size() == 0) { + setDropCargoEnabled(false); + return; + } + + setDropCargoEnabled(true); + } + /** Updates the status of the Load button. */ private void updateLoadButton() { final Entity ce = ce(); @@ -4874,7 +4922,39 @@ public synchronized void actionPerformed(ActionEvent ev) { addStepToMovePath(MoveStepType.UNLOAD, other); } } // else - Player canceled the unload. - } else if (actionCmd.equals(MoveCommand.MOVE_RAISE_ELEVATION.getCmd())) { + } else if (actionCmd.equals(MoveCommand.MOVE_PICKUP_CARGO.getCmd())) { + processPickupCargoCommand(); + } else if (actionCmd.equals(MoveCommand.MOVE_DROP_CARGO.getCmd())) { + var options = ce().getDistinctCarriedObjects(); + + if (options.size() == 1) { + addStepToMovePath(MoveStepType.DROP_CARGO); + updateDonePanel(); + } else if (options.size() > 1) { + // reverse lookup: location name to location ID - we're going to wind up with a name chosen + // but need to send the ID in the move path. + Map locationMap = new HashMap<>(); + + for (int location : ce().getCarriedObjects().keySet()) { + locationMap.put(ce().getLocationName(location), location); + } + + // Dialog for choosing which object to pick up + String title = "Choose Cargo to Drop"; + String body = "Choose the cargo to drop:"; + String option = (String) JOptionPane.showInputDialog(clientgui.getFrame(), + body, title, JOptionPane.QUESTION_MESSAGE, null, + locationMap.keySet().toArray(), locationMap.keySet().toArray()[0]); + + // Verify that we have a valid option... + if (option != null) { + int location = locationMap.get(option); + addStepToMovePath(MoveStepType.DROP_CARGO, location); + updateDonePanel(); + } + } + } + if (actionCmd.equals(MoveCommand.MOVE_RAISE_ELEVATION.getCmd())) { addStepToMovePath(MoveStepType.UP); } else if (actionCmd.equals(MoveCommand.MOVE_LOWER_ELEVATION.getCmd())) { if (ce.isAero()) { @@ -5234,6 +5314,88 @@ public synchronized void actionPerformed(ActionEvent ev) { } } + /** + * Worker function containing the "pickup cargo" command. + */ + private void processPickupCargoCommand() { + var options = game().getGroundObjects(finalPosition()); + var displayedOptions = game().getGroundObjects(finalPosition(), ce()); + + // if there's only one thing to pick up, just pick it up. + // regardless of how many objects we are picking up, + // we may have to choose the location with which to pick it up + if (displayedOptions.size() == 1) { + Integer pickupLocation = getPickupLocation(displayedOptions.get(0)); + + if (pickupLocation != null) { + Map data = new HashMap<>(); + // we pick the only eligible object out of all the objects on the ground + data.put(MoveStep.CARGO_PICKUP_KEY, options.indexOf(displayedOptions.get(0))); + data.put(MoveStep.CARGO_LOCATION_KEY, pickupLocation); + + addStepToMovePath(MoveStepType.PICKUP_CARGO, data); + updateDonePanel(); + } + } else if (displayedOptions.size() > 1) { + // Dialog for choosing which object to pick up + String title = "Choose Cargo to Pick Up"; + String body = "Choose the cargo to pick up:"; + ICarryable option = (ICarryable) JOptionPane.showInputDialog(clientgui.getFrame(), + body, title, JOptionPane.QUESTION_MESSAGE, null, + displayedOptions.toArray(), displayedOptions.get(0)); + + if (option != null) { + Integer pickupLocation = getPickupLocation(option); + + if (pickupLocation != null) { + int cargoIndex = options.indexOf(option); + Map data = new HashMap<>(); + data.put(MoveStep.CARGO_PICKUP_KEY, cargoIndex); + data.put(MoveStep.CARGO_LOCATION_KEY, pickupLocation); + + addStepToMovePath(MoveStepType.PICKUP_CARGO, data); + updateDonePanel(); + } + } + } + } + + /** + * Worker function to chose a limb (or whatever) with which to pick up cargo + */ + private Integer getPickupLocation(ICarryable cargo) { + var validPickupLocations = ce().getValidHalfWeightPickupLocations(cargo); + int pickupLocation = Entity.LOC_NONE; + + // if we need to choose a pickup location, then do so + if (validPickupLocations.size() > 1) { + // reverse lookup: location name to location ID - we're going to wind up with a name chosen + // but need to send the ID in the move path. + Map locationMap = new HashMap<>(); + + for (int location : ce().getValidHalfWeightPickupLocations(cargo)) { + locationMap.put(ce().getLocationName(location), location); + } + + // Dialog for choosing which object to pick up + String title = "Choose Pickup Location"; + String body = "Choose the location with which to pick up cargo:"; + String locationChoice = (String) JOptionPane.showInputDialog(clientgui.getFrame(), + body, title, JOptionPane.QUESTION_MESSAGE, null, + locationMap.keySet().toArray(), locationMap.keySet().toArray()[0]); + + if (locationChoice != null) { + pickupLocation = locationMap.get(locationChoice); + } else { + return null; + } + } else if (validPickupLocations.size() == 1) { + pickupLocation = validPickupLocations.get(0); + } + + return pickupLocation; + } + /** * Add enough MoveStepType.CONVERT_MODE steps to get to the requested mode, or * clear the path if the unit is in the requested mode at the beginning of the turn. @@ -5679,6 +5841,16 @@ private void setBombEnabled(boolean enabled) { getBtn(MoveCommand.MOVE_BOMB).setEnabled(enabled); clientgui.getMenuBar().setEnabled(MoveCommand.MOVE_BOMB.getCmd(), enabled); } + + private void setPickupCargoEnabled(boolean enabled) { + getBtn(MoveCommand.MOVE_PICKUP_CARGO).setEnabled(enabled); + clientgui.getMenuBar().setEnabled(MoveCommand.MOVE_PICKUP_CARGO.getCmd(), enabled); + } + + private void setDropCargoEnabled(boolean enabled) { + getBtn(MoveCommand.MOVE_DROP_CARGO).setEnabled(enabled); + clientgui.getMenuBar().setEnabled(MoveCommand.MOVE_DROP_CARGO.getCmd(), enabled); + } @Override public void removeAllListeners() { diff --git a/megamek/src/megamek/client/ui/swing/boardview/BoardView.java b/megamek/src/megamek/client/ui/swing/boardview/BoardView.java index 5b35b6797fd..bc595b8ae83 100644 --- a/megamek/src/megamek/client/ui/swing/boardview/BoardView.java +++ b/megamek/src/megamek/client/ui/swing/boardview/BoardView.java @@ -378,7 +378,6 @@ public final class BoardView extends AbstractBoardView implements BoardListener, // to specialized lists when created private final TreeSet overTerrainSprites = new TreeSet<>(); private final TreeSet behindTerrainHexSprites = new TreeSet<>(); - private final TreeSet hexSprites = new TreeSet<>(); /** * Construct a new board view for the specified game @@ -906,9 +905,6 @@ public void draw(Graphics g) { drawSprites(g, flyOverSprites); } - // draw onscreen entities - drawSprites(g, entitySprites); - // draw moving onscreen entities drawSprites(g, movingEntitySprites); @@ -1626,9 +1622,6 @@ public BufferedImage getEntireBoardImage(boolean ignoreUnits, boolean useBaseZoo drawSprites(boardGraph, flyOverSprites); } - // draw onscreen entities - drawSprites(boardGraph, entitySprites); - // draw moving onscreen entities drawSprites(boardGraph, movingEntitySprites); @@ -1723,7 +1716,6 @@ private void drawHexes(Graphics g, Rectangle view, boolean saveBoardImage) { if (GUIP.getShowWrecks()) { drawIsometricWreckSpritesForHex(c, g, isometricWreckSprites); } - drawIsometricSpritesForHex(c, g, isometricSprites); } } } @@ -2622,6 +2614,7 @@ public void redrawMovingEntity(Entity entity, Coords position, int facing, int e // Remove sprite for Entity, so it's not displayed while moving if (sprite != null) { + removeSprite(sprite); newSprites = new PriorityQueue<>(entitySprites); newSpriteIds = new HashMap<>(entitySpriteIds); @@ -2633,6 +2626,7 @@ public void redrawMovingEntity(Entity entity, Coords position, int facing, int e } // Remove iso sprite for Entity, so it's not displayed while moving if (isoSprite != null) { + removeSprite(isoSprite); isoSprites = new PriorityQueue<>(isometricSprites); newIsoSpriteIds = new HashMap<>(isometricSpriteIds); @@ -2814,10 +2808,16 @@ public void redrawEntity(Entity entity, Entity oldEntity) { } // Update Sprite state with new collections + removeSprites(entitySprites); + removeSprites(isometricSprites); entitySprites = newSprites; entitySpriteIds = newSpriteIds; isometricSprites = isoSprites; isometricSpriteIds = newIsoSpriteIds; + addSprites(entitySprites); + if (drawIsometric) { + addSprites(isometricSprites); + } // Remove C3 sprites for (Iterator i = c3Sprites.iterator(); i.hasNext(); ) { @@ -2944,12 +2944,21 @@ void redrawAllEntities() { } } + removeSprites(entitySprites); + removeSprites(isometricSprites); + entitySprites = newSprites; entitySpriteIds = newSpriteIds; isometricSprites = newIsometricSprites; isometricSpriteIds = newIsoSpriteIds; + addSprites(entitySprites); + if (drawIsometric) { + addSprites(isometricSprites); + } + + wreckSprites = newWrecks; isometricWreckSprites = newIsometricWrecks; @@ -4297,7 +4306,6 @@ public void clearSprites() { overTerrainSprites.clear(); behindTerrainHexSprites.clear(); - hexSprites.clear(); super.clearSprites(); } @@ -4869,7 +4877,6 @@ private Image scale(Image img, int width, int height) { public boolean toggleIsometric() { drawIsometric = !drawIsometric; allSprites.forEach(Sprite::prepare); - hexSprites.forEach(HexSprite::updateBounds); clearHexImageCache(); updateBoard(); @@ -4885,6 +4892,10 @@ public void updateEntityLabels() { for (EntitySprite eS : entitySprites) { eS.prepare(); } + + for (IsometricSprite eS : isometricSprites) { + eS.prepare(); + } boardPanel.repaint(); } @@ -5097,10 +5108,6 @@ public void addSprites(Collection sprites) { .map(s -> (HexSprite) s) .filter(HexSprite::isBehindTerrain) .forEach(behindTerrainHexSprites::add); - sprites.stream() - .filter(s -> s instanceof HexSprite) - .map(s -> (HexSprite) s) - .forEach(hexSprites::add); } @Override @@ -5108,6 +5115,5 @@ public void removeSprites(Collection sprites) { super.removeSprites(sprites); overTerrainSprites.removeAll(sprites); behindTerrainHexSprites.removeAll(sprites); - hexSprites.removeAll(sprites); } } \ No newline at end of file diff --git a/megamek/src/megamek/client/ui/swing/boardview/GroundObjectSprite.java b/megamek/src/megamek/client/ui/swing/boardview/GroundObjectSprite.java new file mode 100644 index 00000000000..a5e1947558b --- /dev/null +++ b/megamek/src/megamek/client/ui/swing/boardview/GroundObjectSprite.java @@ -0,0 +1,70 @@ +/* + * Copyright (c) 2023 - The MegaMek Team. All Rights Reserved. + * + * This file is part of MegaMek. + * + * MegaMek 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 3 of the License, or + * (at your option) any later version. + * + * MegaMek 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 MegaMek. If not, see . + */ +package megamek.client.ui.swing.boardview; + +import java.awt.Dimension; +import java.awt.Graphics2D; +import java.awt.Image; +import java.awt.Rectangle; + +import megamek.client.ui.swing.util.UIUtil; +import megamek.common.Configuration; +import megamek.common.Coords; +import megamek.common.util.ImageUtil; +import megamek.common.util.fileUtils.MegaMekFile; + +/** + * Represents cargo that can be picked up from the ground. + */ +public class GroundObjectSprite extends HexSprite { + private static final String FILENAME_CARGO_IMAGE = "cargo.png"; + private static final Image CARGO_IMAGE; + + static { + CARGO_IMAGE = ImageUtil.loadImageFromFile( + new MegaMekFile(Configuration.miscImagesDir(), FILENAME_CARGO_IMAGE).toString()); + } + + /** + * @param boardView1 - parent BoardView object this sprite will be displayed on. + * @param loc - Hex location coordinates of building or bridge where warning will be visible. + */ + public GroundObjectSprite(BoardView boardView1, Coords loc) { + super(boardView1, loc); + image = CARGO_IMAGE; + } + + @Override + public Rectangle getBounds() { + Dimension dim = new Dimension(bv.hex_size.width, bv.hex_size.height); + bounds = new Rectangle(dim); + bounds.setLocation(bv.getHexLocation(loc)); + return bounds; + } + + @Override + public void prepare() { + updateBounds(); + image = createNewHexImage(); + Graphics2D graph = (Graphics2D) image.getGraphics(); + UIUtil.setHighQualityRendering(graph); + graph.scale(bv.scale, bv.scale); + graph.drawImage(CARGO_IMAGE, 0, 0, null); + } +} \ No newline at end of file diff --git a/megamek/src/megamek/client/ui/swing/boardview/GroundObjectSpriteHandler.java b/megamek/src/megamek/client/ui/swing/boardview/GroundObjectSpriteHandler.java new file mode 100644 index 00000000000..1babead780d --- /dev/null +++ b/megamek/src/megamek/client/ui/swing/boardview/GroundObjectSpriteHandler.java @@ -0,0 +1,76 @@ +/* + * Copyright (c) 2024 - The MegaMek Team. All Rights Reserved. + * + * This file is part of MegaMek. + * + * MegaMek 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 3 of the License, or + * (at your option) any later version. + * + * MegaMek 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 MegaMek. If not, see . + */ +package megamek.client.ui.swing.boardview; + +import java.util.List; +import java.util.Map; + +import megamek.common.Coords; +import megamek.common.Game; +import megamek.common.ICarryable; +import megamek.common.event.GameBoardChangeEvent; + +public class GroundObjectSpriteHandler extends BoardViewSpriteHandler { + + // Cache the ground object list as it does not change very often + private Map> currentGroundObjectList; + private final Game game; + + public GroundObjectSpriteHandler(BoardView boardView, Game game) { + super(boardView); + this.game = game; + } + + public void setGroundObjectSprites(Map> objectCoordList) { + clear(); + currentGroundObjectList = objectCoordList; + if (currentGroundObjectList != null) { + for (Coords coords : currentGroundObjectList.keySet()) { + for (ICarryable groundObject : currentGroundObjectList.get(coords)) { + GroundObjectSprite gos = new GroundObjectSprite(boardView, coords); + currentSprites.add(gos); + } + } + } + + boardView.addSprites(currentSprites); + } + + @Override + public void clear() { + super.clear(); + currentGroundObjectList = null; + } + + @Override + public void initialize() { + game.addGameListener(this); + } + + @Override + public void dispose() { + clear(); + game.removeGameListener(this); + } + + @Override + public void gameBoardChanged(GameBoardChangeEvent e) { + setGroundObjectSprites(game.getGroundObjects()); + } +} diff --git a/megamek/src/megamek/client/ui/swing/boardview/IsometricSprite.java b/megamek/src/megamek/client/ui/swing/boardview/IsometricSprite.java index 510f7e27828..6f066478acb 100644 --- a/megamek/src/megamek/client/ui/swing/boardview/IsometricSprite.java +++ b/megamek/src/megamek/client/ui/swing/boardview/IsometricSprite.java @@ -16,7 +16,7 @@ /** * Sprite used for isometric rendering to render an entity partially hidden behind a hill. */ -class IsometricSprite extends Sprite { +class IsometricSprite extends HexSprite { Entity entity; private Image radarBlipImage; @@ -26,7 +26,7 @@ class IsometricSprite extends Sprite { private static final GUIPreferences GUIP = GUIPreferences.getInstance(); public IsometricSprite(BoardView boardView1, Entity entity, int secondaryPos, Image radarBlipImage) { - super(boardView1); + super(boardView1, secondaryPos == -1 ? entity.getPosition() : entity.getSecondaryPositions().get(secondaryPos)); this.entity = entity; this.radarBlipImage = radarBlipImage; this.secondaryPos = secondaryPos; @@ -176,6 +176,7 @@ public void drawImmobileElements(Graphics graph, int x, int y, ImageObserver obs @Override public void prepare() { + updateBounds(); // create image for buffer GraphicsConfiguration config = GraphicsEnvironment .getLocalGraphicsEnvironment().getDefaultScreenDevice() @@ -237,6 +238,6 @@ private boolean onlyDetectedBySensors() { @Override protected int getSpritePriority() { - return entity.getSpriteDrawPriority(); + return entity.getSpriteDrawPriority() + 10; } } diff --git a/megamek/src/megamek/client/ui/swing/boardview/Sprite.java b/megamek/src/megamek/client/ui/swing/boardview/Sprite.java index b7a5558f9b6..375a2fb316c 100644 --- a/megamek/src/megamek/client/ui/swing/boardview/Sprite.java +++ b/megamek/src/megamek/client/ui/swing/boardview/Sprite.java @@ -162,4 +162,9 @@ public int compareTo(Sprite o) { public boolean equals(Object obj) { return super.equals(obj); } + + @Override + public String toString() { + return "[" + getClass().getSimpleName() + "] Prio: " + getSpritePriority() + ((image == null) ? "; no image" : ""); + } } \ No newline at end of file diff --git a/megamek/src/megamek/client/ui/swing/boardview/StepSprite.java b/megamek/src/megamek/client/ui/swing/boardview/StepSprite.java index f184fd530da..360185d1ee0 100644 --- a/megamek/src/megamek/client/ui/swing/boardview/StepSprite.java +++ b/megamek/src/megamek/client/ui/swing/boardview/StepSprite.java @@ -205,6 +205,14 @@ public void prepare() { String load = Messages.getString("BoardView1.Load"); drawAnnouncement(g2D, load, step, col); break; + case PICKUP_CARGO: + String pickup = Messages.getString("MovementDisplay.movePickupCargo"); + drawAnnouncement(g2D, pickup, step, col); + break; + case DROP_CARGO: + String dropCargo = Messages.getString("MovementDisplay.moveDropCargo"); + drawAnnouncement(g2D, dropCargo, step, col); + break; case TOW: String tow = Messages.getString("BoardView1.Tow"); drawAnnouncement(g2D, tow, step, col); diff --git a/megamek/src/megamek/client/ui/swing/lobby/PlayerSettingsDialog.java b/megamek/src/megamek/client/ui/swing/lobby/PlayerSettingsDialog.java index 3224ca2de62..d324482f2d1 100644 --- a/megamek/src/megamek/client/ui/swing/lobby/PlayerSettingsDialog.java +++ b/megamek/src/megamek/client/ui/swing/lobby/PlayerSettingsDialog.java @@ -45,19 +45,29 @@ import javax.swing.*; import javax.swing.border.EmptyBorder; +import javax.swing.event.DocumentEvent; +import javax.swing.event.DocumentListener; +import javax.swing.event.UndoableEditEvent; +import javax.swing.event.UndoableEditListener; import javax.swing.filechooser.FileNameExtensionFilter; import javax.swing.text.DefaultFormatterFactory; import javax.swing.text.NumberFormatter; + import java.awt.*; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; +import java.awt.event.KeyEvent; +import java.awt.event.KeyListener; +import java.math.RoundingMode; import java.nio.file.Paths; +import java.text.DecimalFormat; import java.text.NumberFormat; import java.util.ArrayList; import java.util.Comparator; import java.util.List; import static megamek.client.ui.Messages.getString; + import static megamek.client.ui.swing.lobby.LobbyMekPopupActions.resetBombChoices; import static megamek.client.ui.swing.lobby.LobbyUtility.isValidStartPos; import static megamek.client.ui.swing.util.UIUtil.*; @@ -71,6 +81,10 @@ */ public class PlayerSettingsDialog extends AbstractButtonDialog { + private static final String CMD_ADD_GROUND_OBJECT = "CMD_ADD_GROUND_OBJECT"; + private static final String CMD_REMOVE_GROUND_OBJECT = "CMD_REMOVE_GROUND_OBJECT_%d"; + private static final String CMD_REMOVE_GROUND_OBJECT_PREFIX = "CMD_REMOVE_GROUND_OBJECT_"; + public PlayerSettingsDialog(ClientGUI cg, Client cl, BoardView bv) { super(cg.getFrame(), "PlayerSettingsDialog", "PlayerSettingsDialog.title"); client = cl; @@ -81,8 +95,36 @@ public PlayerSettingsDialog(ClientGUI cg, Client cl, BoardView bv) { currentPlayerStartPos -= 10; } + NumberFormat numFormat = NumberFormat.getIntegerInstance(); + numFormat.setGroupingUsed(false); + + NumberFormatter numFormatter = new NumberFormatter(numFormat); numFormatter.setMinimum(0); numFormatter.setCommitsOnValidEdit(true); + + DefaultFormatterFactory formatterFactory = new DefaultFormatterFactory(numFormatter); + + txtOffset = new JFormattedTextField(formatterFactory, 0); + txtWidth = new JFormattedTextField(formatterFactory, 3); + + DecimalFormat tonnageFormat = new DecimalFormat(); + tonnageFormat.setGroupingUsed(false); + tonnageFormat.setRoundingMode(RoundingMode.UNNECESSARY); + + txtGroundObjectTonnage = new JFormattedTextField(tonnageFormat); + txtGroundObjectTonnage.setText("0"); + + txtGroundObjectName = new JTextField(); + txtGroundObjectName.setColumns(20); + // if it's longer than 20 characters, undo the edit + txtGroundObjectName.getDocument().addUndoableEditListener(new UndoableEditListener( ) { + @Override + public void undoableEditHappened(UndoableEditEvent e) { + if (txtGroundObjectName.getText().length() > 20 && e.getEdit().canUndo()) { + e.getEdit().undo(); + } + } + }); initialize(); } @@ -92,7 +134,7 @@ public PlayerSettingsDialog(ClientGUI cg, Client cl, BoardView bv) { @Override public Component getListCellRendererComponent(JList list, Object value, int index, - boolean isSelected, boolean cellHasFocus) { + boolean isSelected, boolean cellHasFocus) { if (value == null) { setText("General"); } else { @@ -196,15 +238,12 @@ public String getEmail() { private Player player; // Initiative Section - private final JLabel labInit = new TipLabel(Messages.getString("PlayerSettingsDialog.initMod"), - SwingConstants.RIGHT); + private final JLabel labInit = new TipLabel(Messages.getString("PlayerSettingsDialog.initMod"), SwingConstants.RIGHT); private final TipTextField fldInit = new TipTextField(3); // Mines Section - private final JLabel labConventional = new JLabel(getString("PlayerSettingsDialog.labConventional"), - SwingConstants.RIGHT); - private final JLabel labVibrabomb = new JLabel(getString("PlayerSettingsDialog.labVibrabomb"), - SwingConstants.RIGHT); + private final JLabel labConventional = new JLabel(getString("PlayerSettingsDialog.labConventional"), SwingConstants.RIGHT); + private final JLabel labVibrabomb = new JLabel(getString("PlayerSettingsDialog.labVibrabomb"), SwingConstants.RIGHT); private final JLabel labActive = new JLabel(getString("PlayerSettingsDialog.labActive"), SwingConstants.RIGHT); private final JLabel labInferno = new JLabel(getString("PlayerSettingsDialog.labInferno"), SwingConstants.RIGHT); private final JTextField fldConventional = new JTextField(3); @@ -222,17 +261,20 @@ public String getEmail() { // Deployment Section private final JPanel panStartButtons = new JPanel(); private final TipButton[] butStartPos = new TipButton[11]; - // this might seem like kind of a dumb way to declare it, but - // JFormattedTextField doesn't have an overload that - // takes both a number formatter and a default value. - private final NumberFormatter numFormatter = new NumberFormatter(NumberFormat.getIntegerInstance()); - private final DefaultFormatterFactory formatterFactory = new DefaultFormatterFactory(numFormatter); - private final JFormattedTextField txtOffset = new JFormattedTextField(formatterFactory, 0); - private final JFormattedTextField txtWidth = new JFormattedTextField(formatterFactory, 3); + + private final JFormattedTextField txtOffset; + private final JFormattedTextField txtWidth; private JSpinner spinStartingAnyNWx; private JSpinner spinStartingAnyNWy; private JSpinner spinStartingAnySEx; private JSpinner spinStartingAnySEy; + + // ground object config section + private Content groundSectionContent = new Content(new GridLayout(2, 3)); + private final JTextField txtGroundObjectName; + private final JFormattedTextField txtGroundObjectTonnage; + private final JCheckBox chkGroundObjectInvulnerable = new JCheckBox(); + private List> groundSectionComponents = new ArrayList<>(); // Bot Settings Section private final JButton butBotSettings = new JButton(Messages.getString("PlayerSettingsDialog.botSettings")); @@ -273,6 +315,7 @@ protected Container createCenterPane() { if (client.getGame().getOptions().booleanOption(OptionsConstants.ADVANCED_MINEFIELDS)) { mainPanel.add(mineSection()); } + mainPanel.add(groundObjectConfigSection()); mainPanel.add(skillsSection()); if (!(client instanceof BotClient)) { mainPanel.add(emailSection()); @@ -334,6 +377,131 @@ private JPanel autoConfigSection() { butRestoreMT.setEnabled(false); return result; } + + private JPanel groundObjectConfigSection() { + JPanel result = new OptionPanel("PlayerSettingsDialog.header.GroundObjects"); + result.setToolTipText("Define carryable objects that can be placed prior to unit deployment"); + groundSectionContent = new Content(new GridBagLayout()); + GridBagConstraints gbc = new GridBagConstraints(); + + gbc.gridx = 0; + gbc.gridy = 0; + JLabel lblName = new JLabel("Name"); + groundSectionContent.add(lblName, gbc); + + gbc.gridx = 1; + JLabel lblTonnage = new JLabel("Tonnage"); + groundSectionContent.add(lblTonnage, gbc); + + gbc.gridx = 2; + JLabel lblInvulnerable = new JLabel("Invulnerable"); + groundSectionContent.add(lblInvulnerable); + + gbc.gridy = 1; + gbc.gridx = 0; + groundSectionContent.add(txtGroundObjectName, gbc); + + gbc.gridx = 1; + groundSectionContent.add(txtGroundObjectTonnage, gbc); + + gbc.gridx = 2; + groundSectionContent.add(chkGroundObjectInvulnerable, gbc); + + gbc.gridx = 3; + JButton btnAdd = new JButton("Add"); + btnAdd.setActionCommand(CMD_ADD_GROUND_OBJECT); + btnAdd.addActionListener(listener); + groundSectionContent.add(btnAdd, gbc); + + for (ICarryable groundObject : player.getGroundObjectsToPlace()) { + addGroundObjectToUI(groundObject); + } + + result.add(groundSectionContent); + return result; + } + + /** + * Worker function that adds the given ground object to the UI + */ + private void addGroundObjectToUI(ICarryable groundObject) { + GridBagConstraints gbc = new GridBagConstraints(); + gbc.gridy = groundSectionComponents.size() + 2; // there's always two extra rows - header + text fields + gbc.gridx = 0; + + JLabel nameLabel = new JLabel(groundObject.generalName()); + groundSectionContent.add(nameLabel, gbc); + List row = new ArrayList<>(); + row.add(nameLabel); + + gbc.gridx = 1; + JLabel tonnageLabel = new JLabel(Double.toString(groundObject.getTonnage())); + groundSectionContent.add(tonnageLabel, gbc); + row.add(tonnageLabel); + + gbc.gridx = 2; + JLabel flagLabel = new JLabel(groundObject.isInvulnerable() ? "Yes" : "No"); + groundSectionContent.add(flagLabel, gbc); + row.add(flagLabel); + + gbc.gridx = 3; + JButton btnRemove = new JButton("Remove"); + btnRemove.setActionCommand(String.format(CMD_REMOVE_GROUND_OBJECT, player.getGroundObjectsToPlace().size() - 1)); + btnRemove.addActionListener(listener); + groundSectionContent.add(btnRemove, gbc); + row.add(btnRemove); + groundSectionComponents.add(row); + validate(); + } + + /** + * Worker function that uses the current state of the ground object inputs to + * add a new ground object to the backing player and the UI + */ + private void addGroundObject() { + Briefcase briefcase = new Briefcase(); + briefcase.setName(txtGroundObjectName.getText()); + + Double tonnage = 0.0; + + try { + tonnage = Double.parseDouble(txtGroundObjectTonnage.getText()); + + // don't allow negative tonnage as we do not have anti-gravity technology + if (tonnage < 0) { + tonnage = 0.0; + } + } catch (Exception ignored) { + + } + + briefcase.setTonnage(tonnage); + briefcase.setInvulnerable(chkGroundObjectInvulnerable.isSelected()); + player.getGroundObjectsToPlace().add(briefcase); + + addGroundObjectToUI(briefcase); + } + + /** + * Worker function that removes the chosen ground object from the backing player and the UI + */ + private void removeGroundObject(String command) { + int index = Integer.parseInt(command.substring(CMD_REMOVE_GROUND_OBJECT_PREFIX.length())); + player.getGroundObjectsToPlace().remove(index); + for(Component component : groundSectionComponents.get(index)) { + groundSectionContent.remove(component); + } + groundSectionComponents.remove(index); + + // kind of a hack, but I'm being lazy - re-index all the CMD_REMOVE_GROUND_OBJECT commands beyond + // the one that just removed, so they're not pointing to higher indexes than they need to + for (int componentIndex = index; componentIndex < groundSectionComponents.size(); componentIndex++) { + ((JButton) groundSectionComponents.get(index).get(2)) + .setActionCommand(String.format(CMD_REMOVE_GROUND_OBJECT, componentIndex)); + } + + validate(); + } private JPanel botSection() { JPanel result = new OptionPanel("PlayerSettingsDialog.header.botPlayer"); @@ -390,6 +558,7 @@ private JPanel deploymentParametersPanel() { return result; } + private void useRuler() { if (bv.getRulerStart() != null && bv.getRulerEnd() != null) { int x = Math.min(bv.getRulerStart().getX(), bv.getRulerEnd().getX()); @@ -415,13 +584,12 @@ private void apply() { final GameOptions gOpts = clientgui.getClient().getGame().getOptions(); - // If the gameoption set_arty_player_homeedge is set, adjust the player's - // offboard + // If the gameoption set_arty_player_homeedge is set, adjust the player's offboard // arty units to be behind the newly selected home edge. OffBoardDirection direction = OffBoardDirection.translateStartPosition(getStartPos()); if (direction != OffBoardDirection.NONE && gOpts.booleanOption(OptionsConstants.BASE_SET_ARTY_PLAYER_HOMEEDGE)) { - for (Entity entity : client.getGame().getPlayerEntities(client.getLocalPlayer(), false)) { + for (Entity entity: client.getGame().getPlayerEntities(client.getLocalPlayer(), false)) { if (entity.getOffBoardDirection() != OffBoardDirection.NONE) { entity.setOffBoard(entity.getOffBoardDistance(), direction); } @@ -433,6 +601,7 @@ private void apply() { team.setFaction(faction); if ((clientgui != null) && (clientgui.chatlounge != null)) { ArrayList updateEntities = clientgui.getClient().getGame().getPlayerEntities(player, false); + if (null != munitionTree && null != rp) { rp.friendlyFaction = faction; rp.binFillPercent = (rp.isPirate) ? TeamLoadoutGenerator.UNSET_FILL_RATIO : 1.0f; @@ -526,7 +695,7 @@ private void setupValues() { int bh = ms.getBoardHeight() * ms.getMapHeight(); int bw = ms.getBoardWidth() * ms.getMapWidth(); - SpinnerNumberModel mStartingAnyNWx = new SpinnerNumberModel(0, 0, bw, 1); + SpinnerNumberModel mStartingAnyNWx = new SpinnerNumberModel(0, 0,bw, 1); spinStartingAnyNWx = new JSpinner(mStartingAnyNWx); SpinnerNumberModel mStartingAnyNWy = new SpinnerNumberModel(0, 0, bh, 1); spinStartingAnyNWy = new JSpinner(mStartingAnyNWy); @@ -683,21 +852,26 @@ public void actionPerformed(ActionEvent e) { // Bot settings button if (butBotSettings.equals(e.getSource()) && client instanceof Princess) { BehaviorSettings behavior = ((Princess) client).getBehaviorSettings(); - var bcd = new BotConfigDialog(clientgui.getFrame(), client.getLocalPlayer().getName(), behavior, - clientgui); + var bcd = new BotConfigDialog(clientgui.getFrame(), client.getLocalPlayer().getName(), behavior, clientgui); bcd.setVisible(true); if (bcd.getResult() == DialogResult.CONFIRMED) { ((Princess) client).setBehaviorSettings(bcd.getBehaviorSettings()); } } + + if (e.getActionCommand().equals(CMD_ADD_GROUND_OBJECT)) { + addGroundObject(); + } + + if (e.getActionCommand().contains(CMD_REMOVE_GROUND_OBJECT_PREFIX)) { + removeGroundObject(e.getActionCommand()); + } } }; /** - * Let user select an ADF file (Autoconfiguration Definition File) from which to - * load munition loadout + * Let user select an ADF file (Autoconfiguration Definition File) from which to load munition loadout * imperatives, which can then be applied to selected units. - * * @return */ private MunitionTree loadLoadout() { @@ -712,7 +886,7 @@ private MunitionTree loadLoadout() { int returnVal = fc.showOpenDialog(this); if ((returnVal != JFileChooser.APPROVE_OPTION) || (fc.getSelectedFile() == null)) { - // No file selected? No loadout! + // No file selected? No loadout! return null; } @@ -724,7 +898,6 @@ private MunitionTree loadLoadout() { } private void saveLoadout(MunitionTree source) { - //ignoreHotKeys = true; JFileChooser fc = new JFileChooser(Paths.get(MMConstants.USER_LOADOUTS_DIR).toAbsolutePath().toString()); FileNameExtensionFilter adfFilter = new FileNameExtensionFilter( "adf files (*.adf)", "adf"); @@ -735,7 +908,7 @@ private void saveLoadout(MunitionTree source) { int returnVal = fc.showSaveDialog(this); if ((returnVal != JFileChooser.APPROVE_OPTION) || (fc.getSelectedFile() == null)) { - // No file selected? No loadout! + // No file selected? No loadout! return; } if (fc.getSelectedFile() != null) { @@ -758,9 +931,8 @@ private int parseField(JTextField field) { return 0; } } - private void adaptToGUIScale() { - UIUtil.adjustDialog(this, UIUtil.FONT_SCALE1); + UIUtil.adjustDialog(this, UIUtil.FONT_SCALE1); } public FactionRecord getFaction() { @@ -773,8 +945,7 @@ public String getFactionCode() { public FactionRecord getFactionFromCode(String code, int year) { for (FactionRecord fRec : RATGenerator.getInstance().getFactionList()) { - if ((!fRec.isMinor()) && !fRec.getKey().contains(".") && fRec.isActiveInYear(year) - && fRec.getKey().equals(code)) { + if ((!fRec.isMinor()) && !fRec.getKey().contains(".") && fRec.isActiveInYear(year) && fRec.getKey().equals(code)) { return fRec; } } diff --git a/megamek/src/megamek/client/ui/swing/tooltip/HexTooltip.java b/megamek/src/megamek/client/ui/swing/tooltip/HexTooltip.java index 34a7d9bbd02..b1ea159ebc1 100644 --- a/megamek/src/megamek/client/ui/swing/tooltip/HexTooltip.java +++ b/megamek/src/megamek/client/ui/swing/tooltip/HexTooltip.java @@ -159,6 +159,16 @@ public static String getHexTip(Hex mhex, @Nullable Client client, GUIPreferences result.append("
"); } } + + if ((game != null) && game.getGroundObjects(mcoords).size() > 0) { + for (ICarryable groundObject : game.getGroundObjects(mcoords)) { + result.append(" "); + result.append(guiScaledFontHTML(UIUtil.uiWhite())); + result.append(groundObject.specificName()); + result.append(""); + result.append("
"); + } + } return result.toString(); } diff --git a/megamek/src/megamek/client/ui/swing/tooltip/UnitToolTip.java b/megamek/src/megamek/client/ui/swing/tooltip/UnitToolTip.java index 55d9383ed02..71dea7b204b 100644 --- a/megamek/src/megamek/client/ui/swing/tooltip/UnitToolTip.java +++ b/megamek/src/megamek/client/ui/swing/tooltip/UnitToolTip.java @@ -164,6 +164,9 @@ private static StringBuilder getEntityTipTable(Entity entity, Player localPlayer // Carried Units result += carriedUnits(entity); + + // carried cargo + result += carriedCargo(entity); // C3 Info result += c3Info(entity, details); @@ -1843,6 +1846,25 @@ private static StringBuilder carriedUnits(Entity entity) { return new StringBuilder().append(result); } + + private static StringBuilder carriedCargo(Entity entity) { + StringBuilder sb = new StringBuilder(); + List cargoList = entity.getDistinctCarriedObjects(); + + if (!cargoList.isEmpty()) { + sb.append(guiScaledFontHTML()); + sb.append(Messages.getString("MissionRole.cargo")); + sb.append(":
  "); + + for (ICarryable cargo : entity.getDistinctCarriedObjects()) { + sb.append(cargo.toString()); + sb.append("
  "); + } + sb.append(""); + } + + return sb; + } /** Returns the full force chain the entity is in as one text line. */ private static StringBuilder forceEntry(Entity entity, Player localPlayer) { diff --git a/megamek/src/megamek/client/ui/swing/unitDisplay/ExtraPanel.java b/megamek/src/megamek/client/ui/swing/unitDisplay/ExtraPanel.java index 1c41b0d1b3a..5cc7320c226 100644 --- a/megamek/src/megamek/client/ui/swing/unitDisplay/ExtraPanel.java +++ b/megamek/src/megamek/client/ui/swing/unitDisplay/ExtraPanel.java @@ -490,6 +490,12 @@ public void displayMech(Entity en) { carrysR.append(club.getName()); carrysR.append("\n"); } + + // show cargo. + for (ICarryable cargo : en.getDistinctCarriedObjects()) { + carrysR.append(cargo.specificName()); + carrysR.append("\n"); + } // Show searchlight if (en.hasSearchlight()) { diff --git a/megamek/src/megamek/common/BipedMech.java b/megamek/src/megamek/common/BipedMech.java index 2203256a7c0..d80461cb1a8 100644 --- a/megamek/src/megamek/common/BipedMech.java +++ b/megamek/src/megamek/common/BipedMech.java @@ -85,6 +85,68 @@ public boolean canFlipArms() { return canFlip; } + + /** + * Returns true if the entity can pick up ground objects + */ + public boolean canPickupGroundObject() { + return hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) || + hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null); + } + + /** + * The maximum tonnage of ground objects that can be picked up by this unit + */ + public double maxGroundObjectTonnage() { + double percentage = 0.0; + + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) && + !isLocationBad(Mech.LOC_LARM)) { + percentage += 0.05; + } + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null) && + !isLocationBad(Mech.LOC_RARM)) { + percentage += 0.05; + } + + return getWeight() * percentage; + } + + @Override + public List getDefaultPickupLocations() { + List result = new ArrayList<>(); + + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) && + !isLocationBad(Mech.LOC_LARM)) { + result.add(Mech.LOC_LARM); + } + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null) && + !isLocationBad(Mech.LOC_RARM)) { + result.add(Mech.LOC_RARM); + } + + return result; + } + + @Override + public List getValidHalfWeightPickupLocations(ICarryable cargo) { + List result = new ArrayList<>(); + + // if we can pick the object up according to "one handed pick up rules" in TacOps + if (cargo.getTonnage() <= (getWeight() / 20)) { + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) && + !isLocationBad(Mech.LOC_LARM)) { + result.add(Mech.LOC_LARM); + } + + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null) && + !isLocationBad(Mech.LOC_RARM)) { + result.add(Mech.LOC_RARM); + } + } + + return result; + } @Override public int getWalkMP(MPCalculationSetting mpCalculationSetting) { diff --git a/megamek/src/megamek/common/Briefcase.java b/megamek/src/megamek/common/Briefcase.java new file mode 100644 index 00000000000..4a5c33e0c0c --- /dev/null +++ b/megamek/src/megamek/common/Briefcase.java @@ -0,0 +1,100 @@ +/* + * Copyright (c) 2024 - The MegaMek Team. All Rights Reserved. + * + * This file is part of MegaMek. + * + * MegaMek 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 3 of the License, or + * (at your option) any later version. + * + * MegaMek 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 MegaMek. If not, see . + */ + +package megamek.common; + +import java.io.Serializable; + +/** + * Represents a basic carryable object with no additional other properties + */ +public class Briefcase implements ICarryable, Serializable { + /** + * + */ + private static final long serialVersionUID = 8849879320465375457L; + + private double tonnage; + private String name; + private boolean invulnerable; + private int id; + private int ownerId; + + public void damage(double amount) { + tonnage -= amount; + } + + public void setTonnage(double value) { + tonnage = value; + } + + public double getTonnage() { + return tonnage; + } + + public boolean isInvulnerable() { + return invulnerable; + } + + public void setInvulnerable(boolean value) { + invulnerable = value; + } + + public void setName(String value) { + name = value; + } + + public String generalName() { + return name; + } + + public String specificName() { + return name + " (" + tonnage + " tons)"; + } + + @Override + public String toString() { + return specificName(); + } + + @Override + public int getId() { + return id; + } + + @Override + public void setId(int newId) { + this.id = newId; + } + + @Override + public int getOwnerId() { + return ownerId; + } + + @Override + public void setOwnerId(int newOwnerId) { + this.ownerId = newOwnerId; + } + + @Override + public int getStrength() { + return 0; + } +} diff --git a/megamek/src/megamek/common/Entity.java b/megamek/src/megamek/common/Entity.java index a42d4d7b6e4..7aee2b7ed40 100644 --- a/megamek/src/megamek/common/Entity.java +++ b/megamek/src/megamek/common/Entity.java @@ -135,7 +135,7 @@ public abstract class Entity extends TurnOrdered implements Transporter, Targeta public static final int DMG_CRIPPLED = 4; public static final int USE_STRUCTURAL_RATING = -1; - + protected transient Game game; protected int id = Entity.NONE; @@ -835,7 +835,17 @@ public abstract class Entity extends TurnOrdered implements Transporter, Targeta * Primarily used by Princess to speed up TAG utility calculations. */ protected ArrayList incomingGuidedAttacks; + + /** + * Map containing all the objects this entity is carrying as cargo, indexed by location + */ + private Map carriedObjects = new HashMap<>(); + /** + * Round-long flag indicating that this entity has picked up an object this round. + */ + private boolean endOfTurnCargoInteraction; + /** The icon for this unit; This is empty unless the unit file has an embedded icon. */ protected Base64Image icon = new Base64Image(); @@ -877,6 +887,7 @@ public Entity() { initTechAdvancement(); offBoardShotObservers = new HashSet<>(); incomingGuidedAttacks = new ArrayList(); + carriedObjects = new HashMap<>(); } /** @@ -2760,6 +2771,141 @@ public boolean canUnjamRAC() { public boolean canFlipArms() { return false; } + + /** + * Returns true if the entity can pick up ground objects + */ + public boolean canPickupGroundObject() { + return false; + } + + /** + * The maximum tonnage of ground objects that can be picked up by this unit + */ + public double maxGroundObjectTonnage() { + return 0.0; + } + + /** + * Put a ground object into the given location + */ + public void pickupGroundObject(ICarryable carryable, Integer location) { + if (carriedObjects == null) { + carriedObjects = new HashMap<>(); + } + + // "none" means we should just put it wherever it goes by default. + // rules checks are done prior to this, so we just set the data + if (location == null || location == LOC_NONE) { + for (Integer defaultLocation : getDefaultPickupLocations()) { + carriedObjects.put(defaultLocation, carryable); + } + } else { + carriedObjects.put(location, carryable); + } + endOfTurnCargoInteraction = true; + } + + /** + * Remove a specific carried object - useful for when you have the object + * but not its location, or when an object is being carried in multiple locations. + */ + public void dropGroundObject(ICarryable carryable, boolean isUnload) { + // build list of locations to clear out + List locationsToClear = new ArrayList<>(); + + for (Integer location : carriedObjects.keySet()) { + if (carriedObjects.get(location).equals(carryable)) { + locationsToClear.add(location); + } + } + + for (Integer location : locationsToClear) { + carriedObjects.remove(location); + } + + // if it's not an "unload", we're going to leave the "end of turn cargo interaction" flag alone + if (isUnload) { + endOfTurnCargoInteraction = true; + } + } + + /** + * Remove a ground object (cargo) from the given location + */ + public void dropGroundObject(int location) { + carriedObjects.remove(location); + } + + /** + * Convenience method to drop all cargo. + */ + public void dropGroundObjects() { + carriedObjects.clear(); + } + + /** + * Get the object carried in the given location. May return null. + */ + public ICarryable getCarriedObject(int location) { + return carriedObjects.get(location); + } + + public Map getCarriedObjects() { + return carriedObjects; + } + + public void setCarriedObjects(Map value) { + carriedObjects = value; + } + + public List getDistinctCarriedObjects() { + return carriedObjects.values().stream().distinct().toList(); + } + + /** + * A list of the "default" cargo pick up locations for when none is specified + */ + protected List getDefaultPickupLocations() { + return Arrays.asList(LOC_NONE); + } + + /** + * A list of all the locations that the entity can use to pick up cargo following the TacOps + * "one handed" pickup rules + */ + public List getValidHalfWeightPickupLocations(ICarryable cargo) { + return Arrays.asList(LOC_NONE); + } + + /** + * Whether a weapon in a given location can be fired, + * given the entity's currently carried cargo + */ + public boolean canFireWeapon(int location) { + if (getBlockedFiringLocations() == null) { + return true; + } + + // loop through everything we are carrying + // if the weapon location is blocked by the carried object, then we cannot fire the weapon + for (int carriedObjectLocation : getCarriedObjects().keySet()) { + if (getBlockedFiringLocations().containsKey(carriedObjectLocation) && + getBlockedFiringLocations().get(carriedObjectLocation).contains(location)) { + return false; + } + } + + return true; + } + + /** + * Method that returns the mapping between locations which, if cargo is carried, + * block other locations from firing. + */ + protected Map> getBlockedFiringLocations() { + return null; + } /** * Returns this entity's original walking movement points @@ -6538,6 +6684,8 @@ public void newRound(int roundNumber) { setClimbMode(GUIP.getMoveDefaultClimbMode()); + endOfTurnCargoInteraction = false; + setTurnInterrupted(false); } @@ -12204,6 +12352,10 @@ public void setTurnInterrupted(boolean interrupted) { public boolean turnWasInterrupted() { return turnWasInterrupted; } + + public boolean endOfTurnCargoInteraction() { + return endOfTurnCargoInteraction; + } public Vector getSensors() { return sensors; diff --git a/megamek/src/megamek/common/Game.java b/megamek/src/megamek/common/Game.java index c8532495966..d66beae0f20 100644 --- a/megamek/src/megamek/common/Game.java +++ b/megamek/src/megamek/common/Game.java @@ -136,6 +136,11 @@ public final class Game extends AbstractGame implements Serializable, PlanetaryC */ private Map botSettings = new HashMap<>(); + /** + * Piles of carry-able objects, sorted by coordinates + */ + private Map> groundObjects = new HashMap<>(); + /** * Constructor */ @@ -1314,7 +1319,6 @@ public synchronized void reset() { resetPSRs(); resetArtilleryAttacks(); resetAttacks(); - // removeMinefields(); Broken and bad! clearMinefields(); removeArtyAutoHitHexes(); flares.removeAllElements(); @@ -1328,6 +1332,7 @@ public synchronized void reset() { lastEntityId = 0; planetaryConditions = new PlanetaryConditions(); forces = new Forces(this); + groundObjects = new HashMap<>(); } private void removeArtyAutoHitHexes() { @@ -3273,8 +3278,77 @@ public Map getBotSettings() { public void setBotSettings(Map botSettings) { this.botSettings = botSettings; } - - /** + + /** + * Place a carryable object on the ground at the given coordinates + */ + public void placeGroundObject(Coords coords, ICarryable carryable) { + if (!groundObjects.containsKey(coords)) { + groundObjects.put(coords, new ArrayList<>()); + } + + groundObjects.get(coords).add(carryable); + } + + /** + * Remove the given carryable object from the ground at the given coordinates + */ + public void removeGroundObject(Coords coords, ICarryable carryable) { + if (groundObjects.containsKey(coords)) { + groundObjects.get(coords).remove(carryable); + } + } + + /** + * Get a list of all the objects on the ground at the given coordinates + * guaranteed to return non-null, but may return empty list + */ + public List getGroundObjects(Coords coords) { + return groundObjects.containsKey(coords) ? groundObjects.get(coords) : new ArrayList<>(); + } + + /** + * Get a list of all objects on the ground at the given coordinates + * that can be picked up by the given entity + */ + public List getGroundObjects(Coords coords, Entity entity) { + if (!groundObjects.containsKey(coords)) { + return new ArrayList<>(); + } + + // if the entity doesn't have working actuators etc + if (!entity.canPickupGroundObject()) { + return new ArrayList<>(); + } + + double maxTonnage = entity.maxGroundObjectTonnage(); + ArrayList result = new ArrayList<>(); + + for (ICarryable object : groundObjects.get(coords)) { + if (maxTonnage >= object.getTonnage()) { + result.add(object); + } + } + + return result; + } + + /** + * @return Collection of objects on the ground. Best to use getGroundObjects(Coords) + * if looking for objects in specific hex + */ + public Map> getGroundObjects() { + return groundObjects; + } + + /** + * @param groundObjects the groundObjects to set + */ + public void setGroundObjects(Map> groundObjects) { + this.groundObjects = groundObjects; + } + + /** * Cancels a victory */ public void cancelVictory() { diff --git a/megamek/src/megamek/common/HexTarget.java b/megamek/src/megamek/common/HexTarget.java index 1fa780b8e25..d963b8b2a53 100644 --- a/megamek/src/megamek/common/HexTarget.java +++ b/megamek/src/megamek/common/HexTarget.java @@ -29,17 +29,6 @@ public HexTarget(Coords c, int nType) { m_bIgnite = (nType == Targetable.TYPE_HEX_IGNITE); } - /** - * Creates a new HexTarget given a set of coordinates and a type defined in Targetable. - * the board parameter is ignored. - */ - @Deprecated - public HexTarget(Coords c, Board board, int nType) { - m_coords = c; - m_type = nType; - m_bIgnite = (nType == Targetable.TYPE_HEX_IGNITE); - } - @Override public int getTargetType() { return m_type; diff --git a/megamek/src/megamek/common/ICarryable.java b/megamek/src/megamek/common/ICarryable.java new file mode 100644 index 00000000000..6492c84d110 --- /dev/null +++ b/megamek/src/megamek/common/ICarryable.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2024 - The MegaMek Team. All Rights Reserved. + * + * This file is part of MegaMek. + * + * MegaMek 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 3 of the License, or + * (at your option) any later version. + * + * MegaMek 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 MegaMek. If not, see . + */ + +package megamek.common; + +/** + * An interface defining all the required properties of a carryable object. + */ +public interface ICarryable extends InGameObject { + double getTonnage(); + void damage(double amount); + boolean isInvulnerable(); + + default boolean isCarryableObject() { + return true; + } +} \ No newline at end of file diff --git a/megamek/src/megamek/common/Mech.java b/megamek/src/megamek/common/Mech.java index d9425f84968..eb05b004782 100644 --- a/megamek/src/megamek/common/Mech.java +++ b/megamek/src/megamek/common/Mech.java @@ -37,6 +37,7 @@ import java.math.BigInteger; import java.time.LocalDate; import java.util.*; +import static java.util.Map.entry; import java.util.stream.Collectors; /** @@ -183,6 +184,26 @@ public abstract class Mech extends Entity { "Small Command", "Tripod Industrial", "Superheavy Tripod Industrial" }; public static final String FULL_HEAD_EJECT_STRING = "Full Head Ejection System"; + + /** + * Contains a mapping of locations which are blocked when carrying cargo in the "key" location + */ + public static final Map> BLOCKED_FIRING_LOCATIONS; + + static { + BLOCKED_FIRING_LOCATIONS = new HashMap<>(); + BLOCKED_FIRING_LOCATIONS.put(LOC_LARM, new ArrayList<>()); + BLOCKED_FIRING_LOCATIONS.get(LOC_LARM).add(LOC_LARM); + BLOCKED_FIRING_LOCATIONS.get(LOC_LARM).add(LOC_LT); + BLOCKED_FIRING_LOCATIONS.get(LOC_LARM).add(LOC_CT); + BLOCKED_FIRING_LOCATIONS.get(LOC_LARM).add(LOC_RT); + + BLOCKED_FIRING_LOCATIONS.put(LOC_RARM, new ArrayList<>()); + BLOCKED_FIRING_LOCATIONS.get(LOC_RARM).add(LOC_RARM); + BLOCKED_FIRING_LOCATIONS.get(LOC_RARM).add(LOC_LT); + BLOCKED_FIRING_LOCATIONS.get(LOC_RARM).add(LOC_CT); + BLOCKED_FIRING_LOCATIONS.get(LOC_RARM).add(LOC_RT); + } // jump types public static final int JUMP_UNKNOWN = -1; @@ -6454,7 +6475,7 @@ public int getGenericBattleValue() { public boolean getsAutoExternalSearchlight() { return true; } - + public static Map getAllCockpitCodeName() { Map result = new HashMap<>(); @@ -6481,4 +6502,13 @@ public static Map getAllCockpitCodeName() { return result; } + + /** + * Method that returns the mapping between locations which, if cargo is carried, + * block other locations from firing. + */ + @Override + protected Map> getBlockedFiringLocations() { + return BLOCKED_FIRING_LOCATIONS; + } } diff --git a/megamek/src/megamek/common/MovePath.java b/megamek/src/megamek/common/MovePath.java index 0dcc284d871..3ba6f4e6d46 100644 --- a/megamek/src/megamek/common/MovePath.java +++ b/megamek/src/megamek/common/MovePath.java @@ -58,7 +58,7 @@ public enum MoveStepType { CLIMB_MODE_OFF, SWIM, DIG_IN, FORTIFY, SHAKE_OFF_SWARMERS, TAKEOFF, VTAKEOFF, LAND, ACC, DEC, EVADE, SHUTDOWN, STARTUP, SELF_DESTRUCT, ACCN, DECN, ROLL, OFF, RETURN, LAUNCH, THRUST, YAW, CRASH, RECOVER, RAM, HOVER, MANEUVER, LOOP, CAREFUL_STAND, JOIN, DROP, VLAND, MOUNT, UNDOCK, TAKE_COVER, - CONVERT_MODE, BOOTLEGGER, TOW, DISCONNECT, BRACE, CHAFF; + CONVERT_MODE, BOOTLEGGER, TOW, DISCONNECT, BRACE, CHAFF, PICKUP_CARGO, DROP_CARGO; /** * Whether this move step type will result in the unit entering a new hex @@ -206,6 +206,10 @@ public MovePath addStep(final MoveStepType type, final boolean noCost) { return addStep(new MoveStep(this, type, noCost)); } + public MovePath addStep(final MoveStepType type, final Map additionalIntData) { + return addStep(new MoveStep(this, type, additionalIntData)); + } + public MovePath addStep(final MoveStepType type, final boolean noCost, final boolean isManeuver, final int maneuverType) { return addStep(new MoveStep(this, type, noCost, isManeuver, maneuverType)); } @@ -541,6 +545,11 @@ && getMpUsed() > getCachedEntityState().getWalkMP() step.setMovementType(EntityMovementType.MOVE_ILLEGAL); } } + + // if we have a PICKUP, then we can't do anything else after it + if (contains(MoveStepType.PICKUP_CARGO)) { + step.setMovementType(EntityMovementType.MOVE_ILLEGAL); + } } public void compile(final Game g, final Entity en) { @@ -575,6 +584,8 @@ public void compile(final Game g, final Entity en, boolean clip) { step = new MoveStep(this, step.getType(), step.hasNoCost()); } else if (null != step.getMinefield()) { step = new MoveStep(this, step.getType(), step.getMinefield()); + } else if (null != step.getAdditionalData() && step.getAdditionalData().size() > 0) { + step = new MoveStep(this, step.getType(), step.getAdditionalData()); } else { step = new MoveStep(this, step.getType()); } diff --git a/megamek/src/megamek/common/MoveStep.java b/megamek/src/megamek/common/MoveStep.java index 71881773e38..f12d21b30b5 100644 --- a/megamek/src/megamek/common/MoveStep.java +++ b/megamek/src/megamek/common/MoveStep.java @@ -28,7 +28,9 @@ import java.io.Serializable; import java.util.ArrayList; +import java.util.HashMap; import java.util.Iterator; +import java.util.Map; import java.util.TreeMap; import java.util.Vector; @@ -40,6 +42,17 @@ */ public class MoveStep implements Serializable { private static final long serialVersionUID = -6075640793056182285L; + /** + * When supplying additional int data, use this to key the index of the cargo being picked up + */ + public static final int CARGO_PICKUP_KEY = 0; + + /** + * When supplying additional int data, use this to key the location of the cargo being picked up + * (i.e. mech left arm/right arm, vehicle body, etc) + */ + public static final int CARGO_LOCATION_KEY = 1; + private MoveStepType type; private int targetId = Entity.NONE; private int targetType = Targetable.TYPE_ENTITY; @@ -156,6 +169,12 @@ public class MoveStep implements Serializable { private boolean maneuver = false; int braceLocation = Entity.LOC_NONE; + + /** + * A map used to hold any additional data that this move step requires. + * Preferable to constantly adding new fields for low-usage one-shot data + */ + Map additionalData = new HashMap<>(); private Minefield mf; @@ -255,10 +274,23 @@ public MoveStep(MovePath path, MoveStepType type, int additionalIntData) { if (type == MoveStepType.BRACE) { this.braceLocation = additionalIntData; - } else { + } else if (type == MoveStepType.LAY_MINE) { this.mineToLay = additionalIntData; + } else if (type == MoveStepType.PICKUP_CARGO) { + this.additionalData.put(CARGO_PICKUP_KEY, additionalIntData); + } else if (type == MoveStepType.DROP_CARGO) { + this.additionalData.put(CARGO_LOCATION_KEY, additionalIntData); } } + + /** + * Creates a step with an arbitrary int-to-int mapping of additional data. + */ + public MoveStep(MovePath path, MoveStepType type, Map additionalIntData) { + this(path, type); + + additionalData.putAll(additionalIntData); + } /** * Create a step with the units to launch or drop. @@ -402,6 +434,10 @@ public String toString() { return "Brace"; case CHAFF: return "Chaff"; + case PICKUP_CARGO: + return "Pickup Cargo"; + case DROP_CARGO: + return "Drop Cargo"; default: return "???"; } @@ -469,6 +505,10 @@ public Coords getTargetPosition() { return targetPos; } + public Integer getAdditionalData(int key) { + return additionalData.containsKey(key) ? additionalData.get(key) : null; + } + public TreeMap> getLaunched() { if (launched == null) { launched = new TreeMap<>(); @@ -1106,6 +1146,9 @@ protected void compile(final Game game, final Entity entity, MoveStep prev, Cach case BRACE: setMp(entity.getBraceMPCost()); break; + case DROP_CARGO: + setMp(1); + break; case CHAFF: default: setMp(0); @@ -1211,6 +1254,7 @@ public void copy(final Game game, MoveStep prev) { nStraight = prev.nStraight; nDown = prev.nDown; nMoved = prev.nMoved; + additionalData = new HashMap<>(additionalData); } /** @@ -2815,6 +2859,11 @@ && isHullDown()) { danger = true; } } + + if (stepType == MoveStepType.PICKUP_CARGO || + stepType == MoveStepType.DROP_CARGO) { + movementType = EntityMovementType.MOVE_NONE; + } // check if this movement is illegal for reasons other than points if (!isMovementPossible(game, lastPos, prev.getElevation(), cachedEntityState) @@ -3371,6 +3420,11 @@ && isThisStepBackwards() if (type == MoveStepType.MOUNT) { return true; } + + if (type == MoveStepType.PICKUP_CARGO || + type == MoveStepType.DROP_CARGO) { + return !isProne(); + } // The entity is trying to load. Check for a valid move. if (type == MoveStepType.LOAD) { @@ -4100,7 +4154,21 @@ public boolean isFacingChangeManeuver() { public Minefield getMinefield() { return mf; } + + /** + * For serialization purposes + */ + public Map getAdditionalData() { + return additionalData; + } + /** + * Setter for serialization purposes + */ + public void setAdditionalData(Map value) { + additionalData = value; + } + /** * Should we treat this movement as if it is occurring for an aerodyne unit * flying in atmosphere? diff --git a/megamek/src/megamek/common/Player.java b/megamek/src/megamek/common/Player.java index 164d3a54662..bddf3526df2 100644 --- a/megamek/src/megamek/common/Player.java +++ b/megamek/src/megamek/common/Player.java @@ -23,6 +23,8 @@ import megamek.common.icons.Camouflage; import megamek.common.options.OptionsConstants; +import java.util.ArrayList; +import java.util.List; import java.util.Objects; import java.util.Vector; @@ -94,6 +96,8 @@ public final class Player extends TurnOrdered { private Vector visibleMinefields = new Vector<>(); private boolean admitsDefeat = false; + + private List groundObjectsToPlace = new ArrayList<>(); //Voting should not be stored in save game so marked transient private transient boolean votedToAllowTeamChange = false; @@ -138,7 +142,8 @@ public boolean containsMinefield(Minefield mf) { } public boolean hasMinefields() { - return (numMfCmd > 0) || (numMfConv > 0) || (numMfVibra > 0) || (numMfActive > 0) || (numMfInferno > 0); + return (numMfCmd > 0) || (numMfConv > 0) || (numMfVibra > 0) || (numMfActive > 0) || (numMfInferno > 0) + || getGroundObjectsToPlace().size() > 0; } public void setNbrMFConventional(int nbrMF) { @@ -456,7 +461,21 @@ public boolean admitsDefeat() { return admitsDefeat; } - public void setVotedToAllowTeamChange(boolean allowChange) { + /** + * Collection of carryable objects that this player will be placing during the game. + */ + public List getGroundObjectsToPlace() { + return groundObjectsToPlace; + } + + /** + * Present for serialization purposes only + */ + public void setGroundObjectsToPlace(List groundObjectsToPlace) { + this.groundObjectsToPlace = groundObjectsToPlace; + } + + public void setVotedToAllowTeamChange(boolean allowChange) { votedToAllowTeamChange = allowChange; } diff --git a/megamek/src/megamek/common/Protomech.java b/megamek/src/megamek/common/Protomech.java index 18d86cc347a..11eebb940ab 100644 --- a/megamek/src/megamek/common/Protomech.java +++ b/megamek/src/megamek/common/Protomech.java @@ -24,8 +24,9 @@ import java.io.PrintWriter; import java.util.ArrayList; -import java.util.Enumeration; +import java.util.HashMap; import java.util.List; +import java.util.Map; import java.util.Vector; import java.util.stream.Collectors; @@ -89,6 +90,22 @@ public class Protomech extends Entity { public static final int[] POSSIBLE_PILOT_DAMAGE = { 0, 1, 3, 1, 1, 1, 0 }; public static final String[] systemNames = { "Arm", "Leg", "Head", "Torso" }; + + /** + * Contains a mapping of locations which are blocked when carrying cargo in the "key" location + */ + public static final Map> BLOCKED_FIRING_LOCATIONS; + + static { + BLOCKED_FIRING_LOCATIONS = new HashMap<>(); + BLOCKED_FIRING_LOCATIONS.put(LOC_LARM, new ArrayList<>()); + BLOCKED_FIRING_LOCATIONS.get(LOC_LARM).add(LOC_LARM); + BLOCKED_FIRING_LOCATIONS.get(LOC_LARM).add(LOC_TORSO); + + BLOCKED_FIRING_LOCATIONS.put(LOC_RARM, new ArrayList<>()); + BLOCKED_FIRING_LOCATIONS.get(LOC_RARM).add(LOC_RARM); + BLOCKED_FIRING_LOCATIONS.get(LOC_RARM).add(LOC_TORSO); + } // For grapple attacks private int grappled_id = Entity.NONE; @@ -557,6 +574,62 @@ public double getArmorWeight() { public boolean hasRearArmor(int loc) { return false; } + + /** + * Returns true if the entity can pick up ground objects + */ + public boolean canPickupGroundObject() { + return !isLocationBad(Protomech.LOC_LARM) && (getCarriedObject(Protomech.LOC_LARM) == null) || + !isLocationBad(Protomech.LOC_RARM) && (getCarriedObject(Protomech.LOC_RARM) == null); + } + + /** + * The maximum tonnage of ground objects that can be picked up by this unit + */ + public double maxGroundObjectTonnage() { + double percentage = 0.0; + + if (!isLocationBad(Protomech.LOC_LARM) && (getCarriedObject(Protomech.LOC_LARM) == null)) { + percentage += 0.05; + } + if (!isLocationBad(Protomech.LOC_RARM) && (getCarriedObject(Protomech.LOC_RARM) == null)) { + percentage += 0.05; + } + + return getWeight() * percentage; + } + + @Override + public List getDefaultPickupLocations() { + List result = new ArrayList<>(); + + if ((getCarriedObject(Protomech.LOC_LARM) == null) && !isLocationBad(Protomech.LOC_LARM)) { + result.add(Protomech.LOC_LARM); + } + if ((getCarriedObject(Protomech.LOC_RARM) == null) && !isLocationBad(Protomech.LOC_RARM)) { + result.add(Protomech.LOC_RARM); + } + + return result; + } + + @Override + public List getValidHalfWeightPickupLocations(ICarryable cargo) { + List result = new ArrayList<>(); + + // if we can pick the object up according to "one handed pick up rules" in TacOps + if (cargo.getTonnage() <= (getWeight() / 20)) { + if ((getCarriedObject(Protomech.LOC_LARM) == null) && !isLocationBad(Protomech.LOC_LARM)) { + result.add(Protomech.LOC_LARM); + } + + if ((getCarriedObject(Protomech.LOC_RARM) == null) && !isLocationBad(Protomech.LOC_RARM)) { + result.add(Protomech.LOC_RARM); + } + } + + return result; + } @Override public int getRunMP(MPCalculationSetting mpCalculationSetting) { @@ -1526,6 +1599,13 @@ protected Mounted getEquipmentForWeaponQuirk(QuirkEntry quirkEntry) { } } return null; - + } + + /** + * Method that returns the mapping between locations which, if cargo is carried, + * block other locations from firing. + */ + protected Map> getBlockedFiringLocations() { + return BLOCKED_FIRING_LOCATIONS; } } diff --git a/megamek/src/megamek/common/Tank.java b/megamek/src/megamek/common/Tank.java index 9fab2b3d9d6..4f8c15ab434 100644 --- a/megamek/src/megamek/common/Tank.java +++ b/megamek/src/megamek/common/Tank.java @@ -285,9 +285,11 @@ protected void addSystemTechAdvancement(CompositeTechLevel ctl) { @Override public int getWalkMP(MPCalculationSetting mpCalculationSetting) { int mp = getOriginalWalkMP(); + if (engineHit || isImmobile()) { return 0; } + if (hasWorkingMisc(MiscType.F_HYDROFOIL)) { mp = (int) Math.round(mp * 1.25); } diff --git a/megamek/src/megamek/common/TripodMech.java b/megamek/src/megamek/common/TripodMech.java index 92ee039874d..809400c0ad3 100644 --- a/megamek/src/megamek/common/TripodMech.java +++ b/megamek/src/megamek/common/TripodMech.java @@ -134,6 +134,66 @@ public boolean canFlipArms() { return canFlip; } + + /** + * Returns true if the entity can pick up ground objects + */ + public boolean canPickupGroundObject() { + return hasSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) || + hasSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null); + } + + /** + * The maximum tonnage of ground objects that can be picked up by this unit + */ + public double maxGroundObjectTonnage() { + double percentage = 0.0; + + if (hasSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null)) { + percentage += 0.05; + } + if (hasSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null)) { + percentage += 0.05; + } + + return getWeight() * percentage; + } + + @Override + public List getDefaultPickupLocations() { + List result = new ArrayList<>(); + + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) && + !isLocationBad(Mech.LOC_LARM)) { + result.add(Mech.LOC_LARM); + } + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null) && + !isLocationBad(Mech.LOC_RARM)) { + result.add(Mech.LOC_RARM); + } + + return result; + } + + @Override + public List getValidHalfWeightPickupLocations(ICarryable cargo) { + List result = new ArrayList<>(); + + // if we can pick the object up according to "one handed pick up rules" in TacOps + if (cargo.getTonnage() <= (getWeight() / 20)) { + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_LARM) && (getCarriedObject(Mech.LOC_LARM) == null) && + !isLocationBad(Mech.LOC_LARM)) { + result.add(Mech.LOC_LARM); + } + + if (hasWorkingSystem(Mech.ACTUATOR_HAND, Mech.LOC_RARM) && (getCarriedObject(Mech.LOC_RARM) == null) && + !isLocationBad(Mech.LOC_RARM)) { + result.add(Mech.LOC_RARM); + } + } + + return result; + } @Override public int getWalkMP(MPCalculationSetting mpCalculationSetting) { diff --git a/megamek/src/megamek/common/actions/ClubAttackAction.java b/megamek/src/megamek/common/actions/ClubAttackAction.java index 23dae2bd785..76798bc263c 100644 --- a/megamek/src/megamek/common/actions/ClubAttackAction.java +++ b/megamek/src/megamek/common/actions/ClubAttackAction.java @@ -285,6 +285,13 @@ public static ToHitData toHit(Game game, int attackerId, if (!(ae instanceof Mech)) { return new ToHitData(TargetRoll.IMPOSSIBLE, "Non-mechs can't club"); } + + // if somehow carrying cargo while holding a club + if (!((Mech) ae).canFireWeapon(Mech.LOC_LARM) || + !((Mech) ae).canFireWeapon(Mech.LOC_LARM) ) { + return new ToHitData(TargetRoll.IMPOSSIBLE, + Messages.getString("WeaponAttackAction.CantFireWhileCarryingCargo")); + } // Quads can't club... // except for torso mounted industrial tools of course! diff --git a/megamek/src/megamek/common/actions/PhysicalAttackAction.java b/megamek/src/megamek/common/actions/PhysicalAttackAction.java index 65d12deedd6..7b42096c56e 100644 --- a/megamek/src/megamek/common/actions/PhysicalAttackAction.java +++ b/megamek/src/megamek/common/actions/PhysicalAttackAction.java @@ -65,6 +65,11 @@ public PhysicalAttackAction(int entityId, int targetType, int targetId) { if (ae.isEvading()) { return "Attacker is evading."; } + + // can't make physical attacks if loading/unloading cargo + if (ae.endOfTurnCargoInteraction()) { + return Messages.getString("WeaponAttackAction.CantFireWhileLoadingUnloadingCargo"); + } if (target.getTargetType() == Targetable.TYPE_ENTITY) { // Checks specific to entity targets diff --git a/megamek/src/megamek/common/actions/PunchAttackAction.java b/megamek/src/megamek/common/actions/PunchAttackAction.java index 90b2f0c31ff..ecc5f9ad18f 100644 --- a/megamek/src/megamek/common/actions/PunchAttackAction.java +++ b/megamek/src/megamek/common/actions/PunchAttackAction.java @@ -176,6 +176,11 @@ protected static String toHitIsImpossible(Game game, Entity ae, if (ae.hasActiveShield(armLoc)) { return "Cannot punch with shield in active mode"; } + + if (!((Mech) ae).canFireWeapon(armLoc)) { + return Messages.getString("WeaponAttackAction.CantFireWhileCarryingCargo"); + } + return null; } diff --git a/megamek/src/megamek/common/actions/PushAttackAction.java b/megamek/src/megamek/common/actions/PushAttackAction.java index 4aa85d7ff76..10639495b44 100644 --- a/megamek/src/megamek/common/actions/PushAttackAction.java +++ b/megamek/src/megamek/common/actions/PushAttackAction.java @@ -46,11 +46,7 @@ public ToHitData toHit(Game game) { */ protected static String toHitIsImpossible(Game game, Entity ae, Targetable target) { String physicalImpossible = PhysicalAttackAction.toHitIsImpossible(game, ae, target); - String extendedBladeImpossible = null; - if ((ae instanceof Mech) && ((Mech) ae).hasExtendedRetractableBlade()) { - extendedBladeImpossible = "Extended retractable blade"; - } - + if (physicalImpossible != null) { return physicalImpossible; } @@ -58,23 +54,19 @@ protected static String toHitIsImpossible(Game game, Entity ae, Targetable targe if (ae.getGrappled() != Entity.NONE) { return "Unit Grappled"; } - - if (ae.isEvading()) { - return "attacker is evading."; - } - - if (!game.getOptions().booleanOption(OptionsConstants.BASE_FRIENDLY_FIRE)) { - // a friendly unit can never be the target of a direct attack. - if ((target.getTargetType() == Targetable.TYPE_ENTITY) - && ((((Entity) target).getOwnerId() == ae.getOwnerId()) - || ((((Entity) target).getOwner().getTeam() != Player.TEAM_NONE) - && (ae.getOwner().getTeam() != Player.TEAM_NONE) - && (ae.getOwner().getTeam() == ((Entity) target).getOwner().getTeam())))) { - return "A friendly unit can never be the target of a direct attack."; - } + + // can't push if carrying any cargo per TW + if ((ae instanceof Mech) && + !((Mech) ae).canFireWeapon(Mech.LOC_LARM) || + !((Mech) ae).canFireWeapon(Mech.LOC_LARM) ) { + return Messages.getString("WeaponAttackAction.CantFireWhileCarryingCargo"); + } + + if ((ae instanceof Mech) && ((Mech) ae).hasExtendedRetractableBlade()) { + return "Extended retractable blade"; } - - return extendedBladeImpossible; + + return null; } /** @@ -248,6 +240,11 @@ public static ToHitData toHit(Game game, int attackerId, Targetable target) { return new ToHitData(TargetRoll.IMPOSSIBLE, "Invalid attack"); } + String otherImpossible = toHitIsImpossible(game, ae, target); + if (otherImpossible != null) { + return new ToHitData(TargetRoll.IMPOSSIBLE, otherImpossible); + } + // Set the base BTH int base = ae.getCrew().getPiloting() - 1; diff --git a/megamek/src/megamek/common/actions/WeaponAttackAction.java b/megamek/src/megamek/common/actions/WeaponAttackAction.java index dd9ebaef9c8..19a137259b1 100644 --- a/megamek/src/megamek/common/actions/WeaponAttackAction.java +++ b/megamek/src/megamek/common/actions/WeaponAttackAction.java @@ -1078,6 +1078,20 @@ private static String toHitIsImpossible(Game game, Entity ae, int attackerId, Ta } } } + + // can't fire weapons if loading/unloading cargo + if (ae.endOfTurnCargoInteraction()) { + return Messages.getString("WeaponAttackAction.CantFireWhileLoadingUnloadingCargo"); + } + + // can't fire arm/forward facing torso weapons if carrying cargo in hands + if ((weapon != null)) { + int loc = weapon.getLocation(); + + if ((ae instanceof Mech) && !weapon.isRearMounted() && !((Mech) ae).canFireWeapon(loc)) { + return Messages.getString("WeaponAttackAction.CantFireWhileCarryingCargo"); + } + } // Only large spacecraft can shoot while evading if (ae.isEvading() && !(ae instanceof Dropship) && !(ae instanceof Jumpship)) { diff --git a/megamek/src/megamek/common/net/enums/PacketCommand.java b/megamek/src/megamek/common/net/enums/PacketCommand.java index efcd119f70c..d140bcacc99 100644 --- a/megamek/src/megamek/common/net/enums/PacketCommand.java +++ b/megamek/src/megamek/common/net/enums/PacketCommand.java @@ -118,6 +118,7 @@ public enum PacketCommand { REMOVE_MINEFIELD, SENDING_MINEFIELDS, UPDATE_MINEFIELDS, + UPDATE_GROUND_OBJECTS, REROLL_INITIATIVE, UNLOAD_STRANDED, SET_ARTILLERY_AUTOHIT_HEXES, diff --git a/megamek/src/megamek/server/GameManager.java b/megamek/src/megamek/server/GameManager.java index df130b2c0f4..b0bb74a4424 100644 --- a/megamek/src/megamek/server/GameManager.java +++ b/megamek/src/megamek/server/GameManager.java @@ -41,8 +41,6 @@ import megamek.common.planetaryconditions.PlanetaryConditions; import megamek.common.planetaryconditions.Wind; import megamek.common.preference.PreferenceManager; -import megamek.common.Report; -import megamek.common.ReportMessages; import megamek.common.util.*; import megamek.common.util.fileUtils.MegaMekFile; import megamek.common.verifier.*; @@ -543,6 +541,7 @@ public void sendCurrentInfo(int connId) { send(connId, createFlarePacket()); send(connId, createSpecialHexDisplayPacket(connId)); send(connId, new Packet(PacketCommand.PRINCESS_SETTINGS, getGame().getBotSettings())); + send(connId, new Packet(PacketCommand.UPDATE_GROUND_OBJECTS, getGame().getGroundObjects())); } } @@ -608,6 +607,9 @@ public void handlePacket(int connId, Packet packet) { case DEPLOY_MINEFIELDS: receiveDeployMinefields(packet, connId); break; + case UPDATE_GROUND_OBJECTS: + receiveGroundObjectUpdate(packet, connId); + break; case ENTITY_ATTACK: receiveAttack(packet, connId); break; @@ -5297,6 +5299,14 @@ private Vector processLeaveMap(MovePath movePath, boolean flewOff, int r game.removeEntity(swarmerId, IEntityRemovalConditions.REMOVE_CAPTURED); send(createRemoveEntityPacket(swarmerId, IEntityRemovalConditions.REMOVE_CAPTURED)); } + + for (ICarryable cargo : entity.getDistinctCarriedObjects()) { + r = new Report(2016, Report.PUBLIC); + r.indent(); + r.add(cargo.generalName()); + addReport(r); + } + entity.setRetreatedDirection(fleeDirection); game.removeEntity(entity.getId(), IEntityRemovalConditions.REMOVE_IN_RETREAT); send(createRemoveEntityPacket(entity.getId(), IEntityRemovalConditions.REMOVE_IN_RETREAT)); @@ -7255,6 +7265,84 @@ else if ((step.getElevation() + entity.height()) == 0) { } } // End STEP_MOUNT + if (step.getType() == MovePath.MoveStepType.PICKUP_CARGO) { + var groundObjects = game.getGroundObjects(step.getPosition()); + Integer cargoPickupIndex; + + // if there's only one object on the ground, let's just get that one and ignore any parameters + if (groundObjects.size() == 1) { + cargoPickupIndex = 0; + } else { + cargoPickupIndex = step.getAdditionalData(MoveStep.CARGO_PICKUP_KEY); + } + + Integer cargoPickupLocation = step.getAdditionalData(MoveStep.CARGO_LOCATION_KEY); + + // there have to be objects on the ground and we have to be trying to pick up one of them + if ((groundObjects.size() > 0) && + (cargoPickupIndex != null) && (cargoPickupIndex >= 0) && (cargoPickupIndex < groundObjects.size())) { + + ICarryable pickupTarget = groundObjects.get(cargoPickupIndex); + if (entity.maxGroundObjectTonnage() >= pickupTarget.getTonnage()) { + game.removeGroundObject(step.getPosition(), pickupTarget); + entity.pickupGroundObject(pickupTarget, cargoPickupLocation); + + r = new Report(2513); + r.subject = entity.getId(); + r.add(entity.getDisplayName()); + r.add(pickupTarget.specificName()); + r.add(step.getPosition().toFriendlyString()); + addReport(r); + + // a pickup should be the last step. Send an update for the overall ground object list. + sendGroundObjectUpdate(); + break; + } else { + LogManager.getLogger().warn(entity.getShortName() + " attempted to pick up object but it is too heavy. Carry capacity: " + + entity.maxGroundObjectTonnage() + ", object weight: " + pickupTarget.getTonnage()); + } + } else { + LogManager.getLogger().warn(entity.getShortName() + " attempted to pick up non existent object at coords " + + step.getPosition() + ", index " + cargoPickupIndex); + } + } + + if (step.getType() == MovePath.MoveStepType.DROP_CARGO) { + Integer cargoLocation = step.getAdditionalData(MoveStep.CARGO_LOCATION_KEY); + ICarryable cargo; + + // if we're not supplied a specific location, then the assumption is we only have one + // piece of cargo and we're going to just drop that one + if (cargoLocation == null) { + cargo = entity.getDistinctCarriedObjects().get(0); + } else { + cargo = entity.getCarriedObject(cargoLocation); + } + + entity.dropGroundObject(cargo, isLastStep); + boolean cargoDestroyed = false; + + if (!isLastStep) { + cargoDestroyed = damageCargo(step.isFlying() || step.isJumping(), entity, cargo); + } + + // note that this should not be moved into the "!isLastStep" block above + // as cargo may be either unloaded peacefully or dumped on the move + if (!cargoDestroyed) { + game.placeGroundObject(step.getPosition(), cargo); + + r = new Report(2514); + r.subject = entity.getId(); + r.add(entity.getDisplayName()); + r.add(cargo.generalName()); + r.add(step.getPosition().toFriendlyString()); + addReport(r); + + // a drop changes board state. Send an update for the overall ground object list. + sendGroundObjectUpdate(); + } + } + // handle fighter recovery, and also DropShip docking with another large craft if (step.getType() == MovePath.MoveStepType.RECOVER) { @@ -12287,6 +12375,18 @@ private void receiveArtyAutoHitHexes(Packet packet, int connId) { } endCurrentTurn(null); } + + /** + * Receives an updated data structure containing carryable objects on the ground + */ + private void receiveGroundObjectUpdate(Packet packet, int connId) { + Map> groundObjects = (Map>) packet.getObject(0); + + getGame().setGroundObjects(groundObjects); + + // make sure to update the other clients with the new ground objects data structure + send(packet); + } /** * receive a packet that contains minefields @@ -21173,6 +21273,43 @@ public Vector damageEntity(Entity te, HitData hit, int damage, // Allocate the damage while (damage > 0) { + + // damage some cargo if we're taking damage + // maybe move past "exterior passenger" check + if (!ammoExplosion) { + int damageLeftToCargo = damage; + + for (ICarryable cargo : te.getDistinctCarriedObjects()) { + if (cargo.isInvulnerable()) { + continue; + } + + double tonnage = cargo.getTonnage(); + cargo.damage(damageLeftToCargo); + damageLeftToCargo -= Math.ceil(tonnage); + + // if we have destroyed the cargo, remove it, add a report + // and move on to the next piece of cargo + if (cargo.getTonnage() <= 0) { + te.dropGroundObject(cargo, false); + + r = new Report(6721); + r.subject = te_n; + r.indent(2); + r.add(cargo.generalName()); + vDesc.addElement(r); + // we have not destroyed the cargo means there is no damage left + // report and stop destroying cargo + } else { + r = new Report(6720); + r.subject = te_n; + r.indent(2); + r.add(cargo.generalName()); + r.add(Double.toString(cargo.getTonnage())); + break; + } + } + } // first check for ammo explosions on aeros separately, because it // must be done before @@ -26911,12 +27048,43 @@ else if ((null != Compute.stackingViolation(game, other.getId(), curPos, entity. sendChangedHex(curPos); } } + + // drop cargo + dropCargo(entity, curPos, vDesc); // update our entity, so clients have correct data needed for MekWars stuff entityUpdate(entity.getId()); return vDesc; } + + /** + * Worker function that drops cargo from an entity at the given coordinates. + */ + private void dropCargo(Entity entity, Coords coords, Vector vPhaseReport) { + boolean cargoDropped = false; + + for (ICarryable cargo : entity.getDistinctCarriedObjects()) { + entity.dropGroundObject(cargo, false); + // if the cargo was dropped but not destroyed. + if (!damageCargo(false, entity, cargo)) { + Report r = new Report(6722); + r.indent(); + r.subject = entity.getId(); + r.add(cargo.generalName()); + vPhaseReport.add(r); + + if (game.getBoard().contains(coords)) { + game.placeGroundObject(coords, cargo); + cargoDropped = true; + } + } + } + + if (cargoDropped) { + sendGroundObjectUpdate(); + } + } /** * Makes a piece of equipment on a mech explode! POW! This expects either @@ -27386,7 +27554,7 @@ else if (waterDepth > 0) { vPhaseReport.addAll(destroyEntity(entity, "a watery grave", false)); return vPhaseReport; } - + // set how deep the mech has fallen if (entity instanceof Mech) { Mech mech = (Mech) entity; @@ -27594,6 +27762,9 @@ else if (waterDepth > 0) { } } // End dislodge-infantry + // drop cargo if necessary + dropCargo(entity, fallPos, vPhaseReport); + // clear all PSRs after a fall -- the Mek has already failed ONE and // fallen, it'd be cruel to make it fail some more! game.resetPSRs(entity); @@ -34015,6 +34186,42 @@ private void layMine(Entity entity, int mineId, Coords coords) { entity.setLayingMines(true); } } + + /** + * Worker function that potentially damages a piece of cargo being carried + * by the given entity during the given move step. + * + * @param isFlying whether the entity's movement involved being in the air in any way + */ + private boolean damageCargo(boolean isFlying, Entity entity, ICarryable cargo) { + if (cargo.isInvulnerable()) { + return false; + } + + boolean cargoDestroyed = false; + + // cargo may be destroyed if we're not carefully unloading it + // very likely to be destroyed if we're airborne for some reason + int destructionThreshold = isFlying ? 6 : 4; + int destructionRoll = Compute.d6(); + + Report r = new Report(2515); + r.subject = entity.getId(); + r.add(cargo.generalName()); + r.add(destructionThreshold); + r.add(destructionRoll); + + if (destructionRoll < destructionThreshold) { + cargoDestroyed = true; + r.choose(false); + } else { + r.choose(true); + } + + addReport(r); + + return cargoDestroyed; + } public Set getHexUpdateSet() { return hexUpdateSet; @@ -34035,4 +34242,11 @@ List getTerrainProcessors() { void clearBombIcons() { game.getBoard().clearBombIcons(); } + + /** + * Convenience function to send a ground object update. + */ + public void sendGroundObjectUpdate() { + send(new Packet(PacketCommand.UPDATE_GROUND_OBJECTS, getGame().getGroundObjects())); + } } diff --git a/megamek/src/megamek/server/Server.java b/megamek/src/megamek/server/Server.java index 2d6f235aaae..49376d18071 100644 --- a/megamek/src/megamek/server/Server.java +++ b/megamek/src/megamek/server/Server.java @@ -594,6 +594,7 @@ private void receivePlayerInfo(Packet packet, int connId) { } gamePlayer.setConstantInitBonus(player.getConstantInitBonus()); gamePlayer.setEmail(player.getEmail()); + gamePlayer.setGroundObjectsToPlace(new ArrayList<>(player.getGroundObjectsToPlace())); } } diff --git a/megamek/src/megamek/server/TWPhaseEndManager.java b/megamek/src/megamek/server/TWPhaseEndManager.java index 2cf18983850..9a3bec51437 100644 --- a/megamek/src/megamek/server/TWPhaseEndManager.java +++ b/megamek/src/megamek/server/TWPhaseEndManager.java @@ -150,6 +150,7 @@ void managePhase() { gameManager.getGame().addReports(gameManager.getvPhaseReport()); gameManager.changePhase(GamePhase.PHYSICAL); } + gameManager.sendGroundObjectUpdate(); // For bomb markers gameManager.sendSpecialHexDisplayPackets(); break; @@ -177,6 +178,7 @@ void managePhase() { gameManager.sendReport(); gameManager.changePhase(GamePhase.END); } + gameManager.sendGroundObjectUpdate(); break; case PHYSICAL_REPORT: gameManager.changePhase(GamePhase.END);