Jump to content

Recommended Posts

Posted

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?

Posted (edited)

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.

Posted

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.

Posted

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.

Posted

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.

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



×
×
  • Create New...

Important Information

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