Jump to content

[1.14.4] LootCondition


Simon_kungen

Recommended Posts

Hi

 

I needed a way to check if a block was destroyed by an explosion or not, but I might have more uses with this if I make it more generic, as in checking if the block has been removed by  a certain thing such as a player or a Enderman, or by an explosion.

 

DestroyedBy:java

Spoiler

public class DestroyedBy implements ILootCondition
{

    private final String destroyer;

    private DestroyedBy(String destroyer)
    {
        this.destroyer = destroyer;
    }


    //public static final DestroyedBy INSTANCE = new DestroyedBy();


    @Override
    public boolean test(LootContext lootContext)
    {
        //System.out.println(destroyer);


        return false;

    }


    public static class Serializer extends ILootCondition.AbstractSerializer<DestroyedBy>
    {

        public Serializer()
        {
            super(new ResourceLocation(Reference.MODID,"destroyed_by"),DestroyedBy.class);
        }

        @Override
        public void serialize(JsonObject json, DestroyedBy value, JsonSerializationContext context)
        {
            json.addProperty("destroyer",value.destroyer);
        }

        @Override
        public DestroyedBy deserialize(JsonObject json, JsonDeserializationContext context)
        {
            return new DestroyedBy(json.has("destroyer") ? JSONUtils.getString(json,"destroyer") : "minecraft:player");
        }
    }
}

 

 

 

How do I get what destroyed a block?

Link to comment
Share on other sites

I'd say "have you looked at the LootContext object?" but I realize that it's not quite that straight forward.

You need to look at how they're created and what parameters get passed. What you're looking for here are the LootParameter objects, conveniently all enumerated in the LootParameters class. And one is called EXPLOSION_RADIUS.

 

If that parameter exists, the block was destroyed by an explosion, if not then no.

 

LootParameters.THIS_ENTITY will be the parameter that holds a reference to the player (if there is one, and may also be null) and LootParameters.TOOL will have the tool used (may be empty even when a player breaks a block).

Edited by Draco18s

Apparently I'm a complete and utter jerk and come to this forum just like to make fun of people, be confrontational, and make your personal life miserable.  If you think this is the case, JUST REPORT ME.  Otherwise you're just going to get reported when you reply to my posts and point it out, because odds are, I was trying to be nice.

 

Exception: If you do not understand Java, I WILL NOT HELP YOU and your thread will get locked.

 

DO NOT PM ME WITH PROBLEMS. No help will be given.

Link to comment
Share on other sites

Oof. I tried.

Apparently I'm a complete and utter jerk and come to this forum just like to make fun of people, be confrontational, and make your personal life miserable.  If you think this is the case, JUST REPORT ME.  Otherwise you're just going to get reported when you reply to my posts and point it out, because odds are, I was trying to be nice.

 

Exception: If you do not understand Java, I WILL NOT HELP YOU and your thread will get locked.

 

DO NOT PM ME WITH PROBLEMS. No help will be given.

Link to comment
Share on other sites

I did a kind of clunky solution to this:

 

switch (destroyer) {
    case "minecraft:explosion": {

        if (lootContext.get(LootParameters.THIS_ENTITY) == null) {
            System.out.println("It was a explosion (probably)!");
            return true;
        }
        break;

    }

    case "minecraft:player": {
        if (lootContext.get(LootParameters.THIS_ENTITY) instanceof PlayerEntity) {
            System.out.println("It's a player!");
            return true;
        }
        break;
    }
}

 

I simply went by extermination method narrowing it down until I'm pretty sure I know what it is. Something I noticed was that I have no way of checking if it was destroyed from the fill command or an explosion.

My second idea is to check in a 3x3 block radius after an explosion. What is that smoke effect? A temporary entity or something else?

 

But at this point, this feels like its getting far too complicated than it needs to be. The plan was to when Cobblestone explodes there is a chance for it to drop Gravel instead, and then blowing up Gravel can turn it into Sand. I did this function first, but I got stuck as there is no way of changing what a block drops, so I changed the blocks right before they blew up, too short for us to notice, but gets the effect of some of it turning into Gravel/Sand.

 

public static void onExplosion(final ExplosionEvent event)
{
    Random random = new Random();

    if (!event.getWorld().isRemote) {
        for (BlockPos pos : event.getExplosion().getAffectedBlockPositions()) {
            Block block = event.getWorld().getBlockState(pos).getBlock();

            if (BlockTags.getCollection().getOrCreate(cobblestoneID).contains(block)) {

                if (random.nextInt(9) == 0)
                    event.getWorld().setBlockState(pos, Blocks.GRAVEL.getDefaultState());

            } else if (BlockTags.getCollection().getOrCreate(gravelID).contains(block)) {
                if (random.nextInt(9) == 0)
                    event.getWorld().setBlockState(pos, Blocks.SAND.getDefaultState());
            }
        }
    }

}

 

But then there is the problem of some of the Gravel turning into Flint, which isn't supposed to happen.

Link to comment
Share on other sites

I ended up with this solution and created two new blocks just as a temporary block placement.

 

