cn.nukkit.block.BlockCauldron Maven / Gradle / Ivy
package cn.nukkit.block;
import cn.nukkit.Player;
import cn.nukkit.api.PowerNukkitDifference;
import cn.nukkit.api.PowerNukkitOnly;
import cn.nukkit.api.PowerNukkitXOnly;
import cn.nukkit.api.Since;
import cn.nukkit.blockentity.BlockEntity;
import cn.nukkit.blockentity.BlockEntityCauldron;
import cn.nukkit.blockproperty.ArrayBlockProperty;
import cn.nukkit.blockproperty.BlockProperties;
import cn.nukkit.blockproperty.IntBlockProperty;
import cn.nukkit.blockproperty.value.CauldronLiquid;
import cn.nukkit.event.player.PlayerBucketEmptyEvent;
import cn.nukkit.event.player.PlayerBucketFillEvent;
import cn.nukkit.item.*;
import cn.nukkit.level.Sound;
import cn.nukkit.level.particle.SmokeParticle;
import cn.nukkit.level.vibration.VibrationEvent;
import cn.nukkit.level.vibration.VibrationType;
import cn.nukkit.math.BlockFace;
import cn.nukkit.nbt.tag.CompoundTag;
import cn.nukkit.nbt.tag.Tag;
import cn.nukkit.network.protocol.LevelEventPacket;
import cn.nukkit.utils.BlockColor;
import org.jetbrains.annotations.NotNull;
import javax.annotation.Nullable;
import java.util.Map;
/**
* @author CreeperFace (Nukkit Project)
*/
@PowerNukkitDifference(since = "1.4.0.0-PN", info = "Implements BlockEntityHolder only in PowerNukkit")
public class BlockCauldron extends BlockSolidMeta implements BlockEntityHolder {
@PowerNukkitOnly
@Since("1.5.0.0-PN")
public static final IntBlockProperty FILL_LEVEL = new IntBlockProperty("fill_level", false, 6);
@PowerNukkitOnly
@Since("1.5.0.0-PN")
public static final ArrayBlockProperty LIQUID = new ArrayBlockProperty<>("cauldron_liquid", false, CauldronLiquid.class);
@PowerNukkitOnly
@Since("1.5.0.0-PN")
public static final BlockProperties PROPERTIES = new BlockProperties(FILL_LEVEL, LIQUID);
public BlockCauldron() {
super(0);
}
public BlockCauldron(int meta) {
super(meta);
}
@Override
public int getId() {
return CAULDRON_BLOCK;
}
@Since("1.4.0.0-PN")
@PowerNukkitOnly
@NotNull
@Override
public BlockProperties getProperties() {
return PROPERTIES;
}
@PowerNukkitOnly
@Since("1.4.0.0-PN")
@NotNull
@Override
public String getBlockEntityType() {
return BlockEntity.CAULDRON;
}
@Since("1.4.0.0-PN")
@PowerNukkitOnly
@NotNull
@Override
public Class extends BlockEntityCauldron> getBlockEntityClass() {
return BlockEntityCauldron.class;
}
@Override
public String getName() {
return "Cauldron Block";
}
@Override
public double getResistance() {
return 10;
}
@Override
public double getHardness() {
return 2;
}
@Override
public int getToolType() {
return ItemTool.TYPE_PICKAXE;
}
@Override
public boolean canBeActivated() {
return true;
}
public boolean isFull() {
return getFillLevel() == FILL_LEVEL.getMaxValue();
}
public boolean isEmpty() {
return getFillLevel() == FILL_LEVEL.getMinValue();
}
@PowerNukkitOnly
public int getFillLevel() {
return getIntValue(FILL_LEVEL);
}
@PowerNukkitOnly
public void setFillLevel(int fillLevel) {
this.setFillLevel(fillLevel, null);
}
@PowerNukkitXOnly
@Since("1.19.21-r4")
public void setFillLevel(int fillLevel, @Nullable Player player) {
if (fillLevel == getFillLevel()) return;
setIntValue(FILL_LEVEL, fillLevel);
if (fillLevel > getFillLevel()) {
this.level.getVibrationManager().callVibrationEvent(new VibrationEvent(player != null ? player : this, this.add(0.5, 0.5, 0.5), VibrationType.FLUID_PLACE));
} else {
this.level.getVibrationManager().callVibrationEvent(new VibrationEvent(player != null ? player : this, this.add(0.5, 0.5, 0.5), VibrationType.FLUID_PICKUP));
}
}
@PowerNukkitXOnly
@Since("1.6.0.0-PNX")
public CauldronLiquid getCauldronLiquid() {
return this.getPropertyValue(LIQUID);
}
@PowerNukkitXOnly
@Since("1.6.0.0-PNX")
public void setCauldronLiquid(CauldronLiquid liquid) {
this.setPropertyValue(LIQUID, liquid);
}
@Override
public boolean onActivate(@NotNull Item item, Player player) {
BlockEntityCauldron cauldron = getBlockEntity();
if (cauldron == null) {
return false;
}
switch (item.getId()) {
case Item.BUCKET:
ItemBucket bucket = (ItemBucket) item;
if (bucket.getFishEntityId() != null) {
break;
}
if (bucket.isEmpty()) {
if (!isFull() || cauldron.isCustomColor() || cauldron.hasPotion()) {
break;
}
PlayerBucketFillEvent ev = new PlayerBucketFillEvent(player, this, null, this, item, MinecraftItemID.WATER_BUCKET.get(1, bucket.getCompoundTag()));
this.level.getServer().getPluginManager().callEvent(ev);
if (!ev.isCancelled()) {
replaceBucket(bucket, player, ev.getItem());
this.setFillLevel(FILL_LEVEL.getMinValue(), player);//empty
this.level.setBlock(this, this, true);
cauldron.clearCustomColor();
this.getLevel().addLevelEvent(this.add(0.5, 0.375 + getFillLevel() * 0.125, 0.5), LevelEventPacket.EVENT_CAULDRON_TAKE_WATER);
}
} else if (bucket.isWater() || bucket.isLava() || bucket.isPowderSnow()) {
if (isFull() && !cauldron.isCustomColor() && !cauldron.hasPotion() && item.getDamage() == 8) {
break;
}
PlayerBucketEmptyEvent ev = new PlayerBucketEmptyEvent(player, this, null, this, item, MinecraftItemID.BUCKET.get(1, bucket.getCompoundTag()));
this.level.getServer().getPluginManager().callEvent(ev);
if (!ev.isCancelled()) {
if (player.isSurvival() || player.isAdventure()) {
replaceBucket(bucket, player, ev.getItem());
}
if (cauldron.hasPotion()) {//if has potion
clearWithFizz(cauldron, player);
} else if (bucket.isWater()) { //water bucket
this.setFillLevel(FILL_LEVEL.getMaxValue(), player);//fill
//default liquid type is water so we don't need to set it
cauldron.clearCustomColor();
this.level.setBlock(this, this, true);
this.getLevel().addSound(this.add(0.5, 1, 0.5), Sound.CAULDRON_FILLWATER);
} else if (bucket.isPowderSnow()) { // powder snow bucket
this.setFillLevel(FILL_LEVEL.getMaxValue(), player);//fill
this.setCauldronLiquid(CauldronLiquid.POWDER_SNOW);
cauldron.clearCustomColor();
this.level.setBlock(this, this, true);
//todo: add the sound of powder snow (I can't find it)
} else { // lava bucket
if (isEmpty()) {
BlockCauldronLava cauldronLava = new BlockCauldronLava(0xE);
cauldronLava.setFillLevel(FILL_LEVEL.getMaxValue(), player);
this.level.setBlock(this, cauldronLava, true, true);
cauldron.clearCustomColor();
cauldron.setType(BlockEntityCauldron.PotionType.LAVA);
this.getLevel().addSound(this.add(0.5, 1, 0.5), Sound.BUCKET_EMPTY_LAVA);
} else {
clearWithFizz(cauldron, player);
}
}
//this.update();
}
}
break;
case ItemID.DYE:
if (isEmpty() || cauldron.hasPotion()) {
break;
}
if (player.isSurvival() || player.isAdventure()) {
item.setCount(item.getCount() - 1);
player.getInventory().setItemInHand(item);
}
BlockColor color = new ItemDye(item.getDamage()).getDyeColor().getLeatherColor();
if (!cauldron.isCustomColor()) {
cauldron.setCustomColor(color);
} else {
BlockColor current = cauldron.getCustomColor();
BlockColor mixed = new BlockColor(
(int)Math.round(Math.sqrt(color.getRed() * current.getRed()) * 0.965),
(int)Math.round(Math.sqrt(color.getGreen() * current.getGreen()) * 0.965),
(int)Math.round(Math.sqrt(color.getBlue() * current.getBlue()) * 0.965)
);
cauldron.setCustomColor(mixed);
}
this.level.addSound(this.add(0.5, 0.5, 0.5), Sound.CAULDRON_ADDDYE);
break;
case ItemID.LEATHER_CAP:
case ItemID.LEATHER_TUNIC:
case ItemID.LEATHER_PANTS:
case ItemID.LEATHER_BOOTS:
case ItemID.LEATHER_HORSE_ARMOR:
if (isEmpty() || cauldron.hasPotion()) {
break;
}
if (cauldron.isCustomColor()) {
CompoundTag compoundTag = item.hasCompoundTag() ? item.getNamedTag() : new CompoundTag();
compoundTag.putInt("customColor", cauldron.getCustomColor().getRGB());
item.setCompoundTag(compoundTag);
player.getInventory().setItemInHand(item);
setFillLevel(FILL_LEVEL.clamp(getFillLevel() - 2), player);
this.level.setBlock(this, this, true, true);
this.level.addSound(add(0.5, 0.5, 0.5), Sound.CAULDRON_DYEARMOR);
} else {
if (!item.hasCompoundTag()) {
break;
}
CompoundTag compoundTag = item.getNamedTag();
if (!compoundTag.exist("customColor")) {
break;
}
compoundTag.remove("customColor");
item.setCompoundTag(compoundTag);
player.getInventory().setItemInHand(item);
setFillLevel(FILL_LEVEL.clamp(getFillLevel() - 2), player);
this.level.setBlock(this, this, true, true);
this.getLevel().addSound(this.add(0.5, 1, 0.5), Sound.CAULDRON_TAKEWATER);
}
break;
case ItemID.POTION:
case ItemID.SPLASH_POTION:
case ItemID.LINGERING_POTION:
if (!isEmpty() && (cauldron.hasPotion()? cauldron.getPotionId() != item.getDamage() : item.getDamage() != 0)) {
clearWithFizz(cauldron, player);
consumePotion(item, player);
break;
}
if (isFull()) {
break;
}
if (item.getDamage() != 0 && isEmpty()) {
cauldron.setPotionId(item.getDamage());
}
cauldron.setType(
item.getId() == ItemID.POTION? BlockEntityCauldron.PotionType.NORMAL :
item.getId() == ItemID.SPLASH_POTION? BlockEntityCauldron.PotionType.SPLASH :
BlockEntityCauldron.PotionType.LINGERING
);
cauldron.spawnToAll();
setFillLevel(FILL_LEVEL.clamp(getFillLevel() + 2), player);
this.level.setBlock(this, this, true);
consumePotion(item, player);
this.level.addLevelEvent(this.add(0.5, 0.375 + getFillLevel() * 0.125, 0.5), LevelEventPacket.EVENT_CAULDRON_FILL_POTION);
break;
case ItemID.GLASS_BOTTLE:
if (isEmpty()) {
break;
}
int meta = cauldron.hasPotion() ? cauldron.getPotionId() : 0;
Item potion;
if (meta == 0) {
potion = new ItemPotion();
} else {
switch (cauldron.getType()) {
case SPLASH:
potion = new ItemPotionSplash(meta);
break;
case LINGERING:
potion = new ItemPotionLingering(meta);
break;
case NORMAL:
default:
potion = new ItemPotion(meta);
break;
}
}
setFillLevel(FILL_LEVEL.clamp(getFillLevel() - 2), player);
if (isEmpty()) {
cauldron.setPotionId(-1);//reset potion
cauldron.clearCustomColor();
}
this.level.setBlock(this, this, true);
boolean consumeBottle = player.isSurvival() || player.isAdventure();
if (consumeBottle && item.getCount() == 1) {
player.getInventory().setItemInHand(potion);
} else if (item.getCount() > 1) {
if (consumeBottle) {
item.setCount(item.getCount() - 1);
player.getInventory().setItemInHand(item);
}
if (player.getInventory().canAddItem(potion)) {
player.getInventory().addItem(potion);
} else {
player.getLevel().dropItem(player.add(0, 1.3, 0), potion, player.getDirectionVector().multiply(0.4));
}
}
this.level.addLevelEvent(this.add(0.5, 0.375 + getFillLevel() * 0.125, 0.5), LevelEventPacket.EVENT_CAULDRON_TAKE_POTION);
break;
case ItemID.BANNER:
if (isEmpty() || cauldron.isCustomColor() || cauldron.hasPotion()) {
break;
}
ItemBanner banner = (ItemBanner) item;
if (!banner.hasPattern()) {
break;
}
banner.removePattern(banner.getPatternsSize() - 1);
boolean consumeBanner = player.isSurvival() || player.isAdventure();
if (consumeBanner && item.getCount() < item.getMaxStackSize()) {
player.getInventory().setItemInHand(banner);
} else {
if (consumeBanner) {
item.setCount(item.getCount() - 1);
player.getInventory().setItemInHand(item);
}
if (player.getInventory().canAddItem(banner)) {
player.getInventory().addItem(banner);
} else {
player.getLevel().dropItem(player.add(0, 1.3, 0), banner, player.getDirectionVector().multiply(0.4));
}
}
setFillLevel(FILL_LEVEL.clamp(getFillLevel() - 2), player);
this.level.setBlock(this, this, true, true);
this.getLevel().addSound(this.add(0.5, 1, 0.5), Sound.CAULDRON_TAKEWATER);
break;
default:
if (item instanceof ItemDye) {
if (isEmpty() || cauldron.hasPotion()) {
break;
}
if (player.isSurvival() || player.isAdventure()) {
item.setCount(item.getCount() - 1);
player.getInventory().setItemInHand(item);
}
color = ((ItemDye)item).getDyeColor().getColor();
if (!cauldron.isCustomColor()) {
cauldron.setCustomColor(color);
} else {
BlockColor current = cauldron.getCustomColor();
BlockColor mixed = new BlockColor(
current.getRed() + (color.getRed() - current.getRed()) / 2,
current.getGreen() + (color.getGreen() - current.getGreen()) / 2,
current.getBlue() + (color.getBlue() - current.getBlue()) / 2
);
cauldron.setCustomColor(mixed);
}
this.level.addSound(this.add(0.5, 0.5, 0.5), Sound.CAULDRON_ADDDYE);
} else {
return true;
}
}
this.level.updateComparatorOutputLevel(this);
return true;
}
protected void replaceBucket(Item oldBucket, Player player, Item newBucket) {
if (player.isSurvival() || player.isAdventure()) {
if (oldBucket.getCount() == 1) {
player.getInventory().setItemInHand(newBucket);
} else {
oldBucket.setCount(oldBucket.getCount() - 1);
if (player.getInventory().canAddItem(newBucket)) {
player.getInventory().addItem(newBucket);
} else {
player.getLevel().dropItem(player.add(0, 1.3, 0), newBucket, player.getDirectionVector().multiply(0.4));
}
}
}
}
private void consumePotion(Item item, Player player) {
if (player.isSurvival() || player.isAdventure()) {
if (item.getCount() == 1) {
player.getInventory().setItemInHand(new ItemBlock(new BlockAir()));
} else if (item.getCount() > 1) {
item.setCount(item.getCount() - 1);
player.getInventory().setItemInHand(item);
Item bottle = new ItemGlassBottle();
if (player.getInventory().canAddItem(bottle)) {
player.getInventory().addItem(bottle);
} else {
player.getLevel().dropItem(player.add(0, 1.3, 0), bottle, player.getDirectionVector().multiply(0.4));
}
}
}
}
@PowerNukkitOnly
public void clearWithFizz(BlockEntityCauldron cauldron) {
clearWithFizz(cauldron, null);
}
public void clearWithFizz(BlockEntityCauldron cauldron, @Nullable Player player) {
this.setFillLevel(FILL_LEVEL.getMinValue(), player);//empty
cauldron.setPotionId(-1);//reset potion
cauldron.setType(BlockEntityCauldron.PotionType.NORMAL);
cauldron.clearCustomColor();
this.level.setBlock(this, new BlockCauldron(0), true);
this.level.addSound(this.add(0.5, 0, 0.5), Sound.RANDOM_FIZZ);
for (int i = 0; i < 8; ++i) {
this.getLevel().addParticle(new SmokeParticle(add(Math.random(), 1.2, Math.random())));
}
}
@Override
public boolean place(@NotNull Item item, @NotNull Block block, @NotNull Block target, @NotNull BlockFace face, double fx, double fy, double fz, Player player) {
CompoundTag nbt = new CompoundTag()
.putShort("PotionId", -1)
.putByte("SplashPotion", 0);
if (item.hasCustomBlockData()) {
Map customData = item.getCustomBlockData().getTags();
for (Map.Entry tag : customData.entrySet()) {
nbt.put(tag.getKey(), tag.getValue());
}
}
return BlockEntityHolder.setBlockAndCreateEntity(this, true, true, nbt) != null;
}
@Override
@PowerNukkitOnly
public int getToolTier() {
return ItemTool.TIER_WOODEN;
}
@Override
public Item toItem() {
return new ItemCauldron();
}
@Override
public boolean hasComparatorInputOverride() {
return true;
}
@Override
public int getComparatorInputOverride() {
return getFillLevel();
}
@Override
public boolean canHarvestWithHand() {
return false;
}
@Since("1.3.0.0-PN")
@PowerNukkitOnly
@Override
public boolean isSolid(BlockFace side) {
return false;
}
@PowerNukkitDifference(since = "1.4.0.0-PN", info = "Will return true")
@Override
public boolean isTransparent() {
return true;
}
@Since("1.4.0.0-PN")
@PowerNukkitOnly
@Override
public int getLightFilter() {
return 3;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy