commonMain.androidx.constraintlayout.core.widgets.ConstraintWidgetContainer.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.widgets
import androidx.constraintlayout.core.LinearSystem
import androidx.constraintlayout.core.Metrics
import androidx.constraintlayout.core.SolverVariable
import androidx.constraintlayout.core.widgets.Chain.applyChainConstraints
import androidx.constraintlayout.core.widgets.analyzer.BasicMeasure
import androidx.constraintlayout.core.widgets.analyzer.DependencyGraph
import androidx.constraintlayout.core.widgets.analyzer.Direct.ls
import androidx.constraintlayout.core.widgets.analyzer.Direct.solvingPass
import androidx.constraintlayout.core.widgets.analyzer.Grouping.simpleSolvingPass
import kotlin.math.max
/**
* A container of ConstraintWidget that can layout its children
*/
class ConstraintWidgetContainer : WidgetContainer {
var mBasicMeasureSolver = BasicMeasure(this)
////////////////////////////////////////////////////////////////////////////////////////////////
// Graph measures
////////////////////////////////////////////////////////////////////////////////////////////////
var mDependencyGraph = DependencyGraph(this)
private var pass // number of layout passes
= 0
/**
* Invalidate the graph of constraints
*/
fun invalidateGraph() {
mDependencyGraph.invalidateGraph()
}
/**
* Invalidate the widgets measures
*/
fun invalidateMeasures() {
mDependencyGraph.invalidateMeasures()
}
fun directMeasure(optimizeWrap: Boolean): Boolean {
return mDependencyGraph.directMeasure(optimizeWrap)
// int paddingLeft = getX();
// int paddingTop = getY();
// if (mDependencyGraph.directMeasureSetup(optimizeWrap)) {
// mDependencyGraph.measureWidgets();
// boolean allResolved = mDependencyGraph.directMeasureWithOrientation(optimizeWrap, HORIZONTAL);
// allResolved &= mDependencyGraph.directMeasureWithOrientation(optimizeWrap, VERTICAL);
// for (ConstraintWidget child : mChildren) {
// child.setDrawX(child.getDrawX() + paddingLeft);
// child.setDrawY(child.getDrawY() + paddingTop);
// }
// setX(paddingLeft);
// setY(paddingTop);
// return allResolved;
// }
// return false;
}
fun directMeasureSetup(optimizeWrap: Boolean): Boolean {
return mDependencyGraph.directMeasureSetup(optimizeWrap)
}
fun directMeasureWithOrientation(optimizeWrap: Boolean, orientation: Int): Boolean {
return mDependencyGraph.directMeasureWithOrientation(optimizeWrap, orientation)
}
fun defineTerminalWidgets() {
mDependencyGraph.defineTerminalWidgets(horizontalDimensionBehaviour, verticalDimensionBehaviour)
}
////////////////////////////////////////////////////////////////////////////////////////////////
/**
* Measure the layout
*
* @param optimizationLevel
* @param widthMode
* @param widthSize
* @param heightMode
* @param heightSize
* @param paddingX
* @param paddingY
*/
fun measure(
optimizationLevel: Int, widthMode: Int, widthSize: Int,
heightMode: Int, heightSize: Int, lastMeasureWidth: Int, lastMeasureHeight: Int, paddingX: Int, paddingY: Int
): Long {
mPaddingLeft = paddingX
mPaddingTop = paddingY
return mBasicMeasureSolver.solverMeasure(
this, optimizationLevel, paddingX, paddingY, widthMode, widthSize, heightMode, heightSize,
lastMeasureWidth, lastMeasureHeight
)
}
fun updateHierarchy() {
mBasicMeasureSolver.updateHierarchy(this)
}
protected var mMeasurer: BasicMeasure.Measurer? = null
var measurer: BasicMeasure.Measurer?
get() = mMeasurer
set(measurer) {
mMeasurer = measurer
mDependencyGraph.setMeasurer(measurer)
}
/**
* Returns the rtl status.
*
* @return true if in RTL, false otherwise.
*/
/**
* Set the rtl status. This has implications for Chains.
*
* @param isRtl true if we are in RTL.
*/
var isRtl = false
var mMetrics: Metrics? = null
fun fillMetrics(metrics: Metrics?) {
mMetrics = metrics
system.fillMetrics(metrics)
}
var system = LinearSystem()
protected set
var mPaddingLeft = 0
var mPaddingTop = 0
var mPaddingRight = 0
var mPaddingBottom = 0
var mHorizontalChainsSize = 0
var mVerticalChainsSize = 0
var mVerticalChainsArray = arrayOfNulls(4)
var mHorizontalChainsArray = arrayOfNulls(4)
var mGroupsWrapOptimized = false
var mHorizontalWrapOptimized = false
var mVerticalWrapOptimized = false
var mWrapFixedWidth = 0
var mWrapFixedHeight = 0
private var mOptimizationLevel = Optimizer.OPTIMIZATION_STANDARD
var mSkipSolver = false
/**
* Return true if the width given is too small for the content laid out
*/
var isWidthMeasuredTooSmall = false
private set
/**
* Return true if the height given is too small for the content laid out
*/
var isHeightMeasuredTooSmall = false
private set
/*-----------------------------------------------------------------------*/ // Construction
/*-----------------------------------------------------------------------*/
/**
* Default constructor
*/
constructor() {}
/**
* Constructor
*
* @param x x position
* @param y y position
* @param width width of the layout
* @param height height of the layout
*/
constructor(x: Int, y: Int, width: Int, height: Int) : super(x, y, width, height) {}
/**
* Constructor
*
* @param width width of the layout
* @param height height of the layout
*/
constructor(width: Int, height: Int) : super(width, height) {}
constructor(debugName: String?, width: Int, height: Int) : super(width, height) {
this.debugName = debugName
}
/**
* Returns the current optimization level
*
* @return
*/
/**
* Resolves the system directly when possible
*
* @param value optimization level
*/
var optimizationLevel: Int
get() = mOptimizationLevel
set(value) {
mOptimizationLevel = value
LinearSystem.USE_DEPENDENCY_ORDERING = optimizeFor(Optimizer.OPTIMIZATION_DEPENDENCY_ORDERING)
}
/**
* Returns true if the given feature should be optimized
*
* @param feature
* @return
*/
fun optimizeFor(feature: Int): Boolean {
return mOptimizationLevel and feature == feature
}
/**
* Specify the xml type for the container
*
* @return
*/
override var type: String?
get() = "ConstraintLayout"
set(type) {
super.type = type
}
override fun reset() {
system.reset()
mPaddingLeft = 0
mPaddingRight = 0
mPaddingTop = 0
mPaddingBottom = 0
mSkipSolver = false
super.reset()
}
var mDebugSolverPassCount = 0
// TODO weak references
private var verticalWrapMin: ConstraintAnchor? = null
private var horizontalWrapMin: ConstraintAnchor? = null
private var verticalWrapMax: ConstraintAnchor? = null
private var horizontalWrapMax: ConstraintAnchor? = null
fun addVerticalWrapMinVariable(top: ConstraintAnchor) {
if (verticalWrapMin == null || verticalWrapMin == null || top.finalValue > verticalWrapMin?.finalValue ?: 0) {
// TODO weak reference
verticalWrapMin = top
}
}
fun addHorizontalWrapMinVariable(left: ConstraintAnchor) {
if (horizontalWrapMin == null || horizontalWrapMin == null || left.finalValue > horizontalWrapMin?.finalValue ?: 0) {
// TODO weak reference
horizontalWrapMin = left
}
}
fun addVerticalWrapMaxVariable(bottom: ConstraintAnchor) {
if (verticalWrapMax == null || verticalWrapMax == null || bottom.finalValue > verticalWrapMax?.finalValue ?: 0) {
// TODO weak reference
verticalWrapMax = bottom
}
}
fun addHorizontalWrapMaxVariable(right: ConstraintAnchor) {
if (horizontalWrapMax == null || horizontalWrapMax == null || right.finalValue > horizontalWrapMax?.finalValue ?: 0) {
// TODO weak reference
horizontalWrapMax = right
}
}
private fun addMinWrap(constraintAnchor: ConstraintAnchor?, parentMin: SolverVariable?) {
val variable = system.createObjectVariable(constraintAnchor)
val wrapStrength = SolverVariable.STRENGTH_EQUALITY
system.addGreaterThan(variable, parentMin, 0, wrapStrength)
}
private fun addMaxWrap(constraintAnchor: ConstraintAnchor?, parentMax: SolverVariable?) {
val variable = system.createObjectVariable(constraintAnchor)
val wrapStrength = SolverVariable.STRENGTH_EQUALITY
system.addGreaterThan(parentMax, variable, 0, wrapStrength)
}
var widgetsToAdd = HashSet()
/**
* Add this widget to the solver
*
* @param system the solver we want to add the widget to
*/
fun addChildrenToSolver(system: LinearSystem?): Boolean {
if (DEBUG) {
println("\n#######################################")
println("## ADD CHILDREN TO SOLVER ($mDebugSolverPassCount) ##")
println("#######################################\n")
mDebugSolverPassCount++
}
val optimize = optimizeFor(Optimizer.OPTIMIZATION_GRAPH)
addToSolver(system!!, optimize)
val count = children.size
var hasBarriers = false
for (i in 0 until count) {
val widget = children[i]
widget.setInBarrier(HORIZONTAL, false)
widget.setInBarrier(VERTICAL, false)
if (widget is Barrier) {
hasBarriers = true
}
}
if (hasBarriers) {
for (i in 0 until count) {
val widget = children[i]
if (widget is Barrier) {
widget.markWidgets()
}
}
}
widgetsToAdd.clear()
for (i in 0 until count) {
val widget = children[i]
if (widget.addFirst()) {
if (widget is VirtualLayout) {
widgetsToAdd.add(widget)
} else {
widget.addToSolver(system, optimize)
}
}
}
// If we have virtual layouts, we need to add them to the solver in the correct
// order (in case they reference one another)
while (widgetsToAdd.size > 0) {
val numLayouts = widgetsToAdd.size
var layout: VirtualLayout? = null
for (widget in widgetsToAdd) {
layout = widget as VirtualLayout?
// we'll go through the virtual layouts that references others first, to give
// them a shot at setting their constraints.
if (layout!!.contains(widgetsToAdd)) {
layout.addToSolver(system, optimize)
widgetsToAdd.remove(layout)
break
}
}
if (numLayouts == widgetsToAdd.size) {
// looks we didn't find anymore dependency, let's add everything.
for (widget in widgetsToAdd) {
widget!!.addToSolver(system, optimize)
}
widgetsToAdd.clear()
}
}
if (LinearSystem.USE_DEPENDENCY_ORDERING) {
val widgetsToAdd = HashSet()
for (i in 0 until count) {
val widget = children[i]
if (!widget.addFirst()) {
widgetsToAdd.add(widget)
}
}
var orientation = VERTICAL
if (horizontalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT) {
orientation = HORIZONTAL
}
addChildrenToSolverByDependency(this, system, widgetsToAdd, orientation, false)
for (widget in widgetsToAdd) {
if (widget == null) continue
Optimizer.checkMatchParent(this, system, widget)
widget.addToSolver(system, optimize)
}
} else {
for (i in 0 until count) {
val widget = children[i]
if (widget is ConstraintWidgetContainer) {
val horizontalBehaviour = widget.mListDimensionBehaviors[DIMENSION_HORIZONTAL]
val verticalBehaviour = widget.mListDimensionBehaviors[DIMENSION_VERTICAL]
if (horizontalBehaviour === DimensionBehaviour.WRAP_CONTENT) {
widget.horizontalDimensionBehaviour = DimensionBehaviour.FIXED
}
if (verticalBehaviour === DimensionBehaviour.WRAP_CONTENT) {
widget.verticalDimensionBehaviour = DimensionBehaviour.FIXED
}
widget.addToSolver(system, optimize)
if (horizontalBehaviour === DimensionBehaviour.WRAP_CONTENT) {
widget.horizontalDimensionBehaviour = horizontalBehaviour
}
if (verticalBehaviour === DimensionBehaviour.WRAP_CONTENT) {
widget.verticalDimensionBehaviour = verticalBehaviour
}
} else {
Optimizer.checkMatchParent(this, system, widget)
if (!widget.addFirst()) {
widget.addToSolver(system, optimize)
}
}
}
}
if (mHorizontalChainsSize > 0) {
applyChainConstraints(this, system, null, HORIZONTAL)
}
if (mVerticalChainsSize > 0) {
applyChainConstraints(this, system, null, VERTICAL)
}
return true
}
/**
* Update the frame of the layout and its children from the solver
*
* @param system the solver we get the values from.
*/
fun updateChildrenFromSolver(system: LinearSystem?, flags: BooleanArray): Boolean {
flags[Optimizer.FLAG_RECOMPUTE_BOUNDS] = false
val optimize = optimizeFor(Optimizer.OPTIMIZATION_GRAPH)
updateFromSolver(system!!, optimize)
val count = children.size
var hasOverride = false
for (i in 0 until count) {
val widget = children[i]
widget.updateFromSolver(system, optimize)
if (widget.hasDimensionOverride()) {
hasOverride = true
}
}
return hasOverride
}
override fun updateFromRuns(updateHorizontal: Boolean, updateVertical: Boolean) {
super.updateFromRuns(updateHorizontal, updateVertical)
val count = children.size
for (i in 0 until count) {
val widget = children[i]
widget.updateFromRuns(updateHorizontal, updateVertical)
}
}
/**
* Set the padding on this container. It will apply to the position of the children.
*
* @param left left padding
* @param top top padding
* @param right right padding
* @param bottom bottom padding
*/
fun setPadding(left: Int, top: Int, right: Int, bottom: Int) {
mPaddingLeft = left
mPaddingTop = top
mPaddingRight = right
mPaddingBottom = bottom
}
/*-----------------------------------------------------------------------*/ // Overloaded methods from ConstraintWidget
/*-----------------------------------------------------------------------*/
var mMeasure = BasicMeasure.Measure()
/**
* Layout the tree of widgets
*/
override fun layout() {
if (DEBUG) {
println("\n#####################################")
println("## CL LAYOUT PASS ##")
println("#####################################\n")
mDebugSolverPassCount = 0
}
mX = 0
mY = 0
isWidthMeasuredTooSmall = false
isHeightMeasuredTooSmall = false
val count = children.size
var preW = max(0, width)
var preH = max(0, height)
val originalVerticalDimensionBehaviour = mListDimensionBehaviors[DIMENSION_VERTICAL]
val originalHorizontalDimensionBehaviour = mListDimensionBehaviors[DIMENSION_HORIZONTAL]
if (DEBUG_LAYOUT) {
println(
"layout with preW: " + preW + " (" + mListDimensionBehaviors[DIMENSION_HORIZONTAL]
+ ") preH: " + preH + " (" + mListDimensionBehaviors[DIMENSION_VERTICAL] + ")"
)
}
if (mMetrics != null) {
mMetrics!!.layouts++
}
var wrap_override = false
if (LinearSystem.FULL_DEBUG) {
println("OPTIMIZATION LEVEL $mOptimizationLevel")
}
// Only try the direct optimization in the first layout pass
if (pass == 0 && Optimizer.enabled(mOptimizationLevel, Optimizer.OPTIMIZATION_DIRECT)) {
if (LinearSystem.FULL_DEBUG) {
println("Direct pass " + myCounter++)
}
solvingPass(this, measurer)
if (LinearSystem.FULL_DEBUG) {
println("Direct pass done.")
}
for (i in 0 until count) {
val child = children[i]
if (LinearSystem.FULL_DEBUG) {
if (child.isInHorizontalChain) {
print("H")
} else {
print(" ")
}
if (child.isInVerticalChain) {
print("V")
} else {
print(" ")
}
if (child.isResolvedHorizontally && child.isResolvedVertically) {
print("*")
} else {
print(" ")
}
println(
"[" + i + "] child " + child.debugName
+ " H: " + child.isResolvedHorizontally
+ " V: " + child.isResolvedVertically
)
}
if (child.isMeasureRequested
&& child !is Guideline
&& child !is Barrier
&& child !is VirtualLayout
&& !child.isInVirtualLayout
) {
val widthBehavior = child.getDimensionBehaviour(HORIZONTAL)
val heightBehavior = child.getDimensionBehaviour(VERTICAL)
val skip =
widthBehavior === DimensionBehaviour.MATCH_CONSTRAINT && child.mMatchConstraintDefaultWidth != MATCH_CONSTRAINT_WRAP && heightBehavior === DimensionBehaviour.MATCH_CONSTRAINT && child.mMatchConstraintDefaultHeight != MATCH_CONSTRAINT_WRAP
if (!skip) {
val measure = BasicMeasure.Measure()
measure(0, child, mMeasurer, measure, BasicMeasure.Measure.SELF_DIMENSIONS)
}
}
}
// let's measure children
if (LinearSystem.FULL_DEBUG) {
println("Direct pass all done.")
}
} else {
if (LinearSystem.FULL_DEBUG) {
println("No DIRECT PASS")
}
}
if (count > 2 && (originalHorizontalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT
|| originalVerticalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT)
&& Optimizer.enabled(mOptimizationLevel, Optimizer.OPTIMIZATION_GROUPING)
) {
if (simpleSolvingPass(this, measurer)) {
if (originalHorizontalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT) {
if (preW < width && preW > 0) {
if (DEBUG_LAYOUT) {
println("Override width $width to $preH")
}
width = preW
isWidthMeasuredTooSmall = true
} else {
preW = width
}
}
if (originalVerticalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT) {
if (preH < height && preH > 0) {
if (DEBUG_LAYOUT) {
println("Override height $height to $preH")
}
height = preH
isHeightMeasuredTooSmall = true
} else {
preH = height
}
}
wrap_override = true
if (DEBUG_LAYOUT) {
println(
"layout post opt, preW: " + preW + " (" + mListDimensionBehaviors[DIMENSION_HORIZONTAL]
+ ") preH: " + preH + " (" + mListDimensionBehaviors[DIMENSION_VERTICAL] + "), new size "
+ width + " x " + height
)
}
}
}
val useGraphOptimizer = optimizeFor(Optimizer.OPTIMIZATION_GRAPH) || optimizeFor(Optimizer.OPTIMIZATION_GRAPH_WRAP)
system.graphOptimizer = false
system.newgraphOptimizer = false
if (mOptimizationLevel != Optimizer.OPTIMIZATION_NONE
&& useGraphOptimizer
) {
system.newgraphOptimizer = true
}
var countSolve = 0
val allChildren = children
val hasWrapContent = horizontalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT || verticalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT
// Reset the chains before iterating on our children
resetChains()
countSolve = 0
// Before we solve our system, we should call layout() on any
// of our children that is a container.
for (i in 0 until count) {
val widget = children[i]
if (widget is WidgetContainer) {
widget.layout()
}
}
val optimize = optimizeFor(Optimizer.OPTIMIZATION_GRAPH)
// Now let's solve our system as usual
var needsSolving = true
while (needsSolving) {
countSolve++
try {
system.reset()
resetChains()
if (DEBUG) {
var debugName = debugName
if (debugName == null) {
debugName = "root"
}
setDebugSolverName(system, debugName)
for (i in 0 until count) {
val widget = children[i]
if (widget.debugName != null) {
widget.setDebugSolverName(system, widget.debugName!!)
}
}
} else {
createObjectVariables(system)
for (i in 0 until count) {
val widget = children[i]
widget.createObjectVariables(system)
}
}
needsSolving = addChildrenToSolver(system)
// TODO weak references
if (verticalWrapMin != null && verticalWrapMin != null) {
addMinWrap(verticalWrapMin, system.createObjectVariable(mTop))
verticalWrapMin = null
}
if (verticalWrapMax != null && verticalWrapMax != null) {
addMaxWrap(verticalWrapMax, system.createObjectVariable(mBottom))
verticalWrapMax = null
}
if (horizontalWrapMin != null && horizontalWrapMin != null) {
addMinWrap(horizontalWrapMin, system.createObjectVariable(mLeft))
horizontalWrapMin = null
}
if (horizontalWrapMax != null && horizontalWrapMax != null) {
addMaxWrap(horizontalWrapMax, system.createObjectVariable(mRight))
horizontalWrapMax = null
}
if (needsSolving) {
system.minimize()
}
} catch (e: Exception) {
e.printStackTrace()
println("EXCEPTION : $e")
}
needsSolving = if (needsSolving) {
updateChildrenFromSolver(system, Optimizer.flags)
} else {
updateFromSolver(system, optimize)
for (i in 0 until count) {
val widget = children[i]
widget.updateFromSolver(system, optimize)
}
false
}
if (hasWrapContent && countSolve < MAX_ITERATIONS && Optimizer.flags[Optimizer.FLAG_RECOMPUTE_BOUNDS]) {
// let's get the new bounds
var maxX = 0
var maxY = 0
for (i in 0 until count) {
val widget = children[i]
maxX = max(maxX, widget.mX + widget.width)
maxY = max(maxY, widget.mY + widget.height)
}
maxX = max(mMinWidth, maxX)
maxY = max(mMinHeight, maxY)
if (originalHorizontalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT) {
if (width < maxX) {
if (DEBUG_LAYOUT) {
println(
"layout override width from $width vs $maxX"
)
}
width = maxX
mListDimensionBehaviors[DIMENSION_HORIZONTAL] = DimensionBehaviour.WRAP_CONTENT // force using the solver
wrap_override = true
needsSolving = true
}
}
if (originalVerticalDimensionBehaviour === DimensionBehaviour.WRAP_CONTENT) {
if (height < maxY) {
if (DEBUG_LAYOUT) {
println(
"layout override height from $height vs $maxY"
)
}
height = maxY
mListDimensionBehaviors[DIMENSION_VERTICAL] = DimensionBehaviour.WRAP_CONTENT // force using the solver
wrap_override = true
needsSolving = true
}
}
}
if (true) {
var width = max(mMinWidth, width)
if (width > this.width) {
if (DEBUG_LAYOUT) {
println(
"layout override 2, width from ${this.width} vs $width"
)
}
this.width = width
mListDimensionBehaviors[DIMENSION_HORIZONTAL] = DimensionBehaviour.FIXED
wrap_override = true
needsSolving = true
}
var height = max(mMinHeight, height)
if (height > this.height) {
if (DEBUG_LAYOUT) {
println(
"layout override 2, height from ${this.height} vs $height"
)
}
this.height = height
mListDimensionBehaviors[DIMENSION_VERTICAL] = DimensionBehaviour.FIXED
wrap_override = true
needsSolving = true
}
if (!wrap_override) {
if (mListDimensionBehaviors[DIMENSION_HORIZONTAL] === DimensionBehaviour.WRAP_CONTENT
&& preW > 0
) {
if (width > preW) {
if (DEBUG_LAYOUT) {
println(
"layout override 3, width from " + width + " vs "
+ preW
)
}
isWidthMeasuredTooSmall = true
wrap_override = true
mListDimensionBehaviors[DIMENSION_HORIZONTAL] = DimensionBehaviour.FIXED
width = preW
needsSolving = true
}
}
if (mListDimensionBehaviors[DIMENSION_VERTICAL] === DimensionBehaviour.WRAP_CONTENT
&& preH > 0
) {
if (height > preH) {
if (DEBUG_LAYOUT) {
println(
"layout override 3, height from " + height + " vs "
+ preH
)
}
isHeightMeasuredTooSmall = true
wrap_override = true
mListDimensionBehaviors[DIMENSION_VERTICAL] = DimensionBehaviour.FIXED
height = preH
needsSolving = true
}
}
}
if (countSolve > MAX_ITERATIONS) {
needsSolving = false
}
}
}
if (DEBUG_LAYOUT) {
println(
"Solved system in " + countSolve + " iterations (" + width + " x "
+ height + ")"
)
}
children = allChildren
if (wrap_override) {
mListDimensionBehaviors[DIMENSION_HORIZONTAL] = originalHorizontalDimensionBehaviour
mListDimensionBehaviors[DIMENSION_VERTICAL] = originalVerticalDimensionBehaviour
}
resetSolverVariables(system.cache)
}
/**
* Indicates if the container knows how to layout its content on its own
*
* @return true if the container does the layout, false otherwise
*/
fun handlesInternalConstraints(): Boolean {
return false
}
/*-----------------------------------------------------------------------*/ // Guidelines
/*-----------------------------------------------------------------------*/
/**
* Accessor to the vertical guidelines contained in the table.
*
* @return array of guidelines
*/
val verticalGuidelines: ArrayList
get() {
val guidelines = ArrayList()
var i = 0
val mChildrenSize = children.size
while (i < mChildrenSize) {
val widget = children[i]
if (widget is Guideline) {
val guideline = widget
if (guideline.orientation == Guideline.VERTICAL) {
guidelines.add(guideline)
}
}
i++
}
return guidelines
}
/**
* Accessor to the horizontal guidelines contained in the table.
*
* @return array of guidelines
*/
val horizontalGuidelines: ArrayList
get() {
val guidelines = ArrayList()
var i = 0
val mChildrenSize = children.size
while (i < mChildrenSize) {
val widget = children[i]
if (widget is Guideline) {
val guideline = widget
if (guideline.orientation == Guideline.HORIZONTAL) {
guidelines.add(guideline)
}
}
i++
}
return guidelines
}
/*-----------------------------------------------------------------------*/ // Chains
/*-----------------------------------------------------------------------*/
/**
* Reset the chains array. Need to be called before layout.
*/
private fun resetChains() {
mHorizontalChainsSize = 0
mVerticalChainsSize = 0
}
/**
* Add the chain which constraintWidget is part of. Called by ConstraintWidget::addToSolver()
*
* @param constraintWidget
* @param type HORIZONTAL or VERTICAL chain
*/
fun addChain(constraintWidget: ConstraintWidget, type: Int) {
if (type == HORIZONTAL) {
addHorizontalChain(constraintWidget)
} else if (type == VERTICAL) {
addVerticalChain(constraintWidget)
}
}
/**
* Add a widget to the list of horizontal chains. The widget is the left-most widget
* of the chain which doesn't have a left dual connection.
*
* @param widget widget starting the chain
*/
private fun addHorizontalChain(widget: ConstraintWidget) {
if (mHorizontalChainsSize + 1 >= mHorizontalChainsArray.size) {
mHorizontalChainsArray = mHorizontalChainsArray.copyOf(mHorizontalChainsArray.size * 2)
}
mHorizontalChainsArray[mHorizontalChainsSize] = ChainHead(widget, HORIZONTAL, isRtl)
mHorizontalChainsSize++
}
/**
* Add a widget to the list of vertical chains. The widget is the top-most widget
* of the chain which doesn't have a top dual connection.
*
* @param widget widget starting the chain
*/
private fun addVerticalChain(widget: ConstraintWidget) {
if (mVerticalChainsSize + 1 >= mVerticalChainsArray.size) {
mVerticalChainsArray = mVerticalChainsArray.copyOf(mVerticalChainsArray.size * 2)
}
mVerticalChainsArray[mVerticalChainsSize] = ChainHead(widget, VERTICAL, isRtl)
mVerticalChainsSize++
}
/**
* Keep track of the # of passes
* @param pass
*/
fun setPass(pass: Int) {
this.pass = pass
}
override fun getSceneString(ret: StringBuilder) {
ret.append("$stringId:{\n")
ret.append(" actualWidth:$mWidth")
ret.append("\n")
ret.append(" actualHeight:$mHeight")
ret.append("\n")
val children = children
for (child in children) {
child.getSceneString(ret)
ret.append(",\n")
}
ret.append("}")
}
companion object {
private const val MAX_ITERATIONS = 8
private const val DEBUG = LinearSystem.FULL_DEBUG
private const val DEBUG_LAYOUT = false
const val DEBUG_GRAPH = false
fun measure(level: Int, widget: ConstraintWidget, measurer: BasicMeasure.Measurer?, measure: BasicMeasure.Measure, measureStrategy: Int): Boolean {
if (DEBUG) {
println(ls(level) + "(M) call to measure " + widget.debugName)
}
if (measurer == null) {
return false
}
if (widget.visibility == GONE || widget is Guideline
|| widget is Barrier
) {
if (DEBUG) {
println(ls(level) + "(M) no measure needed for " + widget.debugName)
}
measure.measuredWidth = 0
measure.measuredHeight = 0
return false
}
measure.horizontalBehavior = widget.horizontalDimensionBehaviour
measure.verticalBehavior = widget.verticalDimensionBehaviour
measure.horizontalDimension = widget.width
measure.verticalDimension = widget.height
measure.measuredNeedsSolverPass = false
measure.measureStrategy = measureStrategy
var horizontalMatchConstraints = measure.horizontalBehavior === DimensionBehaviour.MATCH_CONSTRAINT
var verticalMatchConstraints = measure.verticalBehavior === DimensionBehaviour.MATCH_CONSTRAINT
val horizontalUseRatio = horizontalMatchConstraints && widget.dimensionRatio > 0
val verticalUseRatio = verticalMatchConstraints && widget.dimensionRatio > 0
if (horizontalMatchConstraints && widget.hasDanglingDimension(HORIZONTAL)
&& widget.mMatchConstraintDefaultWidth == MATCH_CONSTRAINT_SPREAD && !horizontalUseRatio
) {
horizontalMatchConstraints = false
measure.horizontalBehavior = DimensionBehaviour.WRAP_CONTENT
if (verticalMatchConstraints && widget.mMatchConstraintDefaultHeight == MATCH_CONSTRAINT_SPREAD) {
// if match x match, size would be zero.
measure.horizontalBehavior = DimensionBehaviour.FIXED
}
}
if (verticalMatchConstraints && widget.hasDanglingDimension(VERTICAL)
&& widget.mMatchConstraintDefaultHeight == MATCH_CONSTRAINT_SPREAD && !verticalUseRatio
) {
verticalMatchConstraints = false
measure.verticalBehavior = DimensionBehaviour.WRAP_CONTENT
if (horizontalMatchConstraints && widget.mMatchConstraintDefaultWidth == MATCH_CONSTRAINT_SPREAD) {
// if match x match, size would be zero.
measure.verticalBehavior = DimensionBehaviour.FIXED
}
}
if (widget.isResolvedHorizontally) {
horizontalMatchConstraints = false
measure.horizontalBehavior = DimensionBehaviour.FIXED
}
if (widget.isResolvedVertically) {
verticalMatchConstraints = false
measure.verticalBehavior = DimensionBehaviour.FIXED
}
if (horizontalUseRatio) {
if (widget.mResolvedMatchConstraintDefault[HORIZONTAL] == MATCH_CONSTRAINT_RATIO_RESOLVED) {
measure.horizontalBehavior = DimensionBehaviour.FIXED
} else if (!verticalMatchConstraints) {
// let's measure here
val measuredHeight: Int
if (measure.verticalBehavior === DimensionBehaviour.FIXED) {
measuredHeight = measure.verticalDimension
} else {
measure.horizontalBehavior = DimensionBehaviour.WRAP_CONTENT
measurer.measure(widget, measure)
measuredHeight = measure.measuredHeight
}
measure.horizontalBehavior = DimensionBehaviour.FIXED
// regardless of which side we are using for the ratio, getDimensionRatio() already
// made sure that it's expressed in WxH format, so we can simply go and multiply
measure.horizontalDimension = (widget.dimensionRatio * measuredHeight).toInt()
if (DEBUG) {
println("(M) Measured once for ratio on horizontal side...")
}
}
}
if (verticalUseRatio) {
if (widget.mResolvedMatchConstraintDefault[VERTICAL] == MATCH_CONSTRAINT_RATIO_RESOLVED) {
measure.verticalBehavior = DimensionBehaviour.FIXED
} else if (!horizontalMatchConstraints) {
// let's measure here
val measuredWidth: Int
if (measure.horizontalBehavior === DimensionBehaviour.FIXED) {
measuredWidth = measure.horizontalDimension
} else {
measure.verticalBehavior = DimensionBehaviour.WRAP_CONTENT
measurer.measure(widget, measure)
measuredWidth = measure.measuredWidth
}
measure.verticalBehavior = DimensionBehaviour.FIXED
if (widget.dimensionRatioSide == -1) {
// regardless of which side we are using for the ratio, getDimensionRatio() already
// made sure that it's expressed in WxH format, so we can simply go and divide
measure.verticalDimension = (measuredWidth / widget.dimensionRatio).toInt()
} else {
// getDimensionRatio() already got reverted, so we can simply multiply
measure.verticalDimension = (widget.dimensionRatio * measuredWidth).toInt()
}
if (DEBUG) {
println("(M) Measured once for ratio on vertical side...")
}
}
}
measurer.measure(widget, measure)
widget.width = measure.measuredWidth
widget.height = measure.measuredHeight
widget.hasBaseline = measure.measuredHasBaseline
widget.baselineDistance = measure.measuredBaseline
measure.measureStrategy = BasicMeasure.Measure.SELF_DIMENSIONS
if (DEBUG) {
println(
"(M) Measured " + widget.debugName + " with : "
+ widget.horizontalDimensionBehaviour + " x " + widget.verticalDimensionBehaviour
+ " => " + widget.width + " x " + widget.height
)
}
return measure.measuredNeedsSolverPass
}
var myCounter = 0
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy