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

it.unibo.alchemist.model.deployments.Polygon.kt Maven / Gradle / Ivy

/*
 * Copyright (C) 2010-2023, Danilo Pianini and contributors
 * listed, for each module, in the respective subproject's build.gradle.kts file.
 *
 * This file is part of Alchemist, and is distributed under the terms of the
 * GNU General Public License, with a linking exception,
 * as described in the file LICENSE in the Alchemist distribution's top directory.
 */

package it.unibo.alchemist.model.deployments

import it.unibo.alchemist.model.Environment
import it.unibo.alchemist.model.GeoPosition
import it.unibo.alchemist.model.Position2D
import org.apache.commons.math3.random.RandomGenerator
import java.awt.geom.Area
import java.awt.geom.Path2D
import java.awt.geom.Rectangle2D

/**
 * Alias for using pairs as bidimensional points.
 */
private typealias Point2D = Pair

/**
 * Creates a new Polygon with the given points.
 *
 * @param nodes the count of nodes that need to get displaced inside the polygon
 *
 * @param pointsInput the points of the polygon.
 * The class does not check for "malformed" polygons (e.g. with intersections).
 * If the provided points do not represent a valid polygon in bidimensional space, the behaviour of this class is
 * undefined. There polygon is closed automatically (there is no need to pass the first point also as last element).
 *
 */
open class Polygon

>( environment: Environment<*, P>, randomGenerator: RandomGenerator, nodes: Int, pointsInput: List<*>, ) : AbstractRandomDeployment

(environment, randomGenerator, nodes) { private val points: List = pointsInput.map { val error: () -> String = { "$it cannot get converted to Pair" } when (it) { is Pair<*, *> -> { require(it.first is Number && it.second is Number, error) @Suppress("UNCHECKED_CAST") it as Point2D } is List<*> -> { require(it.size == 2 && it[0] is Number && it[1] is Number, error) Pair(it[0] as Number, it[1] as Number) } else -> throw IllegalArgumentException(error()) } } /** * The polygon in which positions are generated. */ protected val polygon: Area = Area( Path2D.Double().apply { moveTo(points.first().toPosition) points.asSequence().drop(1).forEach { lineTo(it.toPosition) } closePath() }, ) /** * The rectangular bounds of the polygon. */ protected val bounds: Rectangle2D /** * True if this environment works with [GeoPosition]. */ protected val isOnMaps by lazy { environment.makePosition(0, 0) is GeoPosition } init { require(points.size > 3) { "At least three points are required for a polygonal deployment (provided: ${points.size}: $points)" } require(polygon.isPolygonal) { "The provided points ($points) do not supply a valid polygon" } bounds = polygon.bounds2D } /** * @param i * the node number * @return the position of the node */ final override fun indexToPosition(i: Int): P = bounds .run { val (x, y) = Pair(randomDouble(minX, maxX), randomDouble(minY, maxY)) if (isOnMaps) { environment.makePosition(y, x) // Latitude, Longitude } else { environment.makePosition(x, y) } } .takeIf { polygon.contains(it) } ?: indexToPosition(i) /** * Converts a Point2D to a [P]. */ protected val Point2D.toPosition: Position2D get() = environment.makePosition(first, second) /** * Moves the path to the specified [Position2D]. */ protected fun Path2D.Double.moveTo(destination: Position2D<*>) = moveTo(destination.x, destination.y) /** * Adds a line to the path, connecting it to the specified [Position2D]. */ protected fun Path2D.Double.lineTo(destination: Position2D<*>) = lineTo(destination.x, destination.y) /** * Checks if a [Position2D] is inside the Polygon. */ protected fun Area.contains(target: Position2D<*>) = contains(target.x, target.y) }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy