commonMain.androidx.constraintlayout.core.state.State.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of compose-constraint-layout-jvm Show documentation
Show all versions of compose-constraint-layout-jvm Show documentation
A copy of Android's ConstraintLayout (v2.1.3 core and v1.0.0 compose) with multiplatform capability.
/*
* Copyright (C) 2019 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.state
import androidx.constraintlayout.core.state.helpers.*
import androidx.constraintlayout.core.widgets.ConstraintWidget
import androidx.constraintlayout.core.widgets.ConstraintWidgetContainer
/**
* Represents a full state of a ConstraintLayout
*/
open class State {
protected var mReferences = HashMap()
protected var mHelperReferences = HashMap()
var mTags = HashMap>()
val mParent = ConstraintReference(this)
enum class Constraint {
LEFT_TO_LEFT, LEFT_TO_RIGHT, RIGHT_TO_LEFT, RIGHT_TO_RIGHT, START_TO_START, START_TO_END, END_TO_START, END_TO_END, TOP_TO_TOP, TOP_TO_BOTTOM, BOTTOM_TO_TOP, BOTTOM_TO_BOTTOM, BASELINE_TO_BASELINE, BASELINE_TO_TOP, BASELINE_TO_BOTTOM, CENTER_HORIZONTALLY, CENTER_VERTICALLY, CIRCULAR_CONSTRAINT
}
enum class Direction {
LEFT, RIGHT, START, END, TOP, BOTTOM
}
enum class Helper {
HORIZONTAL_CHAIN, VERTICAL_CHAIN, ALIGN_HORIZONTALLY, ALIGN_VERTICALLY, BARRIER, LAYER, FLOW
}
enum class Chain {
SPREAD, SPREAD_INSIDE, PACKED
}
open fun reset() {
mHelperReferences.clear()
mTags.clear()
}
/**
* Implements a conversion function for values, returning int.
* This can be used in case values (e.g. margins) are represented
* via an object, not directly an int.
*
* @param value the object to convert from
* @return
*/
open fun convertDimension(value: Any?): Int {
if (value is Float) {
return value.toInt()
}
return if (value is Int) {
value
} else 0
}
/**
* Create a new reference given a key.
*
* @param key
* @return
*/
fun createConstraintReference(key: Any?): ConstraintReference {
return ConstraintReference(this)
}
fun sameFixedWidth(width: Int): Boolean {
return mParent.width.equalsFixedValue(width)
}
fun sameFixedHeight(height: Int): Boolean {
return mParent.height.equalsFixedValue(height)
}
fun width(dimension: Dimension): State {
return setWidth(dimension)
}
fun height(dimension: Dimension): State {
return setHeight(dimension)
}
fun setWidth(dimension: Dimension): State {
mParent.width = dimension
return this
}
fun setHeight(dimension: Dimension): State {
mParent.height = dimension
return this
}
fun reference(key: Any?): Reference? {
return mReferences[key]
}
fun constraints(key: Any?): ConstraintReference? {
var reference = mReferences[key]
if (reference == null) {
reference = createConstraintReference(key)
mReferences[key] = reference
reference.key = key
}
return if (reference is ConstraintReference) {
reference
} else null
}
private var numHelpers = 0
private fun createHelperKey(): String {
return "__HELPER_KEY_" + numHelpers++ + "__"
}
fun helper(key: Any?, type: Helper): HelperReference {
var key = key
if (key == null) {
key = createHelperKey()
}
var reference = mHelperReferences[key]
if (reference == null) {
reference = when (type) {
Helper.HORIZONTAL_CHAIN -> {
HorizontalChainReference(this)
}
Helper.VERTICAL_CHAIN -> {
VerticalChainReference(this)
}
Helper.ALIGN_HORIZONTALLY -> {
AlignHorizontallyReference(this)
}
Helper.ALIGN_VERTICALLY -> {
AlignVerticallyReference(this)
}
Helper.BARRIER -> {
BarrierReference(this)
}
else -> {
HelperReference(this, type)
}
}
reference.key = key
mHelperReferences[key] = reference
}
return reference
}
fun horizontalGuideline(key: Any?): GuidelineReference {
return guideline(key, ConstraintWidget.HORIZONTAL)
}
fun verticalGuideline(key: Any?): GuidelineReference {
return guideline(key, ConstraintWidget.VERTICAL)
}
fun guideline(key: Any?, orientation: Int): GuidelineReference {
val reference = constraints(key)
if (reference!!.facade == null || reference.facade !is GuidelineReference) {
val guidelineReference = GuidelineReference(this)
guidelineReference.orientation = orientation
guidelineReference.key = key
reference.facade = guidelineReference
}
return reference.facade as GuidelineReference
}
fun barrier(key: Any?, direction: Direction?): BarrierReference {
val reference = constraints(key)
if (reference!!.facade == null || reference.facade !is BarrierReference) {
val barrierReference = BarrierReference(this)
barrierReference.setBarrierDirection(direction)
reference.facade = barrierReference
}
return reference.facade as BarrierReference
}
fun verticalChain(): VerticalChainReference {
return helper(null, Helper.VERTICAL_CHAIN) as VerticalChainReference
}
fun verticalChain(vararg references: Any): VerticalChainReference {
val reference = helper(null, Helper.VERTICAL_CHAIN) as VerticalChainReference
reference.add(*references)
return reference
}
fun horizontalChain(): HorizontalChainReference {
return helper(null, Helper.HORIZONTAL_CHAIN) as HorizontalChainReference
}
fun horizontalChain(vararg references: Any): HorizontalChainReference {
val reference = helper(null, Helper.HORIZONTAL_CHAIN) as HorizontalChainReference
reference.add(*references)
return reference
}
fun centerHorizontally(vararg references: Any): AlignHorizontallyReference {
val reference = helper(null, Helper.ALIGN_HORIZONTALLY) as AlignHorizontallyReference
reference.add(*references)
return reference
}
fun centerVertically(vararg references: Any): AlignVerticallyReference {
val reference = helper(null, Helper.ALIGN_VERTICALLY) as AlignVerticallyReference
reference.add(*references)
return reference
}
fun directMapping() {
for (key in mReferences.keys) {
val ref = constraints(key) ?: continue
ref.view = key
}
}
fun map(key: Any?, view: Any?) {
val ref: Reference? = constraints(key)
if (ref is ConstraintReference) {
ref.view = view
}
}
fun setTag(key: String, tag: String) {
val ref: Reference? = constraints(key)
if (ref is ConstraintReference) {
ref.tag = tag
var list: ArrayList? = null
if (!mTags.containsKey(tag)) {
list = ArrayList()
mTags[tag] = list
} else {
list = mTags[tag]
}
list!!.add(key)
}
}
fun getIdsForTag(tag: String): ArrayList? {
return if (mTags.containsKey(tag)) {
mTags[tag]
} else null
}
fun apply(container: ConstraintWidgetContainer) {
container.removeAllChildren()
mParent.width.apply(this, container, ConstraintWidget.HORIZONTAL)
mParent.height.apply(this, container, ConstraintWidget.VERTICAL)
for (key in mHelperReferences.keys) {
val reference = mHelperReferences[key]
val helperWidget = reference!!.helperWidget
if (helperWidget != null) {
var constraintReference = mReferences[key]
if (constraintReference == null) {
constraintReference = constraints(key)
}
constraintReference!!.constraintWidget = helperWidget
}
}
for (key in mReferences.keys) {
val reference = mReferences[key]
if (reference !== mParent && reference!!.facade is HelperReference) {
val helperWidget = (reference.facade as HelperReference).helperWidget
if (helperWidget != null) {
var constraintReference = mReferences[key]
if (constraintReference == null) {
constraintReference = constraints(key)
}
constraintReference!!.constraintWidget = helperWidget
}
}
}
for (key in mReferences.keys) {
val reference = mReferences[key]
if (reference !== mParent) {
val widget = reference!!.constraintWidget ?: continue
widget.debugName = reference.key.toString()
widget.parent = null
if (reference.facade is GuidelineReference) {
// we apply Guidelines first to correctly setup their ConstraintWidget.
reference.apply()
}
container.add(widget)
} else {
reference.constraintWidget = container
}
}
for (key in mHelperReferences.keys) {
val reference = mHelperReferences[key]
val helperWidget = reference!!.helperWidget
if (helperWidget != null) {
for (keyRef in reference.mReferences) {
val constraintReference = mReferences[keyRef]
reference.helperWidget?.add(constraintReference!!.constraintWidget)
}
reference.apply()
} else {
reference.apply()
}
}
for (key in mReferences.keys) {
val reference = mReferences[key]
if (reference !== mParent && reference!!.facade is HelperReference) {
val helperReference = reference.facade as HelperReference
val helperWidget = helperReference.helperWidget
if (helperWidget != null) {
for (keyRef in helperReference.mReferences) {
val constraintReference = mReferences[keyRef]
if (constraintReference != null) {
helperWidget.add(constraintReference.constraintWidget)
} else if (keyRef is Reference) {
helperWidget.add(keyRef.constraintWidget)
} else {
println("couldn't find reference for $keyRef")
}
}
reference.apply()
}
}
}
for (key in mReferences.keys) {
val reference = mReferences[key]
reference!!.apply()
val widget = reference.constraintWidget
if (widget != null && key != null) {
widget.stringId = key.toString()
}
}
}
companion object {
const val UNKNOWN = -1
const val CONSTRAINT_SPREAD = 0
const val CONSTRAINT_WRAP = 1
const val CONSTRAINT_RATIO = 2
const val PARENT = 0
}
init {
mReferences[PARENT] = mParent
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy