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

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

package org.openrndr.extra.shapes.primitives

import org.openrndr.math.Vector2
import org.openrndr.math.asRadians
import org.openrndr.shape.ShapeContour
import org.openrndr.shape.contour
import kotlin.math.PI
import kotlin.math.cos
import kotlin.math.sin

/**
 * Creates a regular polygon at [center] with the given [sides] and [radius].
 * Specify a [phase] in degrees to rotate it.
 */
fun regularPolygon(sides: Int, center: Vector2 = Vector2.ZERO, radius: Double = 100.0, phase: Double = 0.0): ShapeContour {
    val c = contour {
        val phi = phase.asRadians
        for (i in 0 until sides) {
            val x = center.x + radius * cos(i.toDouble() / sides * PI * 2 + phi)
            val y = center.y + radius * sin(i.toDouble() / sides * PI * 2 + phi)

            moveOrLineTo(x, y)
        }
        close()
    }
    return c
}

/**
 * Creates a rounded polygon at [center] with the given [sides] and [radius].
 * Specify a [phase] in degrees to rotate it.
 * [roundFactor] 0.0 = no rounding, 0.5 = default, 1.0 = full rounding.
 */
fun regularPolygonRounded(sides: Int, roundFactor: Double = 0.5, center: Vector2 = Vector2.ZERO, radius: Double = 100.0, phase: Double = 0.0): ShapeContour {
    val c = contour {
        val phi = phase.asRadians
        for (i in 0 until sides) {
            val x0 = center.x + radius * cos(i.toDouble() / sides * PI * 2 + phi)
            val y0 = center.y + radius * sin(i.toDouble() / sides * PI * 2 + phi)

            val x1 = center.x + radius * cos((i + 1.0) / sides * PI * 2 + phi)
            val y1 = center.y + radius * sin((i + 1.0) / sides * PI * 2 + phi)

            val x2 = center.x + radius * cos((i + 2.0) / sides * PI * 2 + phi)
            val y2 = center.y + radius * sin((i + 2.0) / sides * PI * 2 + phi)

            val f = roundFactor / 2.0

            val dx10 = x1 - x0
            val dy10 = y1 - y0
            val dx21 = x2 - x1
            val dy21 = y2 - y1

            val x3 = x0 + dx10 * f
            val y3 = y0 + dy10 * f

            val x4 = x1 - dx10 * f
            val y4 = y1 - dy10 * f

            val x5 = x1 + dx21 * f
            val y5 = y1 + dy21 * f

            moveOrLineTo(x3, y3)
            lineTo(x4, y4)
            curveTo(x1, y1, x5, y5)
        }
        close()
    }
    return c
}

/**
 * Creates a beveled polygon at [center] with the given [sides] and [radius].
 * Specify a [phase] in degrees to rotate it.
 * If 0.0 < [bevelFactor] < 1.0 the number of [sides] is doubled.
 * Using 0.5 all sides have equal length. With other values [bevelFactor]
 * determines the length ratio between even and odd sides.
 */
fun regularPolygonBeveled(sides: Int, bevelFactor: Double = 0.5, center: Vector2 = Vector2.ZERO, radius: Double = 100.0, phase: Double = 0.0): ShapeContour {
    val c = contour {
        val phi = phase.asRadians
        for (i in 0 until sides) {
            val x0 = center.x + radius * cos(i.toDouble() / sides * PI * 2 + phi)
            val y0 = center.y + radius * sin(i.toDouble() / sides * PI * 2 + phi)

            val x1 = center.x + radius * cos((i + 1.0) / sides * PI * 2 + phi)
            val y1 = center.y + radius * sin((i + 1.0) / sides * PI * 2 + phi)

            val x2 = center.x + radius * cos((i + 2.0) / sides * PI * 2 + phi)
            val y2 = center.y + radius * sin((i + 2.0) / sides * PI * 2 + phi)

            val f = bevelFactor / 2.0

            val dx10 = x1 - x0
            val dy10 = y1 - y0
            val dx21 = x2 - x1
            val dy21 = y2 - y1

            val x3 = x0 + dx10 * f
            val y3 = y0 + dy10 * f

            val x4 = x1 - dx10 * f
            val y4 = y1 - dy10 * f

            val x5 = x1 + dx21 * f
            val y5 = y1 + dy21 * f

            moveOrLineTo(x3, y3)
            lineTo(x4, y4)
            lineTo(x5, y5)
        }
        close()
    }
    return c
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy