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

commonMain.operators.ChamferCorners.kt Maven / Gradle / Ivy

package org.openrndr.extra.shapes.operators

import org.openrndr.math.Vector2
import org.openrndr.shape.*
import kotlin.math.abs
import kotlin.math.sign
import kotlin.math.sqrt

private fun Segment2D.linearSub(l0: Double, l1: Double): Segment2D {
    return sub(l0 / length, l1 / length)
}

private fun Segment2D.linearPosition(l: Double): Vector2 {
    return position((l / length).coerceIn(0.0, 1.0))
}

private fun pickLength(leftLength: Double, rightLength: Double, s0: Segment2D, s1: Segment2D): Double {
    val p3 = s1.end
    val p2 = s0.end
    val p1 = s0.start

    val det = (p2.x - p1.x) * (p3.y - p1.y) - (p3.x - p1.x) * (p2.y - p1.y)

    return if (det < 0.0) {
        leftLength
    } else {
        rightLength
    }
}

/**
 * Chamfers corners between linear segments
 * @param length the length of the chamfer
 * @param angleThreshold the maximum (smallest) angle between between linear segments
 * @param chamfer the chamfer function to apply
 */
fun ShapeContour.chamferCorners(
        lengths: (index: Int, left: Segment2D, right: Segment2D) -> Double,
        expands: (index: Int, left: Segment2D, right: Segment2D) -> Double = { _, _, _ -> 0.0 },
        clip: Boolean = true,
        angleThreshold: Double = 180.0,
        chamfer: ContourBuilder.(p1: Vector2, p2: Vector2, p3: Vector2) -> Unit
): ShapeContour {

    if (segments.size <= 1) {
        return this
    }

    return contour {
        val sourceSegments = if (closed) {
            ([email protected] + [email protected]())
        } else {
            [email protected]
        }

        var lengthIndex = sourceSegments.size - 1


        sourceSegments.first().let {
            if (it.control.size == 1) {
                moveTo(position(0.0))
            }
            if (it.control.size == 2) {
                moveTo(position(0.0))
            }
            if (it.linear) {
                if ([email protected])
                    moveTo(position(0.0))
            }
        }


        lengthIndex = 0
        for ((s0, s1) in sourceSegments.zipWithNext()) {
            lengthIndex++
            if (s0.control.size == 1) {
                moveOrCurveTo(s0.control[0], s0.end)
            } else if (s0.control.size == 2) {
                moveOrCurveTo(s0.control[0], s0.control[1], s0.end)
            } else if (s0.linear) {

                val length = lengths(lengthIndex, s0, s1)

                if (s0.linear && s1.linear && (clip || (length <= s0.length / 2 && length <= s1.length / 2))) {

                    val expand = expands(lengthIndex, s0, s1)

                    val p0 = s0.linearPosition(s0.length - length)
                    val p1 = s1.linearPosition(length)

                    val d = p1 - p0

                    val q0 = p0 - d * expand
                    val q1 = p1 + d * expand

                    moveOrLineTo(q0)
                    chamfer(q0, s0.end, q1)
                } else {
                    moveOrLineTo(s0.end)
                }
            }
        }

        // Postlude
        if (closed) {
            close()
        } else {
            val last = sourceSegments.last()
            when {
                last.linear -> {
                    lineTo(last.end)
                }
                last.control.size == 1 -> {
                    curveTo(last.control[0], last.end)
                }
                last.control.size == 2 -> {
                    curveTo(last.control[0], last.control[1], last.end)
                }
            }
        }
    }
}

fun ShapeContour.bevelCorners(length: Double, angleThreshold: Double = 180.0): ShapeContour =
        chamferCorners({ _, _, _ -> length }, angleThreshold = angleThreshold) { _, _, p3 ->
            lineTo(p3)
        }

fun ShapeContour.roundCorners(length: Double, angleThreshold: Double = 180.0): ShapeContour =
        chamferCorners({ _, _, _ -> length }, angleThreshold = angleThreshold) { _, p2, p3 ->
            curveTo(p2, p3)
        }

fun ShapeContour.arcCorners(lengths: List,
                            expands: List = listOf(0.0),
                            scales: List = listOf(1.0),
                            largeArcs: List = mutableListOf(false),
                            angleThreshold: Double = 180.0): ShapeContour {


    val scaleRing = scales.ring()
    val lengthRing = lengths.ring()
    val expandRing = expands.ring()
    val largeArcRing = largeArcs.ring()

    var segmentIndex = 0
    return chamferCorners({ index, _, _ -> lengthRing[index] },
            { index, _, _ -> expandRing[index] },
            angleThreshold = angleThreshold) { p1, p2, p3 ->

        val dx = abs(p3.x - p2.x)
        val dy = abs(p3.y - p2.y)
        val radius = sqrt(dx * dx + dy * dy)
        val det = (p2.x - p1.x) * (p3.y - p1.y) - (p3.x - p1.x) * (p2.y - p1.y)
        val scale = scaleRing[segmentIndex]
        val sweep = scale * sign(det)
        val largeArc = largeArcRing[segmentIndex]
        arcTo(radius * abs(scale), radius * abs(scale), 90.0, largeArc, sweep > 0.0, p3)
        segmentIndex++
    }
}

private class Ring(private val x: List) : List by x {
    override operator fun get(index: Int): T {
        return x[index.mod(x.size)]
    }
}

private fun  List.ring(): List {
    return Ring(this)
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy