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

commonMain.jetbrains.datalore.plot.base.stat.ContourFillHelper.kt Maven / Gradle / Ivy

/*
 * Copyright (c) 2019. JetBrains s.r.o.
 * Use of this source code is governed by the MIT license that can be found in the LICENSE file.
 */

package jetbrains.datalore.plot.base.stat

import jetbrains.datalore.base.gcommon.collect.ClosedRange
import jetbrains.datalore.base.geometry.DoubleVector
import jetbrains.datalore.plot.base.stat.ContourStatUtil.removePathByEndpoints

class ContourFillHelper(xRange: ClosedRange, yRange: ClosedRange) {

    private val myLowLeft: DoubleVector
    private val myLowRight: DoubleVector
    private val myUpLeft: DoubleVector
    private val myUpRight: DoubleVector

    init {
        val xMin = xRange.lowerEnd
        val xMax = xRange.upperEnd
        val yMin = yRange.lowerEnd
        val yMax = yRange.upperEnd
        myLowLeft = DoubleVector(xMin, yMin)
        myLowRight = DoubleVector(xMax, yMin)
        myUpLeft = DoubleVector(xMin, yMax)
        myUpRight = DoubleVector(xMax, yMax)
    }

    fun createPolygons(
            pathByLevels: Map>>, levels: List, fillLevels: List): Map> {

        val result = HashMap>()

        //getBoundary
        val allPaths = ArrayList>()
        for (level in pathByLevels.values) {
            allPaths.addAll(level)
        }

        val cornerPoints = ArrayList()
        val outerMap = createOuterMap(allPaths, cornerPoints)


        val n = pathByLevels.keys.size

        // The designated area is always to the left
        for (i in 0 until n + 1) {
            var polygonPieces: MutableList> = ArrayList()
            val value = ArrayList()
            if (i > 0) {
                polygonPieces.addAll(
                    reverseAll(
                        pathByLevels[levels[i - 1]]!!
                    )
                )
            }
            if (i < n) {
                polygonPieces.addAll(pathByLevels[levels[i]]!!)
            }
            polygonPieces = createClosedPolygonLevels(polygonPieces, outerMap, cornerPoints)
            for (path in polygonPieces) {
                value.addAll(path)
            }
            result[fillLevels[i]] = value
        }
        return result
    }

    private fun createClosedPolygonLevels(
            openLevels: List>, outerMap: Map, cornerPoints: List): MutableList> {
        val result = ArrayList>()

        val donePath = HashSet>()
        //Add boundary part to each openPath
        for (level in openLevels) {
            val levelCopy = ArrayList(level)
            val p0 = levelCopy[0]
            var p1 = levelCopy[levelCopy.size - 1]
            //if cycle
            if (p0 == p1) {
                result.add(levelCopy)
                continue
            }
            //p1 remains to be the end of levelCopy
            var pi2 = outerMap[p1]!!
            levelCopy.add(pi2)
            p1 = pi2
            while (cornerPoints.contains(p1)) {
                pi2 = outerMap[p1]!!
                levelCopy.add(pi2)
                p1 = pi2
            }
            donePath.add(levelCopy)
        }

        val pathByEndPoint = HashMap>()

        for (path in donePath) {
            val p0 = path[0]
            val p1 = path[path.size - 1]
            //if cycle
            if (p0 == p1) {
                result.add(path)
                continue
            }

            if (pathByEndPoint.containsKey(p0) || pathByEndPoint.containsKey(p1)) {
                //remove need to be modified paths
                val path0 = pathByEndPoint[p0]
                val path1 = pathByEndPoint[p1]
                removePathByEndpoints(path0, pathByEndPoint)
                removePathByEndpoints(path1, pathByEndPoint)

                val longPath: MutableList
                if (path0 === path1) {
                    //a cycle is formed
                    path0!!.addAll(path.subList(1, path.size))
                    result.add(path0)
                    continue
                } else if (path0 != null && path1 != null) {
                    longPath = path0
                    longPath.addAll(path.subList(1, path.size - 1))
                    longPath.addAll(path1)
                } else if (path0 == null) { //path1!=null
                    longPath = path1!!
                    longPath.addAll(0, path.subList(0, path.size - 1))
                } else { //path0!=null
                    longPath = path0
                    longPath.addAll(path.subList(1, path.size))
                }

                pathByEndPoint[longPath[0]] = longPath
                pathByEndPoint[longPath[longPath.size - 1]] = longPath
            } else {
                pathByEndPoint[path[0]] = path
                pathByEndPoint[path[path.size - 1]] = path
            }
        }

        if (!pathByEndPoint.keys.isEmpty()) {
            throw IllegalArgumentException("Some paths are not cleared yet there is something wrong!" + pathByEndPoint.keys.size)
        }

        // check
        for (path in result) {
            if (path[0] != path[path.size - 1]) {
                throw IllegalArgumentException("The polygons are not entirely closed!")
            }
        }
        return result
    }

    //counterclockwise
    //update corner points for record
    private fun createOuterMap(paths: List>, cornerPoints: MutableList): Map {
        val points = ArrayList()
        for (path in paths) {
            if (path[0] != path[path.size - 1]) {
                points.add(path[0])
                points.add(path[path.size - 1])
            }
        }

        val result = HashMap()

        val list = ArrayList>()
        for (i in 0..3) {
            list.add(ArrayList())
        }

        for (point in points) {
            val kind = getKind(point)
            when (kind) {
                BorderKind.DOWN -> list[0].add(point)
                BorderKind.RIGHT -> list[1].add(point)
                BorderKind.UP -> list[2].add(point)
                BorderKind.LEFT -> list[3].add(point)
            }
        }

        list[0].sortWith(Comparator { o1, o2 -> o1.x.compareTo(o2.x) })
        list[1].sortWith(Comparator { o1, o2 -> o1.y.compareTo(o2.y) })
        list[2].sortWith(Comparator { o1, o2 -> o2.x.compareTo(o1.x) })
        list[3].sortWith(Comparator { o1, o2 -> o2.y.compareTo(o1.y) })

        // According to getKind: myLowLeft -> DOWN; myLowRight -> DOWN; myUpRight -> RIGHT; myUpLeft -> LEFT
        val prepareMap = ArrayList()
        if (!list[0].contains(myLowLeft)) {
            prepareMap.add(myLowLeft)
            cornerPoints.add(myLowLeft)
        }
        prepareMap.addAll(list[0])
        if (!list[0].contains(myLowRight)) {
            prepareMap.add(myLowRight)
            cornerPoints.add(myLowRight)
        }
        prepareMap.addAll(list[1])
        if (!list[1].contains(myUpRight)) {
            prepareMap.add(myUpRight)
            cornerPoints.add(myUpRight)
        }
        prepareMap.addAll(list[2])
        if (!list[3].contains(myUpLeft)) {
            prepareMap.add(myUpLeft)
            cornerPoints.add(myUpLeft)
        }
        prepareMap.addAll(list[3])
        prepareMap.add(prepareMap[0])
        for (i in 0 until prepareMap.size - 1) {
            result[prepareMap[i]] = prepareMap[i + 1]
        }

        return result
    }

    private fun getKind(point: DoubleVector): BorderKind {
        return if (belowOrOnLine(
                myLowLeft,
                myUpRight,
                point
            ) && belowOrOnLine(
                myUpLeft,
                myLowRight,
                point
            )
        ) {
            BorderKind.DOWN
        } else if (belowOrOnLine(
                myLowLeft,
                myUpRight,
                point
            ) && !belowOrOnLine(
                myUpLeft,
                myLowRight,
                point
            )
        ) {
            BorderKind.RIGHT
        } else if (!belowOrOnLine(
                myLowLeft,
                myUpRight,
                point
            ) && !belowOrOnLine(
                myUpLeft,
                myLowRight,
                point
            )
        ) {
            BorderKind.UP
        } else if (!belowOrOnLine(
                myLowLeft,
                myUpRight,
                point
            ) && belowOrOnLine(
                myUpLeft,
                myLowRight,
                point
            )
        ) {
            BorderKind.LEFT
        } else {
            throw IllegalArgumentException("The Contour Point is not on the border $point")
        }
    }

    private enum class BorderKind {
        DOWN, RIGHT, UP, LEFT
    }

    companion object {

        fun computeFillLevels(zRange: ClosedRange, levels: List): List {
            val fillLevels = ArrayList()
            fillLevels.add(zRange.lowerEnd)
            for (i in 0 until levels.size - 1) {
                fillLevels.add((levels[i] + levels[i + 1]) / 2.0)
            }
            fillLevels.add(zRange.upperEnd)
            return fillLevels
        }

        private fun  reverseAll(list: List>): List> {
            val result = ArrayList>()
            for (item in list) {
                val copy = ArrayList(item)
                copy.reverse()
                result.add(copy)
            }
            return result
        }

        // a.x= 0
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy