cn.nukkit.blockentity.BlockEntityBeacon Maven / Gradle / Ivy
package cn.nukkit.blockentity;
import cn.nukkit.Player;
import cn.nukkit.api.PowerNukkitDifference;
import cn.nukkit.api.Since;
import cn.nukkit.block.Block;
import cn.nukkit.block.BlockID;
import cn.nukkit.inventory.BeaconInventory;
import cn.nukkit.item.ItemBlock;
import cn.nukkit.level.Sound;
import cn.nukkit.level.format.FullChunk;
import cn.nukkit.nbt.tag.CompoundTag;
import cn.nukkit.potion.Effect;
import java.util.Map;
/**
* @author Rover656
*/
public class BlockEntityBeacon extends BlockEntitySpawnable {
public BlockEntityBeacon(FullChunk chunk, CompoundTag nbt) {
super(chunk, nbt);
}
@Override
protected void initBlockEntity() {
super.initBlockEntity();
scheduleUpdate();
}
@Since("1.19.60-r1")
@Override
public void loadNBT() {
super.loadNBT();
if (!namedTag.contains("Lock")) {
namedTag.putString("Lock", "");
}
if (!namedTag.contains("Levels")) {
namedTag.putInt("Levels", 0);
}
if (!namedTag.contains("Primary")) {
namedTag.putInt("Primary", 0);
}
if (!namedTag.contains("Secondary")) {
namedTag.putInt("Secondary", 0);
}
}
@Override
public boolean isBlockEntityValid() {
int blockID = getBlock().getId();
return blockID == Block.BEACON;
}
@Override
public CompoundTag getSpawnCompound() {
return new CompoundTag()
.putString("id", BlockEntity.BEACON)
.putInt("x", (int) this.x)
.putInt("y", (int) this.y)
.putInt("z", (int) this.z)
.putString("Lock", this.namedTag.getString("Lock"))
.putInt("Levels", this.namedTag.getInt("Levels"))
.putInt("primary", this.namedTag.getInt("Primary"))
.putInt("secondary", this.namedTag.getInt("Secondary"));
}
private long currentTick = 0;
@PowerNukkitDifference(info = "Using new method to play sounds", since = "1.4.0.0-PN")
@Override
public boolean onUpdate() {
//Only apply effects every 4 secs
if (currentTick++ % 80 != 0) {
return true;
}
int oldPowerLevel = this.getPowerLevel();
//Get the power level based on the pyramid
setPowerLevel(calculatePowerLevel());
int newPowerLevel = this.getPowerLevel();
//Skip beacons that do not have a pyramid or sky access
if (newPowerLevel < 1 || !hasSkyAccess()) {
if (oldPowerLevel > 0) {
this.getLevel().addSound(this, Sound.BEACON_DEACTIVATE);
}
return true;
} else if (oldPowerLevel < 1) {
this.getLevel().addSound(this, Sound.BEACON_ACTIVATE);
} else {
this.getLevel().addSound(this, Sound.BEACON_AMBIENT);
}
//Get all players in game
Map players = this.level.getPlayers();
//Calculate vars for beacon power
int range = 10 + getPowerLevel() * 10;
int duration = 9 + getPowerLevel() * 2;
if (!isPrimaryAllowed(getPrimaryPower(), getPowerLevel())) {
return true;
}
for(Map.Entry entry : players.entrySet()) {
Player p = entry.getValue();
//If the player is in range
if (p.distance(this) < range) {
Effect e;
if (getPrimaryPower() != 0) {
//Apply the primary power
e = Effect.getEffect(getPrimaryPower());
//Set duration
e.setDuration(duration * 20);
//If secondary is selected as the primary too, apply 2 amplification
if (getPowerLevel() == POWER_LEVEL_MAX && getSecondaryPower() == getPrimaryPower()) {
e.setAmplifier(1);
}
//Add the effect
p.addEffect(e);
}
//If we have a secondary power as regen, apply it
if (getPowerLevel() == POWER_LEVEL_MAX && getSecondaryPower() == Effect.REGENERATION) {
//Get the regen effect
e = Effect.getEffect(Effect.REGENERATION);
//Set duration
e.setDuration(duration * 20);
//Add effect
p.addEffect(e);
}
}
}
return true;
}
private static final int POWER_LEVEL_MAX = 4;
private boolean hasSkyAccess() {
int tileX = getFloorX();
int tileY = getFloorY();
int tileZ = getFloorZ();
//Check every block from our y coord to the top of the world
for (int y = tileY + 1; y <= 255; y++) {
int testBlockId = level.getBlockIdAt(tileX, y, tileZ);
if (!Block.isTransparent(testBlockId)) {
//There is no sky access
return false;
}
}
return true;
}
private int calculatePowerLevel() {
int tileX = getFloorX();
int tileY = getFloorY();
int tileZ = getFloorZ();
//The power level that we're testing for
for (int powerLevel = 1; powerLevel <= POWER_LEVEL_MAX; powerLevel++) {
int queryY = tileY - powerLevel; //Layer below the beacon block
for (int queryX = tileX - powerLevel; queryX <= tileX + powerLevel; queryX++) {
for (int queryZ = tileZ - powerLevel; queryZ <= tileZ + powerLevel; queryZ++) {
int testBlockId = level.getBlockIdAt(queryX, queryY, queryZ);
if (
testBlockId != Block.IRON_BLOCK &&
testBlockId != Block.GOLD_BLOCK &&
testBlockId != Block.EMERALD_BLOCK &&
testBlockId != Block.DIAMOND_BLOCK &&
testBlockId != Block.NETHERITE_BLOCK
) {
return powerLevel - 1;
}
}
}
}
return POWER_LEVEL_MAX;
}
public int getPowerLevel() {
return namedTag.getInt("Levels");
}
public void setPowerLevel(int level) {
int currentLevel = getPowerLevel();
if (level != currentLevel) {
namedTag.putInt("Levels", level);
setDirty();
this.spawnToAll();
}
}
public int getPrimaryPower() {
return namedTag.getInt("Primary");
}
public void setPrimaryPower(int power) {
int currentPower = getPrimaryPower();
if (power != currentPower) {
namedTag.putInt("Primary", power);
setDirty();
this.spawnToAll();
}
}
public int getSecondaryPower() {
return namedTag.getInt("Secondary");
}
public void setSecondaryPower(int power) {
int currentPower = getSecondaryPower();
if (power != currentPower) {
namedTag.putInt("Secondary", power);
setDirty();
this.spawnToAll();
}
}
@PowerNukkitDifference(info = "Using new method to play sounds", since = "1.4.0.0-PN")
@Override
public boolean updateCompoundTag(CompoundTag nbt, Player player) {
if (!nbt.getString("id").equals(BlockEntity.BEACON)) {
return false;
}
int primary = nbt.getInt("primary");
if (!isPrimaryAllowed(primary, this.getPowerLevel())) {
return false;
}
int secondary = nbt.getInt("secondary");
if (secondary != 0 && secondary != primary && secondary != Effect.REGENERATION) {
return false;
}
this.setPrimaryPower(primary);
this.setSecondaryPower(secondary);
this.getLevel().addSound(this, Sound.BEACON_POWER);
BeaconInventory inv = (BeaconInventory)player.getWindowById(Player.BEACON_WINDOW_ID);
inv.setItem(0, new ItemBlock(Block.get(BlockID.AIR), 0, 0));
return true;
}
private static boolean isPrimaryAllowed(int primary, int powerLevel) {
return ((primary == Effect.SPEED || primary == Effect.HASTE) && powerLevel >= 1) ||
((primary == Effect.DAMAGE_RESISTANCE || primary == Effect.JUMP) && powerLevel >= 2) ||
(primary == Effect.STRENGTH && powerLevel >= 3);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy