Reorder an item in an array before or after another.
I make a web app in which my users can reorder before or after another.
I did not find any package to do what I search for, and since I use it at multiple place on my app I figured I would open source it in the hope it can help simplify the life of another developer out there!
- Can reorder an item before or after another in a list of items
- Systematically "self-heals" the items orders by re-writing the index of all items starting from the minimum index
- PHP >= 8.1
- Composer
In your terminal, on the root of your project folder, run:
composer require khalyomede/reorder-before-after
- 1. Moving an item before another
- 2. Create a list of items out of any type
- 3. Find an item by its value
- 4. Create a listing from an array
- 5. Getting all items from a listing
- 6. Use a callback to apply the order on your value
- 7. Create a listing from values and specify how to retrieve the order using a callback
- 8. Specify how to match items together
In this example, we will move our book before our table.
use Khalyomede\ReorderBeforeAfter\Item;
use Khalyomede\ReorderBeforeAfter\Listing;
use Khalyomede\ReorderBeforeAfter\Placement;
$listing = new Listing();
$listing->push(new Item("bag", 1));
$listing->push(new Item("chair", 2));
$listing->push(new Item("table", 3));
$listing->push(new Item("book", 4));
$listing->reorder("book", Placement::Before, "table");
assert($listing->find("bag")->order === 1);
assert($listing->find("chair")->order === 2);
assert($listing->find("book")->order === 3);
assert($listing->find("table")->order === 4);
In this example, we will see that we can create a listing of anything, including objects.
use Khalyomede\ReorderBeforeAfter\Item;
use Khalyomede\ReorderBeforeAfter\Listing;
final readonly class Product
{
public function __construct(
public string $name,
public int $quantity,
public float $unitPrice,
) {}
}
$bags = new Product("bag", 15, 149.99);
$tables = new Product("table", 4, 89.99);
$chairs = new Product("chairs", 1, 399.99);
$listing = new Listing();
$listing->push(new Item($bags, 1));
$listing->push(new Item($tables, 2));
$listing->push(new Item($chairs, 3));
In this example, we will see we can find an item by its "value" (the first parameter when creating a new instance of Item
).
use Khalyomede\ReorderBeforeAfter\Item;
use Khalyomede\ReorderBeforeAfter\Listing;
use Khalyomede\ReorderBeforeAfter\Placement;
$listing = new Listing();
$listing->push(new Item("bag", 1));
$listing->push(new Item("chair", 2));
$listing->push(new Item("table", 3));
$listing->push(new Item("book", 4));
echo $listing->find("bag"); // Item(value: "bag", order: 1)
If the method finds no item matching the value, a ItemNotFoundException
will be thrown.
You can also search by anything that can be checked using ===
, including objects.
use Khalyomede\ReorderBeforeAfter\Item;
use Khalyomede\ReorderBeforeAfter\Listing;
use Khalyomede\ReorderBeforeAfter\Placement;
final readonly class Product
{
public function __construct(
public string $name,
public int $quantity,
public float $unitPrice,
) {}
}
$bags = new Product("bag", 15, 149.99);
$tables = new Product("table", 4, 89.99);
$chairs = new Product("chairs", 1, 399.99);
$listing = new Listing();
$listing->push(new Item($bags, 1));
$listing->push(new Item($tables, 2));
$listing->push(new Item($chairs, 3));
echo $listing->find($bags); // Item(value: Product(name: "bag", quantity: 15, unitPrice: 149.99), order: 1)
In this example, we will create the listing from an array instead of manually pushing items.
use Khalyomede\ReorderBeforeAfter\Listing;
$listing = Listing::from([
new Item("bag", 1),
new Item("chair", 2),
new Item("book", 3),
new Item("table", 4),
]);
$listing->reorder("bag", Placement::After, "book");
assert($listing->find("bag")->order === 3);
In this example, we will get all items from a listing. Useful if you want to perform some task after reordering your items.
use Khalyomede\ReorderBeforeAfter\Listing;
$listing = Listing::from([
new Item("bag", 1),
new Item("chair", 2),
new Item("book", 3),
new Item("table", 4),
]);
$products = $listing->all();
foreach ($products as $product) {
echo $product; // "bag" or "chair" or "book" or "table"
}
In this example, we will instruct the listing how it should set the order on our values. Useful if our values are objects holding their order. This saves you from looping again on all your objects.
This examples features an hypothetical Product
Eloquent model.
use App\Models\Product;
use Khalyomede\ReorderBeforeAfter\Listing;
$items = Product::all()->map(fn (Product $product): Item => new Item($product, $product->order));
$listing = Listing::from($items);
$listing->applyWith(function (Item $item): void {
$item->value->order = $item->order;
$item->value->save();
});
In this example, we will see how to create a listing out of an array of values, and using the second parameter to specify how to get the order from these values.
Useful if you have objects that already hold their own order, and you do not want to loop from them and create the Item
by hand.
use App\Models\Product;
use Khalyomede\ReorderBeforeAfter\Listing;
$products = Product::all();
$listing = Listing::outOf($products, fn (Product $product): Item => new Item($product, $product->order));
In this example, we will instruct the listing how it should match our objects. Useful if your objects can contain different set of data, making the whole object different, when in reality both are equal (matching by id).
This examples features an hypothetical Product
Eloquent model, where one of the two can have more attributes (like eager loaded relationships), making the triple equal check invalid.
use App\Models\Product;
use Khalyomede\ReorderBeforeAfter\Listing;
$listing = Listing::outOf(Product::all(), fn (Product $product): Item => new Item($product, $product->order));
$listing->matchWith(fn (Product $left, Product $right): bool => $left->id === $right->id);
// Or with the shorter Eloquent::is() method
$listing->matchWith(fn (Product $left, Product $right): bool => $left->is($right));
composer run test
composer run analyse
composer run lint
composer run check
composer run updates
composer run scan
Or
composer run all