| /* |
| * 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.test |
| |
| import androidx.test.espresso.matcher.ViewMatchers |
| import androidx.compose.ui.geometry.Offset |
| import androidx.compose.ui.geometry.Rect |
| import androidx.compose.ui.node.ExperimentalLayoutNodeApi |
| import androidx.compose.ui.node.LayoutNode |
| import androidx.compose.ui.node.findClosestParentNode |
| import androidx.compose.ui.platform.AndroidOwner |
| import androidx.compose.ui.semantics.SemanticsNode |
| |
| @OptIn(ExperimentalLayoutNodeApi::class) |
| internal actual fun SemanticsNodeInteraction.checkIsDisplayed(): Boolean { |
| // hierarchy check - check layout nodes are visible |
| val errorMessageOnFail = "Failed to perform isDisplayed check." |
| val node = fetchSemanticsNode(errorMessageOnFail) |
| |
| fun isNotPlaced(node: LayoutNode): Boolean { |
| return !node.isPlaced |
| } |
| |
| val layoutNode = node.componentNode |
| if (isNotPlaced(layoutNode) || layoutNode.findClosestParentNode(::isNotPlaced) != null) { |
| return false |
| } |
| |
| (layoutNode.owner as? AndroidOwner)?.let { |
| if (!ViewMatchers.isDisplayed().matches(it.view)) { |
| return false |
| } |
| } |
| |
| // check node doesn't clip unintentionally (e.g. row too small for content) |
| val globalRect = node.globalBounds |
| if (!node.isInScreenBounds()) { |
| return false |
| } |
| |
| return (globalRect.width > 0f && globalRect.height > 0f) |
| } |
| |
| @OptIn(ExperimentalLayoutNodeApi::class) |
| internal actual fun SemanticsNode.clippedNodeBoundsInWindow(): Rect { |
| val composeView = (componentNode.owner as AndroidOwner).view |
| val rootLocationInWindow = intArrayOf(0, 0).let { |
| composeView.getLocationInWindow(it) |
| Offset(it[0].toFloat(), it[1].toFloat()) |
| } |
| return boundsInRoot.shift(rootLocationInWindow) |
| } |
| |
| @OptIn(ExperimentalLayoutNodeApi::class) |
| internal actual fun SemanticsNode.isInScreenBounds(): Boolean { |
| val composeView = (componentNode.owner as AndroidOwner).view |
| |
| // Window relative bounds of our node |
| val nodeBoundsInWindow = clippedNodeBoundsInWindow() |
| if (nodeBoundsInWindow.width == 0f || nodeBoundsInWindow.height == 0f) { |
| return false |
| } |
| |
| // Window relative bounds of our compose root view that are visible on the screen |
| val globalRootRect = android.graphics.Rect() |
| if (!composeView.getGlobalVisibleRect(globalRootRect)) { |
| return false |
| } |
| |
| return nodeBoundsInWindow.top >= globalRootRect.top && |
| nodeBoundsInWindow.left >= globalRootRect.left && |
| nodeBoundsInWindow.right <= globalRootRect.right && |
| nodeBoundsInWindow.bottom <= globalRootRect.bottom |
| } |