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

commonMain.utils.Combinatorics.kt Maven / Gradle / Ivy

The newest version!
package org.openrndr.kartifex.utils

object Combinatorics {
    const val MAX_RESULTS = 32
    inline fun  permutations(values: List): List> {
        // if exhaustive searching is out of the question, put your trust in the RNG
        if (values.size > 4) {
            List(MAX_RESULTS) {
                values.shuffled()
            }
        }
        val result: MutableList> = mutableListOf()
        val ary: Array = values.toTypedArray()
        val c = IntArray(ary.size)
        var i = 0
        result.add(values)
        while (i < ary.size) {
            if (c[i] < i) {
                swap(ary, if (i % 2 == 0) 0 else c[i], i)
                result.add(ary.toList())
                c[i]++
                i = 0
            } else {
                c[i] = 0
                i++
            }
        }
        return result
    }

    fun  swap(ary: Array, i: Int, j: Int) {
        val tmp = ary[i]
        ary[i] = ary[j]
        ary[j] = tmp
    }


    /**
     * Given a list of potential values at each index in a list, returns all possible combinations of those values.
     */
    fun  combinations(paths: List>): List> {
        val count = paths.map { obj -> obj.size }
            .fold(1) { a, b -> a * b }
        if (count == 0) {
            return emptyList()
        } else if (count == 1) {
            return listOf(paths
                .map { obj -> obj.first() }
            )
        } else if (count > MAX_RESULTS) {
            return (0 until MAX_RESULTS)
                .map { _ ->
                    paths
                        .map { list ->
                            list.random()
                        }
                }
        }
        val indices = IntArray(paths.size)
        val result = mutableListOf>()
        while (indices[0] < paths.first().size) {
            val path = mutableListOf()
            for (i in indices.indices) {
                path.add(paths[i][indices[i]])
            }
            result.add(path)
            for (i in indices.indices.reversed()) {
                if (++indices[i] < paths[i].size) {
                    break
                } else if (i > 0) {
                    indices[i] = 0
                }
            }
        }
        return result
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy