All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.spongepowered.api.event.block.ChangeBlockEvent Maven / Gradle / Ivy

/*
 * This file is part of SpongeAPI, licensed under the MIT License (MIT).
 *
 * Copyright (c) SpongePowered 
 * Copyright (c) contributors
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package org.spongepowered.api.event.block;

import org.spongepowered.api.block.BlockSnapshot;
import org.spongepowered.api.block.BlockState;
import org.spongepowered.api.block.BlockType;
import org.spongepowered.api.block.BlockTypes;
import org.spongepowered.api.block.entity.BlockEntity;
import org.spongepowered.api.block.transaction.BlockTransaction;
import org.spongepowered.api.block.transaction.BlockTransactionReceipt;
import org.spongepowered.api.block.transaction.Operation;
import org.spongepowered.api.data.Transaction;
import org.spongepowered.api.entity.Entity;
import org.spongepowered.api.event.Cancellable;
import org.spongepowered.api.event.Cause;
import org.spongepowered.api.event.Event;
import org.spongepowered.api.world.server.ServerLocation;
import org.spongepowered.api.world.server.ServerWorld;

import java.util.List;
import java.util.Objects;
import java.util.function.Predicate;
import java.util.stream.Collectors;
import java.util.stream.Stream;

/**
 * Plain event for when one or many {@link BlockState BlockStates} may be
 * changing within a {@link org.spongepowered.api.world.server.ServerWorld}.
 * Ideally, the {@link All#transactions()} will contain a full list in order
 * of which the changes are taking place, but the overall list may not be the
 * full breadth of what changes take place during some complex operations. To
 * record the entirety of all transactions taking place, {@link Post} provides
 * additional aide with knowing which {@link BlockTransactionReceipt}s are
 * recorded in a particluar {@link org.spongepowered.api.world.server.ServerWorld}
 * without the additional cost of having multiple events thrown throughout.
 * 

To determine whether a particular {@link BlockTransaction} is a * {@code break}, {@code place}, {@code modify}, etc. please refer to the * {@link BlockTransaction#operation()}

*/ public interface ChangeBlockEvent extends Event { /** * Gets the world this event is affecting. * * @return The world encompassing these block changes */ ServerWorld world(); /** * Called before running specific block logic at one or more * {@link ServerLocation}'s such as {@link BlockTypes#WATER}. */ interface Pre extends ChangeBlockEvent, Cancellable { /** * Represents a list of one or more {@link ServerLocation}'s where * {@link BlockState} changes can occur. * *

Canceling this event will prevent block logic from running * and also stop {@link BlockSnapshot}'s from being generated.

* *

Note: This event is not intended to always be fired before * changing a {@link BlockState} but rather it is primarily used to * prevent one or more {@link BlockState}'s from being changed.

* * @return The immutable list of one or more locations that can change */ List locations(); } interface All extends ChangeBlockEvent, Cancellable { /** * Gets a list of the {@link Transaction}s for this event. If a * transaction is requested to be marked as "invalid", * {@link Transaction#setValid(boolean)} can be used. * * @return The unmodifiable list of transactions */ List transactions(); /** * Gets a {@link Stream} of {@link BlockTransaction BlockTransactions} * filtered by a particular {@link Operation block operation}. The difference * between this and {@link #transactions()} is that while the general transactions * is still an ordered {@link List}, this is a filtered stream of that list, equally * unmodifiable. The {@link BlockTransaction transactions} themselves are still * modifiable with {@code BlockTransaction#setCustom(BlockSnapshot)}, but altering * the customized snapshot will NOT alter the {@link Operation} * being performed. As a logical perspective, there is no functional difference * between any two {@link Operation operations}, but it can be important to * differentiate between the two in some contexts. * * @param operation The operation being performed * @return The stream filtering on the given operation, may be empty */ default Stream transactions(final Operation operation) { Objects.requireNonNull(operation, "Operation cannot be null"); return this.transactions() .stream() .filter(transaction -> transaction.operation().equals(operation)); } /** * Applies the provided {@link Predicate} to the {@link List} of * {@link Transaction}s from {@link #transactions()} such that * any time that {@link Predicate#test(Object)} returns {@code false} * on the location of the {@link Transaction}, the {@link Transaction} is * marked as "invalid" and will not apply post event. * *

{@link Transaction#original()} is used to get the {@link ServerLocation}

* * @param predicate The predicate to use for filtering * @return The transactions for which the predicate returned * {@code false} */ default List invalidate(final Predicate predicate) { return this.transactions() .stream() .filter(blockTransaction -> { if (!predicate.test(blockTransaction.original().location().get())) { blockTransaction.setValid(false); return true; } return false; }) .collect(Collectors.toList()); } /** * Invalidates the list as such that all {@link Transaction}s are * marked as "invalid" and will not apply post event. */ default void invalidateAll() { this.transactions().forEach(BlockTransaction::invalidate); } } /** * Called when there are multiple block changes due to a * {@link BlockType} having "ticked", in which the {@link Cause} will * have a {@link BlockSnapshot}, or, in the case that an {@link Entity} * has "ticked", in which the {@link Cause} will have an {@link Entity}, * or, in the case that a {@link BlockEntity} "ticked", the {@link Cause} * will have the {@link BlockEntity}. * *

The {@link Cause} may contain {@link Event}s, such as {@link All}, * {@link NotifyNeighborBlockEvent}, and {@link Pre}. These events may be * cancelled, or have their transactions modified, just like normal events, * but this event is set in stone (pun intended) with regards to the * representative state the world is in. Any Post is considered the "truth" * of what happened, after all {@link All} events may have been thrown for * a particular cause, during some logic, so as to simplify how many * batched events need to be thrown compared to the work being performed. *

* *

For example, a piston extension would cause this event to be fired. * A piston extension involves multiple distinct transactions - * the piston head moving, and the adjacent block being set in a new * position.

* *

Note: This event is fired after processing all other * ChangeBlockEvent's.

*/ interface Post extends ChangeBlockEvent { List receipts(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy