# Adapted from: # http://enichan.darksiren.net/wordpress/?p=49 # which was adapted from some version of # https://devel.nuclex.org/framework/browser/game/Nuclex.Game/trunk/Source/Packing/CygonRectanglePacker.cs # which has the following license: # # Copyright (C) 2002-2009 Nuclex Development Labs # # This library is free software; you can redistribute it and/or # modify it under the terms of the IBM Common Public License as # published by the IBM Corporation; either version 1.0 of the # License, or (at your option) any later version. # # This library is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # IBM Common Public License for more details. from bisect import bisect_left class OutOfSpaceError(Exception): pass class Point(object): def __init__(self, x, y): self.x = x self.y = y def __cmp__(self, other): """Compares the starting position of height slices""" return self.x - other.x class RectanglePacker(object): """Base class for rectangle packing algorithms By uniting all rectangle packers under this common base class, you can easily switch between different algorithms to find the most efficient or performant one for a given job. An almost exhaustive list of packing algorithms can be found here: http://www.csc.liv.ac.uk/~epa/surveyhtml.html""" def __init__(self, packingAreaWidth, packingAreaHeight): """Initializes a new rectangle packer packingAreaWidth: Maximum width of the packing area packingAreaHeight: Maximum height of the packing area""" self.packingAreaWidth = packingAreaWidth self.packingAreaHeight = packingAreaHeight def Pack(self, rectangleWidth, rectangleHeight): """Allocates space for a rectangle in the packing area rectangleWidth: Width of the rectangle to allocate rectangleHeight: Height of the rectangle to allocate Returns the location at which the rectangle has been placed""" point = self.TryPack(rectangleWidth, rectangleHeight) if not point: raise OutOfSpaceError("Rectangle does not fit in packing area") return point def TryPack(self, rectangleWidth, rectangleHeight): """Tries to allocate space for a rectangle in the packing area rectangleWidth: Width of the rectangle to allocate rectangleHeight: Height of the rectangle to allocate Returns a Point instance if space for the rectangle could be allocated be found, otherwise returns None""" raise NotImplementedError class DumbRectanglePacker(RectanglePacker): def __init__(self, packingAreaWidth, packingAreaHeight): RectanglePacker.__init__(self, packingAreaWidth, packingAreaHeight) self.x = 0 self.y = 0 self.rowh = 0 def TryPack(self, rectangleWidth, rectangleHeight): if self.x + rectangleWidth >= self.packingAreaWidth: self.x = 0 self.y += self.rowh self.rowh = 0 if self.y + rectangleHeight >= self.packingAreaHeight: return None r = Point(self.x, self.y) self.x += rectangleWidth self.rowh = max(self.rowh, rectangleHeight) return r class CygonRectanglePacker(RectanglePacker): """ Packer using a custom algorithm by Markus 'Cygon' Ewald Algorithm conceived by Markus Ewald (cygon at nuclex dot org), though I'm quite sure I'm not the first one to come up with it :) The algorithm always places rectangles as low as possible in the packing area. So, for any new rectangle that is to be added, the packer has to determine the X coordinate at which the rectangle can have the lowest overall height without intersecting any other rectangles. To quickly discover these locations, the packer uses a sophisticated data structure that stores the upper silhouette of the packing area. When a new rectangle needs to be added, only the silouette edges need to be analyzed to find the position where the rectangle would achieve the lowest""" def __init__(self, packingAreaWidth, packingAreaHeight): """Initializes a new rectangle packer packingAreaWidth: Maximum width of the packing area packingAreaHeight: Maximum height of the packing area""" RectanglePacker.__init__(self, packingAreaWidth, packingAreaHeight) # Stores the height silhouette of the rectangles self.heightSlices = [] # At the beginning, the packing area is a single slice of height 0 self.heightSlices.append(Point(0,0)) def TryPack(self, rectangleWidth, rectangleHeight): """Tries to allocate space for a rectangle in the packing area rectangleWidth: Width of the rectangle to allocate rectangleHeight: Height of the rectangle to allocate Returns a Point instance if space for the rectangle could be allocated be found, otherwise returns None""" placement = None # If the rectangle is larger than the packing area in any dimension, # it will never fit! if rectangleWidth > self.packingAreaWidth or rectangleHeight > \ self.packingAreaHeight: return None # Determine the placement for the new rectangle placement = self.tryFindBestPlacement(rectangleWidth, rectangleHeight) # If a place for the rectangle could be found, update the height slice # table to mark the region of the rectangle as being taken. if placement: self.integrateRectangle(placement.x, rectangleWidth, placement.y \ + rectangleHeight) return placement def tryFindBestPlacement(self, rectangleWidth, rectangleHeight): """Finds the best position for a rectangle of the given dimensions rectangleWidth: Width of the rectangle to find a position for rectangleHeight: Height of the rectangle to find a position for Returns a Point instance if a valid placement for the rectangle could be found, otherwise returns None""" # Slice index, vertical position and score of the best placement we # could find bestSliceIndex = -1 # Slice index where the best placement was found bestSliceY = 0 # Y position of the best placement found # lower == better! bestScore = self.packingAreaHeight # This is the counter for the currently checked position. The search # works by skipping from slice to slice, determining the suitability # of the location for the placement of the rectangle. leftSliceIndex = 0 # Determine the slice in which the right end of the rectangle is located rightSliceIndex = bisect_left(self.heightSlices, Point(rectangleWidth, 0)) while rightSliceIndex <= len(self.heightSlices): # Determine the highest slice within the slices covered by the # rectangle at its current placement. We cannot put the rectangle # any lower than this without overlapping the other rectangles. highest = self.heightSlices[leftSliceIndex].y for index in range(leftSliceIndex + 1, rightSliceIndex): if self.heightSlices[index].y > highest: highest = self.heightSlices[index].y # Only process this position if it doesn't leave the packing area if highest + rectangleHeight < self.packingAreaHeight: score = highest if score < bestScore: bestSliceIndex = leftSliceIndex bestSliceY = highest bestScore = score # Advance the starting slice to the next slice start leftSliceIndex += 1 if leftSliceIndex >= len(self.heightSlices): break # Advance the ending slice until we're on the proper slice again, # given the new starting position of the rectangle. rightRectangleEnd = self.heightSlices[leftSliceIndex].x + rectangleWidth while rightSliceIndex <= len(self.heightSlices): if rightSliceIndex == len(self.heightSlices): rightSliceStart = self.packingAreaWidth else: rightSliceStart = self.heightSlices[rightSliceIndex].x # Is this the slice we're looking for? if rightSliceStart > rightRectangleEnd: break rightSliceIndex += 1 # If we crossed the end of the slice array, the rectangle's right # end has left the packing area, and thus, our search ends. if rightSliceIndex > len(self.heightSlices): break # Return the best placement we found for this rectangle. If the # rectangle didn't fit anywhere, the slice index will still have its # initialization value of -1 and we can report that no placement # could be found. if bestSliceIndex == -1: return None else: return Point(self.heightSlices[bestSliceIndex].x, bestSliceY) def integrateRectangle(self, left, width, bottom): """Integrates a new rectangle into the height slice table left: Position of the rectangle's left side width: Width of the rectangle bottom: Position of the rectangle's lower side""" # Find the first slice that is touched by the rectangle startSlice = bisect_left(self.heightSlices, Point(left, 0)) # We scored a direct hit, so we can replace the slice we have hit firstSliceOriginalHeight = self.heightSlices[startSlice].y self.heightSlices[startSlice] = Point(left, bottom) right = left + width startSlice += 1 # Special case, the rectangle started on the last slice, so we cannot # use the start slice + 1 for the binary search and the possibly # already modified start slice height now only remains in our temporary # firstSliceOriginalHeight variable if startSlice >= len(self.heightSlices): # If the slice ends within the last slice (usual case, unless it # has the exact same width the packing area has), add another slice # to return to the original height at the end of the rectangle. if right < self.packingAreaWidth: self.heightSlices.append(Point(right, firstSliceOriginalHeight)) else: # The rectangle doesn't start on the last slice endSlice = bisect_left(self.heightSlices, Point(right,0), \ startSlice, len(self.heightSlices)) # Another direct hit on the final slice's end? if endSlice < len(self.heightSlices) and not (Point(right, 0) < self.heightSlices[endSlice]): del self.heightSlices[startSlice:endSlice] else: # No direct hit, rectangle ends inside another slice # Find out to which height we need to return at the right end of # the rectangle if endSlice == startSlice: returnHeight = firstSliceOriginalHeight else: returnHeight = self.heightSlices[endSlice - 1].y # Remove all slices covered by the rectangle and begin a new # slice at its end to return back to the height of the slice on # which the rectangle ends. del self.heightSlices[startSlice:endSlice] if right < self.packingAreaWidth: self.heightSlices.insert(startSlice, Point(right, returnHeight))