/* * This file is part of LanternServer, licensed under the MIT License (MIT). * * Copyright (c) LanternPowered <https://www.lanternpowered.org> * Copyright (c) SpongePowered <https://www.spongepowered.org> * 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.lanternpowered.server.block.behavior.vanilla; import static com.google.common.base.Preconditions.checkNotNull; import org.lanternpowered.server.behavior.Behavior; import org.lanternpowered.server.behavior.BehaviorContext; import org.lanternpowered.server.behavior.BehaviorResult; import org.lanternpowered.server.behavior.Parameters; import org.lanternpowered.server.behavior.pipeline.BehaviorPipeline; import org.lanternpowered.server.block.BlockSnapshotBuilder; import org.lanternpowered.server.block.behavior.types.InteractWithBlockBehavior; import org.lanternpowered.server.text.LanternTexts; import org.spongepowered.api.data.key.Keys; import org.spongepowered.api.item.inventory.ItemStack; import org.spongepowered.api.text.Text; import org.spongepowered.api.world.Location; import org.spongepowered.api.world.World; import java.util.Optional; import javax.annotation.Nullable; public class OpenableInteractionBehavior implements InteractWithBlockBehavior { public static OpenCloseAccess DEFAULT_ACCESS = (context, action, original) -> { final Optional<String> optLockToken = context.tryGet(Parameters.BLOCK_LOCATION).get(Keys.LOCK_TOKEN); if (optLockToken.isPresent()) { final Optional<ItemStack> usedItem = context.get(Parameters.USED_ITEM_STACK); if (usedItem.isPresent()) { final ItemStack itemStack = usedItem.get(); final Optional<String> optItemLockToken = itemStack.get(Keys.LOCK_TOKEN); if (optItemLockToken.isPresent()) { return optItemLockToken.get().equals(optLockToken.get()); } final Optional<Text> optName = itemStack.get(Keys.DISPLAY_NAME); if (optName.isPresent()) { return LanternTexts.toLegacy(optName.get()).equals(optLockToken.get()); } } return false; } return true; }; private final OpenCloseAccess access; public OpenableInteractionBehavior(OpenCloseAccess access) { this.access = checkNotNull(access, "access"); } public OpenableInteractionBehavior() { this.access = DEFAULT_ACCESS; } @Override public BehaviorResult tryInteract(BehaviorPipeline<Behavior> pipeline, BehaviorContext context) { final Location<World> location = context.tryGet(Parameters.BLOCK_LOCATION); final Optional<Boolean> optIsOpen = location.get(Keys.OPEN); if (!optIsOpen.isPresent()) { return BehaviorResult.FAIL; } final boolean isOpen = optIsOpen.get(); final boolean success = this.access.get(context, isOpen ? OpenCloseAccess.Action.CLOSE : OpenCloseAccess.Action.OPEN, DEFAULT_ACCESS); if (!success) { return BehaviorResult.PASS; } final BlockSnapshotBuilder builder = BlockSnapshotBuilder.create(); builder.from(location); context.populateBlockSnapshot(builder, BehaviorContext.PopulationFlags.NOTIFIER); builder.add(Keys.OPEN, !isOpen); context.addBlockChange(builder.build()); return BehaviorResult.SUCCESS; } @FunctionalInterface interface OpenCloseAccess { enum Action { OPEN, CLOSE, } /** * Checks whether the block can be opened or closed for the specific {@link BehaviorContext}. * * @param context The context that should be handled * @param action The action that should occur, when closed before, this will be open and closed if it was open * @return Whether the action may happen */ boolean get(BehaviorContext context, Action action, @Nullable OpenCloseAccess original); } }