A rectangle packing library made in C# for .NET Standard.
Loosely based on the well-known C rectpack-2D library
This started as a side-project for the TrippyGL graphics library but as it grew, I decided to give it it's own repository and open the project for everyone to use.
The libary is quite small, so you can even just chuck the files directly onto your project if you don't want additional DLLs!
Once you have the library, just add using RectpackSharp
to access the library types.
You will see the PackingRectangle
type and the RectanglePacker
static class.
Create a PackingRectangle
for each rectangle you want and put them all in a single array. You can identify your rectangles using the PackingRectangle.Id
field, as the order of the rectangles is not preserved. Afterwards, just call RectanglePacker.Pack
. That's it!
PackingRectangle[] rectangles = new PackingRectangle[amount];
// Set the width and height of your rectangles
// ...
RectanglePacker.Pack(rectangles, out PackingRectangle bounds);
// All the rectangles in the array were assigned X and Y values. Bounds contains the width and height of the bin.
Specifying no extra parameters means the library will try all it's tools in order to find the best bin it can. If performance is important, you can trade space efficiency for performance with the optional parameters:
packingHint
allows you to specify which methods to try when packing. Default isPackingHints.FindBest
.acceptableDensity
makes the library stop searching once it found a solution with said density or better. Density is calculated as usedArea/binArea, so a density of 0 will yield the fastest solution it can, but possibly not an efficient one. Default is 1.stepSize
is by how much to vary the bin size after each try. Higher values might be faster but skip possibly better solutions. Default is 1.
So for example, if you know all your rectangles are squares, you might wanna try
RectanglePacker.Pack(rectangles, out PackingRectangle bounds, PackingHints.Width, 1, 1);
Pack
also provides two additional arguments of type uint?
, called maxBoundsWidth
and maxBoundsHeight
. These may be used to constrain the resulting bin to a given width and/or height. If, for example, you want a max bin height of 500, you may do something like this:
RectanglePacker.Pack(rectangles, out PackingRectangle bounds, PackingHints.FindBest, 1, 1, null, 500);
Feel free to come ask questions over at the TrippyGL Discord server!
Here's a test case where the rectangles have relatively similar dimentions.
In this test case, all the squares are the same size. Currently, the library doesn't handle the edges very well on these cases.
It also works like a charm for texture atlases or sprite sheets!
The most complicated cases are when the rectangles have very irregular dimentions, because there's no good answer to "what to put where". For these next test cases, we simply generated 512 or 2048 random rectangles (each side being from 20 to 200) and packed them.
Fuck it, here's 65536 random rectangles in a ~24k x 24k bin.