Minestom/src/main/java/net/minestom/server/event/EventListener.java

181 lines
6.2 KiB
Java
Raw Normal View History

2021-05-11 04:52:57 +02:00
package net.minestom.server.event;
import net.minestom.server.event.trait.CancellableEvent;
2021-06-08 14:15:30 +02:00
import org.jetbrains.annotations.Contract;
2021-05-11 04:52:57 +02:00
import org.jetbrains.annotations.NotNull;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
2021-05-11 04:52:57 +02:00
import java.util.function.Consumer;
import java.util.function.Predicate;
2021-06-08 22:58:48 +02:00
/**
* Represents an event listener (handler) in an event graph.
* <p>
* A listener is responsible for executing some action based on an event triggering.
*
* @param <T> The event type being handled.
*/
2021-06-02 21:14:48 +02:00
public interface EventListener<T extends Event> {
2021-05-11 04:52:57 +02:00
2021-06-02 21:14:48 +02:00
@NotNull Class<T> getEventType();
2021-05-11 04:52:57 +02:00
2021-06-02 21:14:48 +02:00
@NotNull Result run(@NotNull T event);
2021-05-11 04:52:57 +02:00
2021-06-08 14:15:30 +02:00
@Contract(pure = true)
static <T extends Event> EventListener.@NotNull Builder<T> builder(@NotNull Class<T> eventType) {
2021-06-02 08:17:03 +02:00
return new EventListener.Builder<>(eventType);
}
2021-06-08 22:58:48 +02:00
/**
* Create an event listener without any special options. The given listener will be executed
* if the event passes all parent filtering.
*
* @param eventType The event type to handle
* @param listener The handler function
* @param <T> The event type to handle
2021-06-08 22:58:48 +02:00
* @return An event listener with the given properties
*/
2021-06-08 14:15:30 +02:00
@Contract(pure = true)
static <T extends Event> @NotNull EventListener<T> of(@NotNull Class<T> eventType, @NotNull Consumer<@NotNull T> listener) {
2021-06-02 21:14:48 +02:00
return new EventListener<>() {
@Override
public @NotNull Class<T> getEventType() {
return eventType;
}
@Override
public @NotNull Result run(@NotNull T event) {
listener.accept(event);
return Result.SUCCESS;
}
};
2021-06-02 20:56:47 +02:00
}
2021-06-02 21:14:48 +02:00
class Builder<T extends Event> {
2021-05-11 04:52:57 +02:00
private final Class<T> eventType;
2021-06-02 20:23:50 +02:00
private final List<Predicate<T>> filters = new ArrayList<>();
2021-06-09 22:01:46 +02:00
private boolean ignoreCancelled = true;
2021-06-08 14:07:27 +02:00
private int expireCount;
private Predicate<T> expireWhen;
2021-05-11 04:52:57 +02:00
private Consumer<T> handler;
protected Builder(Class<T> eventType) {
this.eventType = eventType;
}
2021-06-08 22:58:48 +02:00
/**
* Adds a filter to the executor of this listener. The executor will only
* be called if this condition passes on the given event.
*/
2021-06-08 14:15:30 +02:00
@Contract(value = "_ -> this")
public @NotNull EventListener.Builder<T> filter(Predicate<T> filter) {
2021-05-11 04:52:57 +02:00
this.filters.add(filter);
return this;
}
/**
* Specifies if the handler should still be called if {@link CancellableEvent#isCancelled()} returns {@code true}.
* <p>
2021-06-09 22:01:46 +02:00
* Default is set to {@code true}.
*
2021-06-09 22:01:46 +02:00
* @param ignoreCancelled True to stop processing the event when cancelled
*/
@Contract(value = "_ -> this")
public @NotNull EventListener.Builder<T> ignoreCancelled(boolean ignoreCancelled) {
this.ignoreCancelled = ignoreCancelled;
return this;
}
2021-06-08 22:58:48 +02:00
/**
* Removes this listener after it has been executed the given number of times.
*
* @param expireCount The number of times to execute
*/
2021-06-08 14:15:30 +02:00
@Contract(value = "_ -> this")
public @NotNull EventListener.Builder<T> expireCount(int expireCount) {
2021-06-08 14:07:27 +02:00
this.expireCount = expireCount;
return this;
}
2021-06-08 22:58:48 +02:00
/**
* Expires this listener when it passes the given condition. The expiration will
* happen before the event is executed.
*
* @param expireWhen The condition to test
*/
2021-06-08 14:15:30 +02:00
@Contract(value = "_ -> this")
public @NotNull EventListener.Builder<T> expireWhen(Predicate<T> expireWhen) {
2021-06-08 14:07:27 +02:00
this.expireWhen = expireWhen;
2021-06-02 20:23:50 +02:00
return this;
}
2021-06-08 22:58:48 +02:00
/**
* Sets the handler for this event listener. This will be executed if the listener passes
* all conditions.
*/
2021-06-08 14:15:30 +02:00
@Contract(value = "_ -> this")
public @NotNull EventListener.Builder<T> handler(Consumer<T> handler) {
2021-05-11 04:52:57 +02:00
this.handler = handler;
return this;
}
2021-06-08 14:15:30 +02:00
@Contract(value = "-> new", pure = true)
2021-06-02 21:14:48 +02:00
public @NotNull EventListener<T> build() {
final boolean ignoreCancelled = this.ignoreCancelled;
2021-06-08 14:07:27 +02:00
AtomicInteger expirationCount = new AtomicInteger(this.expireCount);
final boolean hasExpirationCount = expirationCount.get() > 0;
2021-06-08 14:15:30 +02:00
final Predicate<T> expireWhen = this.expireWhen;
final var filters = new ArrayList<>(this.filters);
final var handler = this.handler;
2021-06-02 21:14:48 +02:00
return new EventListener<>() {
@Override
public @NotNull Class<T> getEventType() {
return eventType;
2021-05-11 04:52:57 +02:00
}
2021-06-02 21:14:48 +02:00
@Override
public @NotNull Result run(@NotNull T event) {
// Event cancellation
2021-06-09 22:01:46 +02:00
if (ignoreCancelled && event instanceof CancellableEvent &&
((CancellableEvent) event).isCancelled()) {
return Result.INVALID;
}
2021-06-08 14:07:27 +02:00
// Expiration predicate
if (expireWhen != null && expireWhen.test(event)) {
return Result.EXPIRED;
}
2021-06-02 21:14:48 +02:00
// Filtering
if (!filters.isEmpty()) {
2021-06-10 11:53:50 +02:00
for (var filter : filters) {
if (!filter.test(event)) {
// Cancelled
return Result.INVALID;
}
2021-06-02 21:14:48 +02:00
}
}
// Handler
if (handler != null) {
handler.accept(event);
}
2021-06-08 14:07:27 +02:00
// Expiration count
2021-06-02 21:14:48 +02:00
if (hasExpirationCount && expirationCount.decrementAndGet() == 0) {
return Result.EXPIRED;
}
return Result.SUCCESS;
}
2021-06-02 21:14:48 +02:00
};
2021-05-11 04:52:57 +02:00
}
}
enum Result {
SUCCESS,
INVALID,
2021-06-09 14:05:37 +02:00
EXPIRED,
EXCEPTION
}
2021-05-11 04:52:57 +02:00
}