public static void onExplosion(final ExplosionEvent event)
{
    Random random = new Random();

    if (!event.getWorld().isRemote) {
        for (BlockPos pos : event.getExplosion().getAffectedBlockPositions()) {
            Block block = event.getWorld().getBlockState(pos).getBlock();

            if (BlockTags.getCollection().getOrCreate(cobblestoneID).contains(block)) {
                if (random.nextInt(9) == 0) {
                    if (random.nextInt(9) == 0)
                        event.getWorld().setBlockState(pos, IntercraftBlocks.SANDSUBSTITUTE.getDefaultState());
                    else
                        event.getWorld().setBlockState(pos, IntercraftBlocks.GRAVELSUBSTITUTE.getDefaultState());
                }


            } else if (BlockTags.getCollection().getOrCreate(gravelID).contains(block)) {
                if (random.nextInt(9) == 0) {
                    event.getWorld().setBlockState(pos, IntercraftBlocks.SANDSUBSTITUTE.getDefaultState());
                } else {
                    event.getWorld().setBlockState(pos, IntercraftBlocks.GRAVELSUBSTITUTE.getDefaultState());
                }
            }
        }
    }
}

 

I then did a generic loot table drop where it drops the vanilla variant. Something I was wondering though is how I would make this hardcoded drop value to be in JSON format? n% chance for this drop, and otherwise this drop instead. The only thing I found was the looting and fortune functions, but those only apply if broken with a tool. Do I have to make my own function for that? Because I would prefer doing it all in JSON format if it's possible.

Link to comment
Share on other sites

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.
Note: Your post will require moderator approval before it will be visible.

Guest
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Reply to this topic...

×   Pasted as rich text.   Restore formatting

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

Announcements



  • Recently Browsing

    • No registered users viewing this page.
  • Posts

    • They were already updated, and just to double check I even did a cleanup and fresh update from that same page. I'm quite sure drivers are not the problem here. 
    • i tried downloading the drivers but it says no AMD graphics hardware has been detected    
    • Update your AMD/ATI drivers - get the drivers from their website - do not update via system  
    • As the title says i keep on crashing on forge 1.20.1 even without any mods downloaded, i have the latest drivers (nvidia) and vanilla minecraft works perfectly fine for me logs: https://pastebin.com/5UR01yG9
    • Hello everyone, I'm making this post to seek help for my modded block, It's a special block called FrozenBlock supposed to take the place of an old block, then after a set amount of ticks, it's supposed to revert its Block State, Entity, data... to the old block like this :  The problem I have is that the system breaks when handling multi blocks (I tried some fix but none of them worked) :  The bug I have identified is that the function "setOldBlockFields" in the item's "setFrozenBlock" function gets called once for the 1st block of multiblock getting frozen (as it should), but gets called a second time BEFORE creating the first FrozenBlock with the data of the 1st block, hence giving the same data to the two FrozenBlock :   Old Block Fields set BlockState : Block{minecraft:black_bed}[facing=east,occupied=false,part=head] BlockEntity : net.minecraft.world.level.block.entity.BedBlockEntity@73681674 BlockEntityData : id:"minecraft:bed",x:3,y:-60,z:-6} Old Block Fields set BlockState : Block{minecraft:black_bed}[facing=east,occupied=false,part=foot] BlockEntity : net.minecraft.world.level.block.entity.BedBlockEntity@6d1aa3da BlockEntityData : {id:"minecraft:bed",x:2,y:-60,z:-6} Frozen Block Entity set BlockState : Block{minecraft:black_bed}[facing=east,occupied=false,part=foot] BlockPos{x=3, y=-60, z=-6} BlockEntity : net.minecraft.world.level.block.entity.BedBlockEntity@6d1aa3da BlockEntityData : {id:"minecraft:bed",x:2,y:-60,z:-6} Frozen Block Entity set BlockState : Block{minecraft:black_bed}[facing=east,occupied=false,part=foot] BlockPos{x=2, y=-60, z=-6} BlockEntity : net.minecraft.world.level.block.entity.BedBlockEntity@6d1aa3da BlockEntityData : {id:"minecraft:bed",x:2,y:-60,z:-6} here is the code inside my custom "freeze" item :    @Override     public @NotNull InteractionResult useOn(@NotNull UseOnContext pContext) {         if (!pContext.getLevel().isClientSide() && pContext.getHand() == InteractionHand.MAIN_HAND) {             BlockPos blockPos = pContext.getClickedPos();             BlockPos secondBlockPos = getMultiblockPos(blockPos, pContext.getLevel().getBlockState(blockPos));             if (secondBlockPos != null) {                 createFrozenBlock(pContext, secondBlockPos);             }             createFrozenBlock(pContext, blockPos);             return InteractionResult.SUCCESS;         }         return super.useOn(pContext);     }     public static void createFrozenBlock(UseOnContext pContext, BlockPos blockPos) {         BlockState oldState = pContext.getLevel().getBlockState(blockPos);         BlockEntity oldBlockEntity = oldState.hasBlockEntity() ? pContext.getLevel().getBlockEntity(blockPos) : null;         CompoundTag oldBlockEntityData = oldState.hasBlockEntity() ? oldBlockEntity.serializeNBT() : null;         if (oldBlockEntity != null) {             pContext.getLevel().removeBlockEntity(blockPos);         }         BlockState FrozenBlock = setFrozenBlock(oldState, oldBlockEntity, oldBlockEntityData);         pContext.getLevel().setBlockAndUpdate(blockPos, FrozenBlock);     }     public static BlockState setFrozenBlock(BlockState blockState, @Nullable BlockEntity blockEntity, @Nullable CompoundTag blockEntityData) {         BlockState FrozenBlock = BlockRegister.FROZEN_BLOCK.get().defaultBlockState();         ((FrozenBlock) FrozenBlock.getBlock()).setOldBlockFields(blockState, blockEntity, blockEntityData);         return FrozenBlock;     }  
  • Topics

×
×
  • Create New...

Important Information

By using this site, you agree to our Terms of Use.