Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.spongepowered.api.event.Cause 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;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.spongepowered.api.util.CopyableBuilder;
import org.spongepowered.api.util.annotation.DoNotStore;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.Optional;
import java.util.StringJoiner;
import java.util.stream.Collectors;
/**
* A cause represents the reason or initiator of an event.
*
* For example, if a block of sand is placed where it drops, the block of
* sand would create a falling sand entity, which then would place another block
* of sand. The block place event for the final block of sand would have the
* cause chain of the block of sand -> falling sand entity.
*
* It is not possible to accurately describe the chain of causes in all
* scenarios so a best effort approach is generally acceptable. For example, a
* player might press a lever, activating a complex Redstone circuit, which
* would then launch TNT and cause the destruction of some blocks, but tracing
* this event would be too complicated and thus may not be attempted.
*/
@DoNotStore
@SuppressWarnings("unchecked")
public final class Cause implements Iterable {
/**
* Creates a new {@link Builder} to make a new {@link Cause}.
*
* @return The new builder
*/
public static Builder builder() {
return new Builder();
}
/**
* Constructs a new cause with the specified event context and cause.
*
* @param ctx The event context
* @param cause The direct object cause
* @return The constructed cause
*/
public static Cause of(final EventContext ctx, final Object cause) {
java.util.Objects.requireNonNull(ctx, "Context");
java.util.Objects.requireNonNull(cause, "Cause cannot be null!");
return new Cause(ctx, new Object[] {cause});
}
/**
* Constructs a new cause with the specified event context and causes.
*
* @param ctx The event context
* @param cause The direct object cause
* @param causes Other associated causes
* @return The built cause
*/
public static Cause of(final EventContext ctx, final Object cause, final Object... causes) {
java.util.Objects.requireNonNull(ctx, "Context");
final Builder builder = Cause.builder();
builder.append(cause);
for (final Object namedCause : causes) {
builder.append(namedCause);
}
return builder.build(ctx);
}
/**
* Constructs a new cause with the specified event context and causes.
*
* @param ctx The event context
* @param iterable The associated causes
* @return The built cause
*/
public static Cause of(final EventContext ctx, final Iterable iterable) {
java.util.Objects.requireNonNull(ctx, "Context");
final Builder builder = Cause.builder();
for (final Object cause : iterable) {
builder.append(cause);
}
return builder.build(ctx);
}
final Object[] cause;
private final EventContext context;
// lazy load
@Nullable private List immutableCauses;
/**
* Constructs a new cause.
*
* @param ctx The event context
* @param causes The causes
*/
Cause(final EventContext ctx, final Object[] causes) {
java.util.Objects.requireNonNull(ctx, "Context");
final Object[] objects = new Object[causes.length];
for (int index = 0; index < causes.length; index++) {
objects[index] = java.util.Objects.requireNonNull(causes[index], "Null cause element!");
}
this.cause = objects;
this.context = ctx;
}
/**
* Constructs a new cause.
*
* @param ctx The event context
* @param causes The causes
*/
Cause(final EventContext ctx, final Collection causes) {
java.util.Objects.requireNonNull(ctx, "Context");
final Object[] objects = new Object[causes.size()];
int index = 0;
for (final Object cause : causes) {
objects[index++] = java.util.Objects.requireNonNull(cause, "Null cause element!");
}
this.cause = objects;
this.context = ctx;
}
/**
* Gets the event context relating to this cause.
*
* @return The event context
*/
public EventContext context() {
return this.context;
}
/**
* Gets the root {@link Object} of this cause.
*
* @return The root object cause for this cause
*/
public Object root() {
return this.cause[0];
}
/**
* Gets the first T
object of this {@link Cause}, if available.
*
* @param target The class of the target type
* @param The type of object being queried for
* @return The first element of the type, if available
*/
public Optional first(final Class target) {
for (final Object aCause : this.cause) {
if (target.isInstance(aCause)) {
return Optional.of((T) aCause);
}
}
return Optional.empty();
}
/**
* Gets the last object instance of the {@link Class} of type
* T
.
*
* @param target The class of the target type
* @param The type of object being queried for
* @return The last element of the type, if available
*/
public Optional last(final Class target) {
for (int i = this.cause.length - 1; i >= 0; i--) {
if (target.isInstance(this.cause[i])) {
return Optional.of((T) this.cause[i]);
}
}
return Optional.empty();
}
/**
* Gets the object immediately before the object that is an instance of the
* {@link Class} passed in.
*
* @param clazz The class of the object
* @return The object
*/
public Optional> before(final Class> clazz) {
if (clazz == null) {
throw new IllegalArgumentException("The provided class cannot be null!");
}
if (this.cause.length == 1) {
return Optional.empty();
}
for (int i = 0; i < this.cause.length; i++) {
if (clazz.isInstance(this.cause[i]) && i > 0) {
return Optional.of(this.cause[i - 1]);
}
}
return Optional.empty();
}
/**
* Gets the object immediately after the object that is an instance of the
* {@link Class} passed in.
*
* @param clazz The class to type check
* @return The object after, if available
*/
public Optional> after(final Class> clazz) {
if (clazz == null) {
throw new IllegalArgumentException("The provided class cannot be null!");
}
if (this.cause.length == 1) {
return Optional.empty();
}
for (int i = 0; i < this.cause.length; i++) {
if (clazz.isInstance(this.cause[i]) && i + 1 < this.cause.length) {
return Optional.of(this.cause[i + 1]);
}
}
return Optional.empty();
}
/**
* Returns whether the target class matches any object of this {@link Cause}.
*
* @param target The class of the target type
* @return True if found, false otherwise
*/
public boolean containsType(final Class> target) {
java.util.Objects.requireNonNull(target, "The provided class cannot be null!");
for (final Object aCause : this.cause) {
if (target.isInstance(aCause)) {
return true;
}
}
return false;
}
/**
* Checks if this cause contains of any of the provided {@link Object}. This
* is the equivalent to checking based on {@link #equals(Object)} for each
* object in this cause.
*
* @param object The object to check if it is contained
* @return True if the object is contained within this cause
*/
public boolean contains(final Object object) {
for (final Object aCause : this.cause) {
if (aCause.equals(object)) {
return true;
}
}
return false;
}
/**
* Gets an immutable {@link List} of all objects that are instances of the
* given {@link Class} type T
.
*
* @param The type of objects to query for
* @param target The class of the target type
* @return An immutable list of the objects queried
*/
public List allOf(final Class target) {
return Arrays.stream(this.cause).filter(target::isInstance).map(entry -> (T) entry).collect(Collectors.toUnmodifiableList());
}
/**
* Gets an immutable {@link List} with all object causes that are not
* instances of the provided {@link Class}.
*
* @param ignoredClass The class of object types to ignore
* @return The list of objects not an instance of the provided class
*/
public List noneOf(final Class> ignoredClass) {
return Arrays.stream(this.cause).filter(entry -> !ignoredClass.isInstance(entry)).collect(Collectors.toUnmodifiableList());
}
/**
* Gets an {@link List} of all causes within this {@link Cause}.
*
* @return An immutable list of all the causes
*/
public List all() {
if (this.immutableCauses == null) {
this.immutableCauses = List.of(this.cause);
}
return this.immutableCauses;
}
/**
* Creates a new {@link Cause} where the objects are added at the end of the
* cause array of objects.
*
* @param additional The additional object to add
* @return The new cause
*/
public Cause with(final Object additional) {
java.util.Objects.requireNonNull(additional, "No null arguments allowed!");
final List list = new ArrayList<>();
list.add(additional);
return this.with(list);
}
/**
* Creates a new {@link Cause} where the objects are added at the end of the
* cause array of objects.
*
* @param additional The additional object to add
* @param additionals The remaining objects to add
* @return The new cause
*/
public Cause with(final Object additional, final Object... additionals) {
java.util.Objects.requireNonNull(additional, "No null arguments allowed!");
final List list = new ArrayList<>();
list.add(additional);
for (final Object object : additionals) {
java.util.Objects.requireNonNull(object, "Cannot add null objects!");
list.add(object);
}
return this.with(list);
}
/**
* Creates a new {@link Cause} where the objects are added at the end of the
* cause array of objects.
*
* @param iterable The additional objects
* @return The new cause
*/
public Cause with(final Iterable iterable) {
final Cause.Builder builder = new Builder().from(this);
for (final Object o : iterable) {
java.util.Objects.requireNonNull(o, "Cannot add null causes");
builder.append(o);
}
return builder.build(this.context);
}
/**
* Merges this cause with the other cause.
*
* @param cause The cause to merge with this
* @return The new merged cause
*/
public Cause with(final Cause cause) {
final Cause.Builder builder = Cause.builder().from(this);
for (int i = 0; i < cause.cause.length; i++) {
builder.append(cause.cause[i]);
}
return builder.build(this.context);
}
@Override
public Iterator iterator() {
return new Itr();
}
@Override
public boolean equals(final @Nullable Object object) {
if (object instanceof Cause) {
final Cause cause = ((Cause) object);
return Arrays.equals(this.cause, cause.cause);
}
return false;
}
@Override
public int hashCode() {
return Arrays.hashCode(this.cause);
}
@Override
public String toString() {
final String causeString = "Cause[Context=" + this.context.toString() + ", Stack={";
final StringJoiner joiner = new StringJoiner(", ");
for (int i = 0; i < this.cause.length; i++) {
joiner.add(this.cause[i].toString());
}
return causeString + joiner.toString() + "}]";
}
private class Itr implements Iterator {
private int index = 0;
Itr() { }
@Override
public Object next() {
if (this.index >= Cause.this.cause.length) {
throw new NoSuchElementException();
}
return Cause.this.cause[this.index++];
}
@Override
public boolean hasNext() {
return this.index != Cause.this.cause.length;
}
}
public static final class Builder implements org.spongepowered.api.util.Builder, CopyableBuilder {
final List causes = new ArrayList<>();
Builder() {
}
/**
* Appends the specified object to the cause.
*
* @param cause The object to append to the cause.
* @return The modified builder, for chaining
*/
public Builder append(final Object cause) {
java.util.Objects.requireNonNull(cause, "Cause cannot be null!");
if (!this.causes.isEmpty() && this.causes.get(this.causes.size() - 1) == cause) {
return this;
}
this.causes.add(cause);
return this;
}
/**
* Inserts the specified object into the cause.
*
* @param position The position to insert into
* @param cause The object to insert into the cause
* @return The modified builder, for chaining
*/
public Builder insert(final int position, final Object cause) {
java.util.Objects.requireNonNull(cause, "Cause cannot be null!");
this.causes.add(position, cause);
return this;
}
/**
* Appends all specified objects onto the cause.
*
* @param causes The objects to add onto the cause
* @return The modified builder, for chaining
*/
public Builder appendAll(final Collection causes) {
java.util.Objects.requireNonNull(causes, "Causes cannot be null!");
causes.forEach(this::append);
return this;
}
@Override
public Builder from(final Cause value) {
for (int i = 0; i < value.cause.length; i++) {
this.causes.add(value.cause[i]);
}
return this;
}
@Override
public Builder reset() {
this.causes.clear();
return this;
}
@Override
public Cause build() {
if (this.causes.isEmpty()) {
throw new IllegalStateException("Cannot create an empty Cause!");
}
return new Cause(EventContext.empty(), this.causes);
}
/**
* Constructs a new {@link Cause} with information added to the builder.
*
* @param ctx The context to build the cause with
* @return The built cause
*/
public Cause build(final EventContext ctx) {
if (this.causes.isEmpty()) {
throw new IllegalStateException("Cannot create an empty Cause!");
}
return new Cause(ctx, this.causes);
}
}
}