| /* |
| * 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.test |
| |
| import android.app.Activity |
| import android.os.Handler |
| import android.view.View |
| import android.view.ViewGroup |
| import android.view.ViewTreeObserver |
| import androidx.compose.Composable |
| import androidx.test.rule.ActivityTestRule |
| import androidx.ui.core.AlignmentLine |
| import androidx.ui.core.AndroidComposeView |
| import androidx.ui.core.Constraints |
| import androidx.ui.core.Layout |
| import androidx.ui.core.LayoutDirection |
| import androidx.ui.core.Modifier |
| import androidx.ui.core.OnPositioned |
| import androidx.ui.core.Ref |
| import androidx.ui.core.enforce |
| import androidx.ui.core.setContent |
| import androidx.ui.layout.Arrangement |
| import androidx.ui.layout.Constraints |
| import androidx.ui.layout.DpConstraints |
| import androidx.ui.unit.Density |
| import androidx.ui.unit.IntPx |
| import androidx.ui.unit.IntPxSize |
| import androidx.ui.unit.PxPosition |
| import androidx.ui.unit.PxSize |
| import androidx.ui.unit.ipx |
| import androidx.ui.unit.px |
| import androidx.ui.unit.round |
| import androidx.ui.unit.toPx |
| import org.junit.Assert.assertEquals |
| import org.junit.Assert.assertNotNull |
| import org.junit.Assert.assertTrue |
| import org.junit.Assert.fail |
| import org.junit.Before |
| import org.junit.Rule |
| import java.util.concurrent.CountDownLatch |
| import java.util.concurrent.TimeUnit |
| |
| open class LayoutTest { |
| @get:Rule |
| val activityTestRule = ActivityTestRule<TestActivity>( |
| TestActivity::class.java |
| ) |
| lateinit var activity: TestActivity |
| lateinit var handler: Handler |
| internal lateinit var density: Density |
| |
| @Before |
| fun setup() { |
| activity = activityTestRule.activity |
| density = Density(activity) |
| activity.hasFocusLatch.await(5, TimeUnit.SECONDS) |
| |
| // Kotlin IR compiler doesn't seem too happy with auto-conversion from |
| // lambda to Runnable, so separate it here |
| val runnable: Runnable = object : Runnable { |
| override fun run() { |
| handler = Handler() |
| } |
| } |
| activityTestRule.runOnUiThread(runnable) |
| } |
| |
| internal fun show(composable: @Composable() () -> Unit) { |
| val runnable: Runnable = object : Runnable { |
| override fun run() { |
| activity.setContent(composable) |
| } |
| } |
| activityTestRule.runOnUiThread(runnable) |
| } |
| |
| internal fun findAndroidComposeView(): AndroidComposeView { |
| return findAndroidComposeView(activity) |
| } |
| |
| internal fun findAndroidComposeView(activity: Activity): AndroidComposeView { |
| val contentViewGroup = activity.findViewById<ViewGroup>(android.R.id.content) |
| return findAndroidComposeView(contentViewGroup)!! |
| } |
| |
| internal fun findAndroidComposeView(parent: ViewGroup): AndroidComposeView? { |
| for (index in 0 until parent.childCount) { |
| val child = parent.getChildAt(index) |
| if (child is AndroidComposeView) { |
| return child |
| } else if (child is ViewGroup) { |
| val composeView = findAndroidComposeView(child) |
| if (composeView != null) { |
| return composeView |
| } |
| } |
| } |
| return null |
| } |
| |
| internal fun waitForDraw(view: View) { |
| val viewDrawLatch = CountDownLatch(1) |
| val listener = object : ViewTreeObserver.OnDrawListener { |
| override fun onDraw() { |
| viewDrawLatch.countDown() |
| } |
| } |
| view.post(object : Runnable { |
| override fun run() { |
| view.viewTreeObserver.addOnDrawListener(listener) |
| view.invalidate() |
| } |
| }) |
| assertTrue(viewDrawLatch.await(1, TimeUnit.SECONDS)) |
| } |
| |
| @Composable |
| internal fun SaveLayoutInfo( |
| size: Ref<IntPxSize>, |
| position: Ref<PxPosition>, |
| positionedLatch: CountDownLatch |
| ) { |
| OnPositioned(onPositioned = { coordinates -> |
| size.value = IntPxSize(coordinates.size.width, coordinates.size.height) |
| position.value = coordinates.localToGlobal(PxPosition(0.px, 0.px)) |
| positionedLatch.countDown() |
| }) |
| } |
| |
| internal fun testIntrinsics( |
| vararg layouts: @Composable() () -> Unit, |
| test: ((IntPx) -> IntPx, (IntPx) -> IntPx, (IntPx) -> IntPx, (IntPx) -> IntPx) -> Unit |
| ) { |
| layouts.forEach { layout -> |
| val layoutLatch = CountDownLatch(1) |
| show { |
| Layout( |
| layout, |
| minIntrinsicWidthMeasureBlock = { _, _ -> 0.ipx }, |
| minIntrinsicHeightMeasureBlock = { _, _ -> 0.ipx }, |
| maxIntrinsicWidthMeasureBlock = { _, _ -> 0.ipx }, |
| maxIntrinsicHeightMeasureBlock = { _, _ -> 0.ipx } |
| ) { measurables, _ -> |
| val measurable = measurables.first() |
| test( |
| { h -> measurable.minIntrinsicWidth(h) }, |
| { w -> measurable.minIntrinsicHeight(w) }, |
| { h -> measurable.maxIntrinsicWidth(h) }, |
| { w -> measurable.maxIntrinsicHeight(w) } |
| ) |
| layoutLatch.countDown() |
| layout(0.ipx, 0.ipx) {} |
| } |
| } |
| assertTrue(layoutLatch.await(1, TimeUnit.SECONDS)) |
| } |
| } |
| |
| @Composable |
| internal fun FixedSizeLayout( |
| width: IntPx, |
| height: IntPx, |
| alignmentLines: Map<AlignmentLine, IntPx> |
| ) { |
| Layout({}) { _, constraints -> |
| layout( |
| width.coerceIn(constraints.minWidth, constraints.maxWidth), |
| height.coerceIn(constraints.minHeight, constraints.maxHeight), |
| alignmentLines |
| ) {} |
| } |
| } |
| |
| @Composable |
| internal fun WithInfiniteConstraints(children: @Composable() () -> Unit) { |
| Layout(children) { measurables, _ -> |
| val placeables = measurables.map { it.measure(Constraints()) } |
| layout(0.ipx, 0.ipx) { |
| placeables.forEach { it.place(0.ipx, 0.ipx) } |
| } |
| } |
| } |
| |
| @Composable |
| internal fun ConstrainedBox( |
| constraints: DpConstraints, |
| modifier: Modifier = Modifier.None, |
| children: @Composable() () -> Unit |
| ) { |
| Layout( |
| children, |
| modifier = modifier, |
| minIntrinsicWidthMeasureBlock = { measurables, h -> |
| val width = measurables.firstOrNull()?.minIntrinsicWidth(h) ?: 0.ipx |
| width.coerceIn(constraints.minWidth.toIntPx(), constraints.maxWidth.toIntPx()) |
| }, |
| minIntrinsicHeightMeasureBlock = { measurables, w -> |
| val height = measurables.firstOrNull()?.minIntrinsicHeight(w) ?: 0.ipx |
| height.coerceIn(constraints.minHeight.toIntPx(), constraints.maxHeight.toIntPx()) |
| }, |
| maxIntrinsicWidthMeasureBlock = { measurables, h -> |
| val width = measurables.firstOrNull()?.maxIntrinsicWidth(h) ?: 0.ipx |
| width.coerceIn(constraints.minWidth.toIntPx(), constraints.maxWidth.toIntPx()) |
| }, |
| maxIntrinsicHeightMeasureBlock = { measurables, w -> |
| val height = measurables.firstOrNull()?.maxIntrinsicHeight(w) ?: 0.ipx |
| height.coerceIn(constraints.minHeight.toIntPx(), constraints.maxHeight.toIntPx()) |
| } |
| ) { measurables, incomingConstraints -> |
| val measurable = measurables.firstOrNull() |
| val childConstraints = Constraints(constraints).enforce(incomingConstraints) |
| val placeable = measurable?.measure(childConstraints) |
| |
| val layoutWidth = placeable?.width ?: childConstraints.minWidth |
| val layoutHeight = placeable?.height ?: childConstraints.minHeight |
| layout(layoutWidth, layoutHeight) { |
| placeable?.place(IntPx.Zero, IntPx.Zero) |
| } |
| } |
| } |
| |
| internal fun assertEquals(expected: PxSize?, actual: PxSize?) { |
| assertNotNull("Null expected size", expected) |
| expected as PxSize |
| assertNotNull("Null actual size", actual) |
| actual as PxSize |
| |
| assertEquals( |
| "Expected width ${expected.width.value} but obtained ${actual.width.value}", |
| expected.width.value, |
| actual.width.value, |
| 0f |
| ) |
| assertEquals( |
| "Expected height ${expected.height.value} but obtained ${actual.height.value}", |
| expected.height.value, |
| actual.height.value, |
| 0f |
| ) |
| if (actual.width.value != actual.width.value.toInt().toFloat()) { |
| fail("Expected integer width") |
| } |
| if (actual.height.value != actual.height.value.toInt().toFloat()) { |
| fail("Expected integer height") |
| } |
| } |
| |
| internal fun assertEquals(expected: PxPosition?, actual: PxPosition?) { |
| assertNotNull("Null expected position", expected) |
| expected as PxPosition |
| assertNotNull("Null actual position", actual) |
| actual as PxPosition |
| |
| assertEquals( |
| "Expected x ${expected.x.value} but obtained ${actual.x.value}", |
| expected.x.value, |
| actual.x.value, |
| 0f |
| ) |
| assertEquals( |
| "Expected y ${expected.y.value} but obtained ${actual.y.value}", |
| expected.y.value, |
| actual.y.value, |
| 0f |
| ) |
| if (actual.x.value != actual.x.value.toInt().toFloat()) { |
| fail("Expected integer x coordinate") |
| } |
| if (actual.y.value != actual.y.value.toInt().toFloat()) { |
| fail("Expected integer y coordinate") |
| } |
| } |
| |
| internal fun assertEquals(expected: IntPx, actual: IntPx) { |
| assertEquals( |
| "Expected $expected but obtained $actual", |
| expected.value.toFloat(), |
| actual.value.toFloat(), |
| 0f |
| ) |
| } |
| |
| internal val customVerticalArrangement = object : Arrangement.Vertical { |
| override fun arrange( |
| totalSize: IntPx, |
| size: List<IntPx>, |
| layoutDirection: LayoutDirection |
| ): List<IntPx> { |
| val positions = mutableListOf<IntPx>() |
| var current = 0.px |
| val usedSpace = size.fold(0.ipx) { sum, e -> sum + e } |
| val step = if (size.size < 2) { |
| 0.px |
| } else { |
| (totalSize - usedSpace).toPx() * 2 / (size.lastIndex * size.size) |
| } |
| size.forEachIndexed { i, childSize -> |
| current += step * i |
| positions.add(current.round()) |
| current += childSize.toPx() |
| } |
| return positions |
| } |
| } |
| |
| internal val customHorizontalArrangement = object : Arrangement.Horizontal { |
| override fun arrange( |
| totalSize: IntPx, |
| size: List<IntPx>, |
| layoutDirection: LayoutDirection |
| ): List<IntPx> { |
| val positions = mutableListOf<IntPx>() |
| var current = 0.px |
| if (layoutDirection == LayoutDirection.Rtl) { |
| size.forEach { |
| positions.add(current.round()) |
| current += it |
| } |
| } else { |
| val usedSpace = size.fold(0.ipx) { sum, e -> sum + e } |
| val step = if (size.size < 2) { |
| 0.px |
| } else { |
| (totalSize - usedSpace).toPx() * 2 / (size.lastIndex * size.size) |
| } |
| size.forEachIndexed { i, childSize -> |
| current += step * i |
| positions.add(current.round()) |
| current += childSize.toPx() |
| } |
| } |
| return positions |
| } |
| } |
| } |