All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.primitives.BoxGrid.kt Maven / Gradle / Ivy

package org.openrndr.extra.shapes.primitives

import org.openrndr.shape.Box
import kotlin.math.round

/**
 * Split [Box] into a grid of [Box]es
 * @param columns the number of columns in the resulting grid
 * @param rows the number of rows in the resulting grid
 * @param marginX the unitless margin width
 * @param marginY the unitless margin height
 * @param marginZ the untless margin depth
 * @param gutterX the unitless gutter width, the horizontal space between grid cells
 * @param gutterY the unitless gutter height, the vertical space between grid cells
 * @param gutterZ the unitless gutter depth
 */
fun Box.grid(
    columns: Int,
    rows: Int,
    slices: Int,
    marginX: Double = 0.0,
    marginY: Double = 0.0,
    marginZ: Double = 0.0,
    gutterX: Double = 0.0,
    gutterY: Double = 0.0,
    gutterZ: Double = 0.0
) = grid(
    (width - marginX * 2 - gutterX * (columns - 1)) / columns,
    (height - marginY * 2 - gutterY * (rows - 1)) / rows,
    (depth - marginZ * 2 - gutterZ * (slices - 1)) / slices,
    marginX, marginY, marginZ,
    gutterX, gutterY, gutterZ
)

/**
 * Split [Box] into a grid of [Box]es
 * @param cellWidth the unitless width of a cell
 * @param cellHeight the unitless height of a cell
 * @param cellDepth the unitless depth of a cell
 * @param minMarginX the unitless minimum margin width (may increase to produce
 * the desired cell aspect ratio)
 * @param minMarginY the unitless minimum margin height (may increase to produce
 * the desired cell aspect ratio)
 * @param minMarginZ the unitless minimum margin depth (may increase to produce
 * @param gutterX the unitless gutter width, the horizontal space between grid cells
 * @param gutterY the unitless gutter height, the vertical space between grid cells
 * @param gutterZ the unitless gutter depth
 */
fun Box.grid(
    cellWidth: Double,
    cellHeight: Double,
    cellDepth: Double,
    minMarginX: Double = 0.0,
    minMarginY: Double = 0.0,
    minMarginZ: Double = 0.0,
    gutterX: Double = 0.0,
    gutterY: Double = 0.0,
    gutterZ: Double = 0.0
): List>> {

    val availableWidth = (width - minMarginX * 2).coerceAtLeast(0.0)
    val availableHeight = (height - minMarginY * 2).coerceAtLeast(0.0)
    val availableDepth = (depth - minMarginZ * 2).coerceAtLeast(0.0)

    val cellSpaceX = cellWidth + gutterX
    val cellSpaceY = cellHeight + gutterY
    val cellSpaceZ = cellDepth + gutterZ

    val columns = round((availableWidth + gutterX) / cellSpaceX).toInt()
    val rows = round((availableHeight + gutterY) / cellSpaceY).toInt()
    val slices = round((availableDepth + gutterZ) / cellSpaceZ).toInt()

    if (columns == 0 || rows == 0 || slices == 0) {
        return emptyList()
    }

    val totalGutterWidth = gutterX * (columns - 1).coerceAtLeast(0)
    val totalGutterHeight = gutterY * (rows - 1).coerceAtLeast(0)
    val totalGutterDepth = gutterZ * (slices - 1).coerceAtLeast(0)

    val totalWidth = cellWidth * columns + totalGutterWidth
    val totalHeight = cellHeight * rows + totalGutterHeight
    val totalDepth = cellDepth * slices + totalGutterDepth

    val x0 = corner.x + (width - totalWidth) / 2
    val y0 = corner.y + (height - totalHeight) / 2
    val z0 = corner.z + (depth - totalDepth) / 2

    return (0 until slices).map { slice ->
        (0 until rows).map { row ->
            (0 until columns).map { column ->
                Box(
                    x0 + column * cellSpaceX,
                    y0 + row * cellSpaceY,
                    z0 + slice * cellSpaceZ,
                    cellWidth, cellHeight, cellDepth
                )
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy