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

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