[SPIGOT-5140] Missing API - Chorus Flower being destroyed by arrow/trident Created: 05/Jul/19  Updated: 16/Jan/22  Resolved: 16/Jan/22

Status: Resolved
Project: Spigot
Component/s: None
Affects Version/s: None
Fix Version/s: None

Type: New Feature Priority: Minor
Reporter: Adam Assignee: Unassigned
Resolution: Fixed Votes: 4
Labels: 1.14, 1.14.1, 1.14.2, 1.14.3, 1.14.3-pre4, 1.15, 1.15.1, trident

Version: git-Spigot-595711b-c699792 (MC: 1.14.3) (Implementing API version 1.14.3-R0.1-SNAPSHOT)
Guidelines Read: Yes

 Description   

The problem is that there is no proper API to protect Chorus Flower from being destroyed by Arrow/Trident. 

 

Video: https://www.curseforge.com/minecraft/customization/arrowbreakchorusfruitflower

 

*Discussion on spigot: https://www.spigotmc.org/threads/spigotapi-missing.381491/



 Comments   
Comment by Marvin Rieple [ 16/Jan/22 ]

With the addition of the ability to cancel the ProjectileHitEvent you can now prevent chorus flower from breaking by cancelling the ProjectileHitEvent .

Comment by Florian CUNY [ 28/May/20 ]

I have sort of achieved protecting this block, but this is far from being both convenient and satisfactory. It can cause visual glitches or even a ghost block, but this is the best I could do.

    /**
     * Prevents Chorus Flowers from being broken by an arrow or a trident
     * @param e event
     */
    @EventHandler(priority = EventPriority.LOW, ignoreCancelled = true)
    public void onProjectileHitBreakBlock(ProjectileHitEvent e) {
        // We want to make sure this is an actual projectile (arrow or trident)
        if (!(e.getEntity() instanceof AbstractArrow)) {
            return;
        }

        // We want to make sure it hit a CHORUS_FLOWER
        if (e.getHitBlock() == null || !e.getHitBlock().getType().equals(Material.CHORUS_FLOWER)) {
            return;
        }

        // Find out who fired the arrow
        if (e.getEntity().getShooter() instanceof Player &&
                !checkIsland(e, (Player) e.getEntity().getShooter(), e.getHitBlock().getLocation(), Flags.BREAK_BLOCKS)) {
            final BlockData data = e.getHitBlock().getBlockData();
            // We seemingly can't prevent the block from being destroyed
            // So we need to put it back with a slight delay (yup, this is hacky - it makes the block flicker sometimes)
            e.getHitBlock().setType(Material.AIR); // prevents the block from dropping a chorus flower
            getPlugin().getServer().getScheduler().runTask(getPlugin(), () -> e.getHitBlock().setBlockData(data, true));
            // Sorry, this might also cause some ghost blocks!
        }
    }
Comment by xtomyserrax [ 28/May/20 ]

Hey, yeah no protection plugin is actually preventing that block from being broken.

I think that using EntityChangeBlockEvent would have a lot of sense and wouldnt break any stuff. For example, if you hit a TNT block with a burning arrow, this event will be called and you are able to cancel it and prevent the TNT from priming

Comment by Grzegorz [ 04/May/20 ]

Hello, I also have such a problem and I don't know how to solve it :/

Comment by Craig Parton [ 01/Jan/20 ]

I'm still not sure why people want to make the ProjectileHitEvent cancellable. Cancelling an event should actually stop the action named in the event. If it doesn't, or can't, that event shouldn't be cancellable or the event should be renamed. If I cancel a ProjectileHitEvent.. the projectile still hits the block. Yes you could code it to not process the hit if you cancel the event, but it goes against how the event is named and isn't very intuitive.

Another issue.. cancelling ProjectileHitEvent has the potential to cancel other events. For example, if an arrow hits an Entity and one plugin cancels that ProjectileHitEvent, then EntityDamageEntityEvent can't occur because the hit didn't happen. If other plugins are correctly monitoring EntityDamageEntityEvent for projectile hits, they won't even see that event occur because one plugin decided to cancel a completely different event. At least with the current system, if I cancel EntityDamageEntityEvent, other plugins still get it and can make their own decisions what to do. If the change is made, will I now have to somehow monitor ProjectileHitEvent to make sure some other plugin doesn't snipe my EntityDamageEntityEvent?

As I said before, there is already a perfectly named event, EntityChangeBlockEvent, that not only has the correct parameter set but SHOULD be triggered when this happens based on its name. 

Would it be difficult to add some checks in ProjectileHitEvent (if Entity == Arrow and Block.getType() == Material.CHORUS_PLANT) and then just create an EntityChangeBlockEvent if those are true? And yes I realize the actual code might look a little different. The event already has the correct parameter set, you'd just need to pass the arrow and chorus plant block to it.

Comment by Yannick Lamprecht [ 31/Dec/19 ]

Sure the first three.

Comment by RoboMWM [ 31/Dec/19 ]

Did you even read any of the comments before commenting...

Comment by Yannick Lamprecht [ 31/Dec/19 ]

Isn't this captured by ProjectileHitEvent? Add a possibility to set/get a EventResult or implement Cancleable?

Comment by Craig Parton [ 17/Jul/19 ]

Wooden buttons are also affected by arrows (they trigger when you shoot them)

When I was trying to implement this protection for chorus plants in my plugin I thought for sure it would trigger the EntityChangeBlockEvent. You have an Entity (arrow) changing a block (chorus plant). This would have been easy to implement too.. it's simple to get the player as the ProjectileSource from the arrow and the block directly from the event.

Making the ProjectileHitEvent cancellable makes a bit less sense. Cancelling the event doesn't stop the projectile from hitting the block.. the only way to really do that is to delete it or change it's velocity. It seems like that event was meant as just a notification. But cancelling the EntityChangeBlockEvent does make sense, because you are actually stopping the entity from changing the block. Just my $.02

Comment by md_5 [ 07/Jul/19 ]

Bells, Campfires and TNT are the other blocks affected by arrows.

Comment by Black Hole [ 07/Jul/19 ]

Or using world.capturedBlockStates for the block changes and call EntityChangeBlockEvent for them.

Comment by md_5 [ 07/Jul/19 ]

Arguably the best way to do this is to make the projectile event cancellable.

What exactly cancelling that event would do in the general case is a bit iffy though - if cancelling was defined as to just not run the hit code then that would likely be ok imo.

Generated at Thu Apr 03 15:33:33 UTC 2025 using Jira 10.3.3#10030003-sha1:d220e3fefc8dfc6d47f522d3b9a20c1455e12b7b.