Skip to content

Commit

Permalink
Transfer API: Add slotted storage and non-empty iterator (#2908)
Browse files Browse the repository at this point in the history
* Transfer API: Add non-empty iterator

* Add SlottedStorage

* Add StorageUtil.extractAny

* Undeprecate ContainerItemContext.withInitial

* Add licenses

* Revert "Undeprecate ContainerItemContext.withInitial"

This reverts commit dcf123e.

* Tweaks

* Make SlottedStorage#getSlots return a view, remove useless field, add UnmodifiableView annotations

* Remove useless @inheritdoc

* Fix infinite loop in the tests
  • Loading branch information
Technici4n authored Apr 11, 2023
1 parent 36f9902 commit d51205d
Show file tree
Hide file tree
Showing 11 changed files with 333 additions and 23 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@

import org.jetbrains.annotations.ApiStatus;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.annotations.UnmodifiableView;

import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.entity.player.PlayerInventory;
Expand Down Expand Up @@ -293,5 +294,6 @@ default long exchange(ItemVariant newVariant, long maxAmount, TransactionContext
*
* @return An unmodifiable list containing additional slots of this context. If no additional slot is available, the list is empty.
*/
@UnmodifiableView
List<SingleSlotStorage<ItemVariant>> getAdditionalSlots();
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,14 +21,15 @@

import org.jetbrains.annotations.ApiStatus;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.annotations.UnmodifiableView;

import net.minecraft.entity.player.PlayerInventory;
import net.minecraft.inventory.Inventory;
import net.minecraft.inventory.SidedInventory;
import net.minecraft.inventory.SimpleInventory;
import net.minecraft.util.math.Direction;

import net.fabricmc.fabric.api.transfer.v1.storage.Storage;
import net.fabricmc.fabric.api.transfer.v1.storage.SlottedStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.base.CombinedStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.base.SingleSlotStorage;
import net.fabricmc.fabric.impl.transfer.item.InventoryStorageImpl;
Expand All @@ -50,7 +51,7 @@
*/
@ApiStatus.Experimental
@ApiStatus.NonExtendable
public interface InventoryStorage extends Storage<ItemVariant> {
public interface InventoryStorage extends SlottedStorage<ItemVariant> {
/**
* Return a wrapper around an {@link Inventory}.
*
Expand All @@ -69,11 +70,16 @@ static InventoryStorage of(Inventory inventory, @Nullable Direction direction) {
* Retrieve an unmodifiable list of the wrappers for the slots in this inventory.
* Each wrapper corresponds to a single slot in the inventory.
*/
@Override
@UnmodifiableView
List<SingleSlotStorage<ItemVariant>> getSlots();

/**
* Retrieve a wrapper around a specific slot of the inventory.
*/
@Override
default int getSlotCount() {
return getSlots().size();
}

@Override
default SingleSlotStorage<ItemVariant> getSlot(int slot) {
return getSlots().get(slot);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,10 +32,12 @@
import net.minecraft.util.math.Direction;

import net.fabricmc.fabric.api.lookup.v1.block.BlockApiLookup;
import net.fabricmc.fabric.api.transfer.v1.storage.base.SidedStorageBlockEntity;
import net.fabricmc.fabric.api.transfer.v1.item.base.SingleStackStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.SlottedStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.Storage;
import net.fabricmc.fabric.api.transfer.v1.storage.base.CombinedSlottedStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.base.CombinedStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.base.SidedStorageBlockEntity;
import net.fabricmc.fabric.impl.transfer.item.ComposterWrapper;
import net.fabricmc.fabric.mixin.transfer.DoubleInventoryAccessor;

Expand Down Expand Up @@ -119,10 +121,10 @@ private ItemStorage() {

// For double chests, we need to retrieve a wrapper for each part separately.
if (inventoryToWrap instanceof DoubleInventoryAccessor accessor) {
Storage<ItemVariant> first = InventoryStorage.of(accessor.fabric_getFirst(), direction);
Storage<ItemVariant> second = InventoryStorage.of(accessor.fabric_getSecond(), direction);
SlottedStorage<ItemVariant> first = InventoryStorage.of(accessor.fabric_getFirst(), direction);
SlottedStorage<ItemVariant> second = InventoryStorage.of(accessor.fabric_getSecond(), direction);

return new CombinedStorage<>(List.of(first, second));
return new CombinedSlottedStorage<>(List.of(first, second));
}
} else {
inventoryToWrap = inventory;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
/*
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package net.fabricmc.fabric.api.transfer.v1.storage;

import java.util.List;

import org.jetbrains.annotations.ApiStatus;
import org.jetbrains.annotations.UnmodifiableView;

import net.fabricmc.fabric.api.transfer.v1.storage.base.SingleSlotStorage;
import net.fabricmc.fabric.api.transfer.v1.context.ContainerItemContext;
import net.fabricmc.fabric.impl.transfer.TransferApiImpl;

/**
* A {@link Storage} implementation made of indexed slots.
*
* <p>Please note that some storages may not implement this interface.
* It is up to the storage implementation to decide whether to implement this interface or not.
* Checking whether a storage is slotted can be done using {@code instanceof}.
*
* @param <T> The type of the stored resources.
*
* <b>Experimental feature</b>, we reserve the right to remove or change it without further notice.
* The transfer API is a complex addition, and we want to be able to correct possible design mistakes.
*/
@ApiStatus.Experimental
public interface SlottedStorage<T> extends Storage<T> {
/**
* Retrieve the number of slots in this storage.
*/
int getSlotCount();

/**
* Retrieve a specific slot of this storage.
*
* @throws IndexOutOfBoundsException If the slot index is out of bounds.
*/
SingleSlotStorage<T> getSlot(int slot);

/**
* Retrieve a list containing all the slots of this storage. <b>The list must not be modified.</b>
*
* <p>This function can be used to interface with code that requires a slot list,
* for example {@link StorageUtil#insertStacking} or {@link ContainerItemContext#getAdditionalSlots()}.
*
* <p>It is guaranteed that calling this function is fast.
* The default implementation returns a view over the storage that delegates to {@link #getSlotCount} and {@link #getSlot}.
*/
@UnmodifiableView
default List<SingleSlotStorage<T>> getSlots() {
return TransferApiImpl.makeListView(this);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,39 @@ default long simulateExtract(T resource, long maxAmount, @Nullable TransactionCo
@Override
Iterator<StorageView<T>> iterator();

/**
* Same as {@link #iterator()}, but the iterator is guaranteed to skip over empty views,
* i.e. views that {@linkplain StorageView#isResourceBlank() contain blank resources} or have a zero {@linkplain StorageView#getAmount() amount}.
*
* <p>This can provide a large performance benefit over {@link #iterator()} if the caller is only interested in non-empty views,
* for example because it is trying to extract resources from the storage.
*
* <p>This function should only be overridden if the storage is able to provide an optimized iterator over non-empty views,
* for example because it is keeping an index of non-empty views.
* Otherwise, the default implementation simply calls {@link #iterator()} and filters out empty views.
*
* <p>When implementing this function, note that the guarantees of {@link #iterator()} still apply.
* In particular, {@link #insert} and {@link #extract} may be called safely during iteration.
*
* @return An iterator over the non-empty views of this storage. Calling remove on the iterator is not allowed.
*/
default Iterator<StorageView<T>> nonEmptyIterator() {
return TransferApiImpl.filterEmptyViews(iterator());
}

/**
* Convenient helper to get an {@link Iterable} over the {@linkplain #nonEmptyIterator() non-empty views} of this storage, for use in for-each loops.
*
* <p><pre>{@code
* for (StorageView<T> view : storage.nonEmptyViews()) {
* // Do something with the view
* }
* }</pre>
*/
default Iterable<StorageView<T>> nonEmptyViews() {
return this::nonEmptyIterator;
}

/**
* Return a view over this storage, for a specific resource, or {@code null} if none is quickly available.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -88,8 +88,7 @@ public static <T> long move(@Nullable Storage<T> from, @Nullable Storage<T> to,
long totalMoved = 0;

try (Transaction iterationTransaction = Transaction.openNested(transaction)) {
for (StorageView<T> view : from) {
if (view.isResourceBlank()) continue;
for (StorageView<T> view : from.nonEmptyViews()) {
T resource = view.getResource();
if (!filter.test(resource)) continue;
long maxExtracted;
Expand Down Expand Up @@ -124,14 +123,40 @@ public static <T> long move(@Nullable Storage<T> from, @Nullable Storage<T> to,
return totalMoved;
}

/**
* Try to extract any resource from a storage, up to a maximum amount.
*
* <p>This function will only ever pull from one storage view of the storage, even if multiple storage views contain the same resource.
*
* @param storage The storage, may be null.
* @param maxAmount The maximum to extract.
* @param transaction The transaction this operation is part of.
* @return A non-blank resource and the strictly positive amount of it that was extracted from the storage,
* or {@code null} if none could be found.
*/
@Nullable
public static <T> ResourceAmount<T> extractAny(@Nullable Storage<T> storage, long maxAmount, TransactionContext transaction) {
StoragePreconditions.notNegative(maxAmount);

if (storage == null) return null;

for (StorageView<T> view : storage.nonEmptyViews()) {
T resource = view.getResource();
long amount = view.extract(resource, maxAmount, transaction);
if (amount > 0) return new ResourceAmount<>(resource, amount);
}

return null;
}

/**
* Try to insert up to some amount of a resource into a list of storage slots, trying to "stack" first,
* i.e. prioritizing slots that already contain the resource.
*
* @return How much was inserted.
* @see Storage#insert
*/
public static <T> long insertStacking(List<SingleSlotStorage<T>> slots, T resource, long maxAmount, TransactionContext transaction) {
public static <T> long insertStacking(List<? extends SingleSlotStorage<T>> slots, T resource, long maxAmount, TransactionContext transaction) {
StoragePreconditions.notNegative(maxAmount);
long amount = 0;

Expand All @@ -150,6 +175,27 @@ public static <T> long insertStacking(List<SingleSlotStorage<T>> slots, T resour
return amount;
}

/**
* Insert resources in a storage, attempting to stack them with existing resources first if possible.
*
* @param storage The storage, may be null.
* @param resource The resource to insert. May not be blank.
* @param maxAmount The maximum amount of resource to insert. May not be negative.
* @param transaction The transaction this operation is part of.
* @return A nonnegative integer not greater than maxAmount: the amount that was inserted.
*/
public static <T> long tryInsertStacking(@Nullable Storage<T> storage, T resource, long maxAmount, TransactionContext transaction) {
StoragePreconditions.notNegative(maxAmount);

if (storage instanceof SlottedStorage<T> slottedStorage) {
return insertStacking(slottedStorage.getSlots(), resource, maxAmount, transaction);
} else if (storage != null) {
return storage.insert(resource, maxAmount, transaction);
} else {
return 0;
}
}

/**
* Attempt to find a resource stored in the passed storage.
*
Expand All @@ -174,8 +220,8 @@ public static <T> T findStoredResource(@Nullable Storage<T> storage, Predicate<T
Objects.requireNonNull(filter, "Filter may not be null");
if (storage == null) return null;

for (StorageView<T> view : storage) {
if (!view.isResourceBlank() && filter.test(view.getResource())) {
for (StorageView<T> view : storage.nonEmptyViews()) {
if (filter.test(view.getResource())) {
return view.getResource();
}
}
Expand Down Expand Up @@ -209,11 +255,11 @@ public static <T> T findExtractableResource(@Nullable Storage<T> storage, Predic
if (storage == null) return null;

try (Transaction nested = Transaction.openNested(transaction)) {
for (StorageView<T> view : storage) {
for (StorageView<T> view : storage.nonEmptyViews()) {
// Extract below could change the resource, so we have to query it before extracting.
T resource = view.getResource();

if (!view.isResourceBlank() && filter.test(resource) && view.extract(resource, Long.MAX_VALUE, nested) > 0) {
if (filter.test(resource) && view.extract(resource, Long.MAX_VALUE, nested) > 0) {
// Will abort the extraction.
return resource;
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
/*
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package net.fabricmc.fabric.api.transfer.v1.storage.base;

import java.util.List;

import org.jetbrains.annotations.ApiStatus;

import net.fabricmc.fabric.api.transfer.v1.storage.SlottedStorage;
import net.fabricmc.fabric.api.transfer.v1.storage.Storage;

/**
* A {@link Storage} wrapping multiple slotted storages.
* Same as {@link CombinedStorage}, but for {@link SlottedStorage}s.
*
* @param <T> The type of the stored resources.
* @param <S> The class of every part. {@code ? extends Storage<T>} can be used if the parts are of different types.
*
* <b>Experimental feature</b>, we reserve the right to remove or change it without further notice.
* The transfer API is a complex addition, and we want to be able to correct possible design mistakes.
*/
@ApiStatus.Experimental
public class CombinedSlottedStorage<T, S extends SlottedStorage<T>> extends CombinedStorage<T, S> implements SlottedStorage<T> {
public CombinedSlottedStorage(List<S> parts) {
super(parts);
}

@Override
public int getSlotCount() {
int count = 0;

for (S part : parts) {
count += part.getSlotCount();
}

return count;
}

@Override
public SingleSlotStorage<T> getSlot(int slot) {
int updatedSlot = slot;

for (SlottedStorage<T> part : parts) {
if (updatedSlot < part.getSlotCount()) {
return part.getSlot(updatedSlot);
}

updatedSlot -= part.getSlotCount();
}

throw new IndexOutOfBoundsException("Slot " + slot + " is out of bounds. This storage has size " + getSlotCount());
}
}
Loading

0 comments on commit d51205d

Please sign in to comment.