commonMain.androidx.constraintlayout.core.SolverVariable.kt Maven / Gradle / Ivy
/*
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package androidx.constraintlayout.core
import kotlin.jvm.JvmField
/**
* Represents a given variable used in the [linear expression solver][LinearSystem].
*/
class SolverVariable : Comparable {
var inGoal = false
/**
* Accessor for the name
*
* @return the name of the variable
*/
var name: String? = null
@JvmField
var id = -1
var definitionId = -1
var strength = 0
var computedValue = 0f
var isFinalValue = false
var strengthVector = FloatArray(MAX_STRENGTH)
var goalStrengthVector = FloatArray(MAX_STRENGTH)
var mType: Type
var mClientEquations = arrayOfNulls(16)
var mClientEquationsCount = 0
@JvmField
var usageInRowCount = 0
var isSynonym = false
var synonym = -1
var synonymDelta = 0f
/**
* Type of variables
*/
enum class Type {
/**
* The variable can take negative or positive values
*/
UNRESTRICTED,
/**
* The variable is actually not a variable :) , but a constant number
*/
CONSTANT,
/**
* The variable is restricted to positive values and represents a slack
*/
SLACK,
/**
* The variable is restricted to positive values and represents an error
*/
ERROR,
/**
* Unknown (invalid) type.
*/
UNKNOWN
}
/**
* Base constructor
* @param name the variable name
* @param type the type of the variable
*/
constructor(name: String?, type: Type) {
this.name = name
mType = type
}
constructor(type: Type, prefix: String?) {
mType = type
if (INTERNAL_DEBUG) {
//mName = getUniqueName(type, prefix);
}
}
fun clearStrengths() {
for (i in 0 until MAX_STRENGTH) {
strengthVector[i] = 0f
}
}
fun strengthsToString(): String {
var representation = "$this["
var negative = false
var empty = true
for (j in strengthVector.indices) {
representation += strengthVector[j]
if (strengthVector[j] > 0) {
negative = false
} else if (strengthVector[j] < 0) {
negative = true
}
if (strengthVector[j] != 0f) {
empty = false
}
representation += if (j < strengthVector.size - 1) {
", "
} else {
"] "
}
}
if (negative) {
representation += " (-)"
}
if (empty) {
representation += " (*)"
}
// representation += " {id: " + id + "}";
return representation
}
var inRows = if (VAR_USE_HASH) HashSet() else null
fun addToRow(row: ArrayRow) {
if (VAR_USE_HASH) {
inRows!!.add(row)
} else {
for (i in 0 until mClientEquationsCount) {
if (mClientEquations[i] === row) {
return
}
}
if (mClientEquationsCount >= mClientEquations.size) {
mClientEquations = mClientEquations.copyOf(mClientEquations.size * 2)
}
mClientEquations[mClientEquationsCount] = row
mClientEquationsCount++
}
}
fun removeFromRow(row: ArrayRow) {
if (VAR_USE_HASH) {
inRows!!.remove(row)
} else {
val count = mClientEquationsCount
for (i in 0 until count) {
if (mClientEquations[i] === row) {
for (j in i until count - 1) {
mClientEquations[j] = mClientEquations[j + 1]
}
mClientEquationsCount--
return
}
}
}
}
fun updateReferencesWithNewDefinition(system: LinearSystem?, definition: ArrayRow?) {
if (VAR_USE_HASH) {
for (row in inRows!!) {
row.updateFromRow(system, definition, false)
}
inRows!!.clear()
} else {
val count = mClientEquationsCount
for (i in 0 until count) {
mClientEquations[i]!!.updateFromRow(system, definition, false)
}
mClientEquationsCount = 0
}
}
fun setFinalValue(system: LinearSystem?, value: Float) {
if (false && INTERNAL_DEBUG) {
println("Set final value for $this of $value")
}
computedValue = value
isFinalValue = true
isSynonym = false
synonym = -1
synonymDelta = 0f
val count = mClientEquationsCount
definitionId = -1
for (i in 0 until count) {
mClientEquations[i]!!.updateFromFinalVariable(system, this, false)
}
mClientEquationsCount = 0
}
fun setSynonym(system: LinearSystem, synonymVariable: SolverVariable, value: Float) {
if (INTERNAL_DEBUG) {
println("Set synonym for $this = $synonymVariable + $value")
}
isSynonym = true
synonym = synonymVariable.id
synonymDelta = value
val count = mClientEquationsCount
definitionId = -1
for (i in 0 until count) {
mClientEquations[i]!!.updateFromSynonymVariable(system, this, false)
}
mClientEquationsCount = 0
system.displayReadableRows()
}
fun reset() {
name = null
mType = Type.UNKNOWN
strength = STRENGTH_NONE
id = -1
definitionId = -1
computedValue = 0f
isFinalValue = false
isSynonym = false
synonym = -1
synonymDelta = 0f
if (VAR_USE_HASH) {
inRows!!.clear()
} else {
val count = mClientEquationsCount
for (i in 0 until count) {
mClientEquations[i] = null
}
mClientEquationsCount = 0
}
usageInRowCount = 0
inGoal = false
goalStrengthVector.fill(0f)
}
fun setType(type: Type, prefix: String?) {
mType = type
if (INTERNAL_DEBUG && name == null) {
name = getUniqueName(type, prefix)
}
}
override fun compareTo(v: SolverVariable): Int {
return id - v.id
}
/**
* Override the toString() method to display the variable
*/
override fun toString(): String {
var result = ""
if (INTERNAL_DEBUG) {
result += name + "(" + id + "):" + strength
if (isSynonym) {
result += ":S($synonym)"
}
if (isFinalValue) {
result += ":F($computedValue)"
}
} else {
if (name != null) {
result += name
} else {
result += id
}
}
return result
}
companion object {
private const val INTERNAL_DEBUG = LinearSystem.FULL_DEBUG
private const val VAR_USE_HASH = false
const val STRENGTH_NONE = 0
const val STRENGTH_LOW = 1
const val STRENGTH_MEDIUM = 2
const val STRENGTH_HIGH = 3
const val STRENGTH_HIGHEST = 4
const val STRENGTH_EQUALITY = 5
const val STRENGTH_BARRIER = 6
const val STRENGTH_CENTERING = 7
const val STRENGTH_FIXED = 8
private var uniqueSlackId = 1
private var uniqueErrorId = 1
private var uniqueUnrestrictedId = 1
private var uniqueConstantId = 1
private var uniqueId = 1
const val MAX_STRENGTH = 9
fun increaseErrorId() {
uniqueErrorId++
}
private fun getUniqueName(type: Type, prefix: String?): String {
return if (prefix != null) {
prefix + uniqueErrorId
} else when (type) {
Type.UNRESTRICTED -> "U" + ++uniqueUnrestrictedId
Type.CONSTANT -> "C" + ++uniqueConstantId
Type.SLACK -> "S" + ++uniqueSlackId
Type.ERROR -> {
"e" + ++uniqueErrorId
}
Type.UNKNOWN -> "V" + ++uniqueId
}
throw AssertionError(type.name)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy