package emu.grasscutter.command.commands; import emu.grasscutter.command.Command; import emu.grasscutter.command.CommandHandler; import emu.grasscutter.data.GameData; import emu.grasscutter.data.excels.GadgetData; import emu.grasscutter.data.excels.ItemData; import emu.grasscutter.data.excels.MonsterData; import emu.grasscutter.game.entity.*; import emu.grasscutter.game.player.Player; import emu.grasscutter.game.props.FightProperty; import emu.grasscutter.utils.Position; import emu.grasscutter.game.world.Scene; import java.util.List; import static emu.grasscutter.config.Configuration.*; import static emu.grasscutter.utils.Language.translate; @Command( label = "spawn", usage = {"spawn [amount] [level(monster only)] [ (monster only)]"}, aliases = {"drop"}, permission = "server.spawn", permissionTargeted = "server.spawn.others") public final class SpawnCommand implements CommandHandler { @Override public void execute(Player sender, Player targetPlayer, List args) { int id = 0; // This is just to shut up the linter, it's not a real default int amount = 1; int level = 1; float x = 0, y = 0, z = 0; switch (args.size()) { case 6: try { x = Float.parseFloat(args.get(3)); y = Float.parseFloat(args.get(4)); z = Float.parseFloat(args.get(5)); } catch (NumberFormatException ignored) { CommandHandler.sendMessage(sender, translate(sender, "commands.execution.argument_error")); } // Fallthrough case 3: try { level = Integer.parseInt(args.get(2)); } catch (NumberFormatException ignored) { CommandHandler.sendMessage(sender, translate(sender, "commands.execution.argument_error")); } // Fallthrough case 2: try { amount = Integer.parseInt(args.get(1)); } catch (NumberFormatException ignored) { CommandHandler.sendMessage(sender, translate(sender, "commands.generic.invalid.amount")); } // Fallthrough case 1: try { id = Integer.parseInt(args.get(0)); } catch (NumberFormatException ignored) { CommandHandler.sendMessage(sender, translate(sender, "commands.generic.invalid.entityId")); } break; default: sendUsageMessage(sender); return; } MonsterData monsterData = GameData.getMonsterDataMap().get(id); GadgetData gadgetData = GameData.getGadgetDataMap().get(id); ItemData itemData = GameData.getItemDataMap().get(id); if (monsterData == null && gadgetData == null && itemData == null) { CommandHandler.sendMessage(sender, translate(sender, "commands.generic.invalid.entityId")); return; } Scene scene = targetPlayer.getScene(); if (scene.getEntities().size() + amount > GAME_OPTIONS.sceneEntityLimit) { amount = Math.max(Math.min(GAME_OPTIONS.sceneEntityLimit - scene.getEntities().size(), amount), 0); CommandHandler.sendMessage(sender, translate(sender, "commands.spawn.limit_reached", amount)); if (amount <= 0) { return; } } double maxRadius = Math.sqrt(amount * 0.2 / Math.PI); for (int i = 0; i < amount; i++) { Position pos = GetRandomPositionInCircle(targetPlayer.getPosition(), maxRadius).addY(3); if (x != 0 && y != 0 && z != 0) { pos = GetRandomPositionInCircle(new Position(x, y, z), maxRadius).addY(3); } GameEntity entity = null; if (itemData != null) { entity = new EntityItem(scene, null, itemData, pos, 1, true); } if (gadgetData != null) { pos.addY(-3); entity = new EntityVehicle(scene, targetPlayer.getSession().getPlayer(), gadgetData.getId(), 0, pos, targetPlayer.getRotation()); // TODO: does targetPlayer.getSession().getPlayer() have some meaning? int gadgetId = gadgetData.getId(); switch (gadgetId) { // TODO: Not hardcode this. Waverider (skiff) case 45001001, 45001002 -> { entity.addFightProperty(FightProperty.FIGHT_PROP_BASE_HP, 10000); entity.addFightProperty(FightProperty.FIGHT_PROP_BASE_ATTACK, 100); entity.addFightProperty(FightProperty.FIGHT_PROP_CUR_ATTACK, 100); entity.addFightProperty(FightProperty.FIGHT_PROP_CUR_HP, 10000); entity.addFightProperty(FightProperty.FIGHT_PROP_CUR_DEFENSE, 0); entity.addFightProperty(FightProperty.FIGHT_PROP_CUR_SPEED, 0); entity.addFightProperty(FightProperty.FIGHT_PROP_CHARGE_EFFICIENCY, 0); entity.addFightProperty(FightProperty.FIGHT_PROP_MAX_HP, 10000); } default -> {} } } if (monsterData != null) { entity = new EntityMonster(scene, monsterData, pos, level); } scene.addEntity(entity); } CommandHandler.sendMessage(sender, translate(sender, "commands.spawn.success", amount, id)); } private Position GetRandomPositionInCircle(Position origin, double radius) { Position target = origin.clone(); double angle = Math.random() * 360; double r = Math.sqrt(Math.random() * radius * radius); target.addX((float) (r * Math.cos(angle))).addZ((float) (r * Math.sin(angle))); return target; } }