2023-03-03 16:23:46 +01:00
|
|
|
import { injectable } from "tsyringe";
|
|
|
|
|
|
|
|
export class FindSlotResult
|
|
|
|
{
|
|
|
|
success: boolean;
|
|
|
|
x: any;
|
|
|
|
y: any;
|
|
|
|
rotation: boolean;
|
|
|
|
constructor(success = false, x = null, y = null, rotation = false)
|
|
|
|
{
|
|
|
|
this.success = success;
|
|
|
|
this.x = x;
|
|
|
|
this.y = y;
|
|
|
|
this.rotation = rotation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class ContainerHelper
|
|
|
|
{
|
2023-07-24 16:52:55 +02:00
|
|
|
/**
|
|
|
|
* Finds a slot for an item in a given 2D container map
|
|
|
|
* @param container2D Array of container with slots filled/free
|
|
|
|
* @param itemWidth Width of item
|
|
|
|
* @param itemHeight Height of item
|
|
|
|
* @returns Location to place item in container
|
2023-03-03 16:23:46 +01:00
|
|
|
*/
|
|
|
|
public findSlotForItem(container2D: number[][], itemWidth: number, itemHeight: number): FindSlotResult
|
|
|
|
{
|
|
|
|
let rotation = false;
|
|
|
|
const minVolume = (itemWidth < itemHeight ? itemWidth : itemHeight) - 1;
|
|
|
|
const containerY = container2D.length;
|
|
|
|
const containerX = container2D[0].length;
|
|
|
|
const limitY = containerY - minVolume;
|
|
|
|
const limitX = containerX - minVolume;
|
|
|
|
|
2024-02-09 18:13:19 +01:00
|
|
|
// Every x+y slot taken up in container, exit
|
2024-05-17 21:32:41 +02:00
|
|
|
if (container2D.every((x) => x.every((y) => y === 1)))
|
2024-02-09 16:13:49 +01:00
|
|
|
{
|
|
|
|
return new FindSlotResult(false);
|
|
|
|
}
|
|
|
|
|
2024-02-09 18:13:19 +01:00
|
|
|
// Down
|
2023-03-03 16:23:46 +01:00
|
|
|
for (let y = 0; y < limitY; y++)
|
|
|
|
{
|
2024-02-09 18:13:19 +01:00
|
|
|
// Across
|
2024-05-17 21:32:41 +02:00
|
|
|
if (container2D[y].every((x) => x === 1))
|
2024-02-09 18:13:19 +01:00
|
|
|
{
|
|
|
|
// Every item in row is full, skip row
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
for (let x = 0; x < limitX; x++)
|
|
|
|
{
|
|
|
|
let foundSlot = this.locateSlot(container2D, containerX, containerY, x, y, itemWidth, itemHeight);
|
|
|
|
|
2024-02-09 18:13:19 +01:00
|
|
|
// Failed to find slot, rotate item and try again
|
2023-03-03 16:23:46 +01:00
|
|
|
if (!foundSlot && itemWidth * itemHeight > 1)
|
2024-02-09 18:13:19 +01:00
|
|
|
{
|
|
|
|
// Bigger than 1x1
|
|
|
|
foundSlot = this.locateSlot(container2D, containerX, containerY, x, y, itemHeight, itemWidth); // Height/Width swapped
|
2023-03-03 16:23:46 +01:00
|
|
|
if (foundSlot)
|
|
|
|
{
|
2024-02-09 16:13:49 +01:00
|
|
|
// Found a slot for it when rotated
|
2023-03-03 16:23:46 +01:00
|
|
|
rotation = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!foundSlot)
|
|
|
|
{
|
2024-02-09 16:13:49 +01:00
|
|
|
// Didn't fit this hole, try again
|
2023-03-03 16:23:46 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new FindSlotResult(true, x, y, rotation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-09 16:13:49 +01:00
|
|
|
// Tried all possible holes, nothing big enough for the item
|
|
|
|
return new FindSlotResult(false);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-07-24 16:52:55 +02:00
|
|
|
/**
|
|
|
|
* Find a slot inside a container an item can be placed in
|
|
|
|
* @param container2D Container to find space in
|
|
|
|
* @param containerX Container x size
|
|
|
|
* @param containerY Container y size
|
|
|
|
* @param x ???
|
|
|
|
* @param y ???
|
|
|
|
* @param itemW Items width
|
|
|
|
* @param itemH Items height
|
|
|
|
* @returns True - slot found
|
|
|
|
*/
|
2023-11-16 02:35:05 +01:00
|
|
|
protected locateSlot(
|
|
|
|
container2D: number[][],
|
|
|
|
containerX: number,
|
|
|
|
containerY: number,
|
|
|
|
x: number,
|
|
|
|
y: number,
|
|
|
|
itemW: number,
|
|
|
|
itemH: number,
|
|
|
|
): boolean
|
2023-07-24 16:52:55 +02:00
|
|
|
{
|
|
|
|
let foundSlot = true;
|
|
|
|
|
|
|
|
for (let itemY = 0; itemY < itemH; itemY++)
|
|
|
|
{
|
|
|
|
if (foundSlot && y + itemH - 1 > containerY - 1)
|
|
|
|
{
|
|
|
|
foundSlot = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2024-02-09 18:13:19 +01:00
|
|
|
// Does item fit x-ways across
|
2023-07-24 16:52:55 +02:00
|
|
|
for (let itemX = 0; itemX < itemW; itemX++)
|
|
|
|
{
|
|
|
|
if (foundSlot && x + itemW - 1 > containerX - 1)
|
|
|
|
{
|
|
|
|
foundSlot = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (container2D[y + itemY][x + itemX] !== 0)
|
|
|
|
{
|
|
|
|
foundSlot = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!foundSlot)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return foundSlot;
|
|
|
|
}
|
|
|
|
|
2023-07-23 13:29:00 +02:00
|
|
|
/**
|
|
|
|
* Find a free slot for an item to be placed at
|
2024-02-09 18:13:19 +01:00
|
|
|
* @param container2D Container to place item in
|
2023-07-23 13:29:00 +02:00
|
|
|
* @param x Container x size
|
|
|
|
* @param y Container y size
|
|
|
|
* @param itemW Items width
|
|
|
|
* @param itemH Items height
|
|
|
|
* @param rotate is item rotated
|
|
|
|
*/
|
2023-11-16 02:35:05 +01:00
|
|
|
public fillContainerMapWithItem(
|
|
|
|
container2D: number[][],
|
|
|
|
x: number,
|
|
|
|
y: number,
|
|
|
|
itemW: number,
|
|
|
|
itemH: number,
|
|
|
|
rotate: boolean,
|
2024-02-09 16:13:49 +01:00
|
|
|
): void
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2024-02-09 18:13:19 +01:00
|
|
|
// Swap height/width if we want to fit it in rotated
|
2023-03-03 16:23:46 +01:00
|
|
|
const itemWidth = rotate ? itemH : itemW;
|
|
|
|
const itemHeight = rotate ? itemW : itemH;
|
|
|
|
|
|
|
|
for (let tmpY = y; tmpY < y + itemHeight; tmpY++)
|
|
|
|
{
|
|
|
|
for (let tmpX = x; tmpX < x + itemWidth; tmpX++)
|
|
|
|
{
|
|
|
|
if (container2D[tmpY][tmpX] === 0)
|
|
|
|
{
|
2024-02-09 18:13:19 +01:00
|
|
|
// Flag slot as used
|
2023-03-03 16:23:46 +01:00
|
|
|
container2D[tmpY][tmpX] = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-02-09 18:13:19 +01:00
|
|
|
throw new Error(`Slot at (${x}, ${y}) is already filled. Cannot fit a ${itemW} by ${itemH}`);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-11-16 02:35:05 +01:00
|
|
|
}
|