[go: nahoru, domu]

blob: c687157814f3119b2d123fac6b3c92ed388212b8 [file] [log] [blame]
/*
* Copyright 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.ui.layout
import androidx.annotation.FloatRange
import androidx.ui.core.Constraints
import androidx.ui.core.IntrinsicMeasurable
import androidx.ui.core.IntrinsicMeasureScope
import androidx.ui.core.LayoutDirection
import androidx.ui.core.LayoutModifier
import androidx.ui.core.Measurable
import androidx.ui.core.MeasureScope
import androidx.ui.core.Modifier
import androidx.ui.core.satisfiedBy
import androidx.ui.unit.IntPx
import androidx.ui.unit.IntPxSize
import androidx.ui.unit.ipx
import androidx.ui.unit.isFinite
/**
* Attempts to size the content to match a specified aspect ratio by trying to match one of the
* incoming constraints in the following order:
* [Constraints.maxWidth], [Constraints.maxHeight], [Constraints.minWidth], [Constraints.minHeight].
* The size in the other dimension is determined by the aspect ratio.
*
* Example usage:
* @sample androidx.ui.layout.samples.SimpleAspectRatio
*
* @param ratio the desired width/height positive ratio
*/
fun Modifier.aspectRatio(
@FloatRange(from = 0.0, fromInclusive = false)
ratio: Float
) = this + AspectRatioModifier(ratio)
private data class AspectRatioModifier(val aspectRatio: Float) : LayoutModifier {
init {
require(aspectRatio > 0) { "aspectRatio $aspectRatio must be > 0" }
}
override fun MeasureScope.measure(
measurable: Measurable,
constraints: Constraints,
layoutDirection: LayoutDirection
): MeasureScope.MeasureResult {
val size = constraints.findSizeWith(aspectRatio)
val wrappedConstraints = if (size != null) {
Constraints.fixed(size.width, size.height)
} else {
constraints
}
val placeable = measurable.measure(wrappedConstraints)
return layout(placeable.width, placeable.height) {
placeable.place(0.ipx, 0.ipx)
}
}
override fun IntrinsicMeasureScope.minIntrinsicWidth(
measurable: IntrinsicMeasurable,
height: IntPx,
layoutDirection: LayoutDirection
) = if (height != IntPx.Infinity) {
height * aspectRatio
} else {
measurable.minIntrinsicWidth(height)
}
override fun IntrinsicMeasureScope.maxIntrinsicWidth(
measurable: IntrinsicMeasurable,
height: IntPx,
layoutDirection: LayoutDirection
) = if (height != IntPx.Infinity) {
height * aspectRatio
} else {
measurable.maxIntrinsicWidth(height)
}
override fun IntrinsicMeasureScope.minIntrinsicHeight(
measurable: IntrinsicMeasurable,
width: IntPx,
layoutDirection: LayoutDirection
) = if (width != IntPx.Infinity) {
width / aspectRatio
} else {
measurable.minIntrinsicHeight(width)
}
override fun IntrinsicMeasureScope.maxIntrinsicHeight(
measurable: IntrinsicMeasurable,
width: IntPx,
layoutDirection: LayoutDirection
) = if (width != IntPx.Infinity) {
width / aspectRatio
} else {
measurable.maxIntrinsicHeight(width)
}
private fun Constraints.findSizeWith(aspectRatio: Float): IntPxSize? {
return listOf(
IntPxSize(this.maxWidth, this.maxWidth / aspectRatio),
IntPxSize(this.maxHeight * aspectRatio, this.maxHeight),
IntPxSize(this.minWidth, this.minWidth / aspectRatio),
IntPxSize(this.minHeight * aspectRatio, this.minHeight)
).find {
this.satisfiedBy(it) &&
it.width != 0.ipx && it.height != 0.ipx &&
it.width.isFinite() && it.height.isFinite()
}
}
}