/*
* This file is part of Skript.
*
* Skript is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Skript is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Skript. If not, see <http://www.gnu.org/licenses/>.
*
*
* Copyright 2011-2014 Peter Güttinger
*
*/
package ch.njol.skript.conditions;
import org.bukkit.event.Event;
import org.bukkit.inventory.Inventory;
import org.bukkit.inventory.ItemStack;
import org.eclipse.jdt.annotation.Nullable;
import ch.njol.skript.Skript;
import ch.njol.skript.aliases.ItemType;
import ch.njol.skript.doc.Description;
import ch.njol.skript.doc.Examples;
import ch.njol.skript.doc.Name;
import ch.njol.skript.doc.Since;
import ch.njol.skript.lang.Condition;
import ch.njol.skript.lang.Expression;
import ch.njol.skript.lang.Literal;
import ch.njol.skript.lang.SkriptParser.ParseResult;
import ch.njol.skript.log.ErrorQuality;
import ch.njol.util.Checker;
import ch.njol.util.Kleenean;
/**
* @author Peter Güttinger
*/
@Name("Can Hold")
@Description("Tests whether a player or a chest can hold the given item.")
@Examples({"block can hold 200 cobblestone",
"player has enough space for 64 feathers"})
@Since("1.0")
public class CondCanHold extends Condition {
static {
Skript.registerCondition(CondCanHold.class,
"%inventories% (can hold|ha(s|ve) [enough] space (for|to hold)) %itemtypes%",
"%inventories% (can(no|')t hold|(ha(s|ve) not|ha(s|ve)n't|do[es]n't have) [enough] space (for|to hold)) %itemtypes%");
}
@SuppressWarnings("null")
private Expression<Inventory> invis;
@SuppressWarnings("null")
Expression<ItemType> items;
@SuppressWarnings({"unchecked", "null"})
@Override
public boolean init(final Expression<?>[] vars, final int matchedPattern, final Kleenean isDelayed, final ParseResult parser) {
invis = (Expression<Inventory>) vars[0];
items = (Expression<ItemType>) vars[1];
if (items instanceof Literal) {
for (ItemType t : ((Literal<ItemType>) items).getAll()) {
t = t.getItem();
if (!(t.isAll() || (t.getTypes().size() == 1 && !t.getTypes().get(0).hasDataRange() && t.getTypes().get(0).getId() != -1))) {
Skript.error("The condition 'can hold' can currently only be used with aliases that start with 'every' or 'all', or only represent one item and one data value.", ErrorQuality.SEMANTIC_ERROR);
return false;
}
}
}
setNegated(matchedPattern == 1);
return true;
}
@Override
public boolean check(final Event e) {
return invis.check(e, new Checker<Inventory>() {
@Override
public boolean check(final Inventory invi) {
if (!items.getAnd()) {
return items.check(e, new Checker<ItemType>() {
@Override
public boolean check(final ItemType t) {
return t.getItem().hasSpace(invi);
}
}, isNegated());
}
final ItemStack[] buf = ItemType.getCopiedContents(invi);
return items.check(e, new Checker<ItemType>() {
@Override
public boolean check(final ItemType t) {
return t.getItem().addTo(buf);
}
}, isNegated());
}
});
}
@Override
public String toString(final @Nullable Event e, final boolean debug) {
return invis.toString(e, debug) + " can" + (isNegated() ? "'t" : "") + " hold " + items.toString(e, debug);
}
}