Skip to content

Commit

Permalink
fix UIKitView z-order (#965)
Browse files Browse the repository at this point in the history
Fix Z-order issue with UIKitView
JetBrains/compose-multiplatform#4004
  • Loading branch information
dima-avdeev-jb authored Jan 22, 2024
1 parent a53700a commit 6526397
Show file tree
Hide file tree
Showing 6 changed files with 66 additions and 7 deletions.
52 changes: 52 additions & 0 deletions compose/mpp/demo/src/uikitMain/kotlin/UIKitViewOrder.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
/*
* Copyright 2023 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.
*/

import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.offset
import androidx.compose.foundation.layout.size
import androidx.compose.mpp.demo.Screen
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.interop.UIKitView
import androidx.compose.ui.unit.dp
import platform.UIKit.UIColor
import platform.UIKit.UIView

/**
* Issue https://github.com/JetBrains/compose-multiplatform/issues/4004
*/
val UIKitViewOrder = Screen.Example("UIKitViewOrder") {
Box(modifier = Modifier.fillMaxSize().background(Color.Yellow)) {
UIKitView(
factory = {
UIView().apply {
backgroundColor = UIColor.blueColor
}
},
modifier = Modifier.size(100.dp),
)
UIKitView(
factory = {
UIView().apply {
backgroundColor = UIColor.redColor
}
},
modifier = Modifier.size(100.dp).offset(50.dp, 50.dp),
)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package androidx.compose.mpp.demo

import NativeModalWithNaviationExample
import SwiftUIInteropExample
import UIKitViewOrder
import androidx.compose.runtime.*
import androidx.compose.ui.main.defaultUIKitMain
import androidx.compose.ui.window.ComposeUIViewController
Expand All @@ -25,6 +26,7 @@ fun IosDemo(arg: String, makeHostingController: ((Int) -> UIViewController)? = n
extraScreens = listOf(
IosBugs,
NativeModalWithNaviationExample,
UIKitViewOrder,
) + listOf(makeHostingController).mapNotNull {
it?.let {
SwiftUIInteropExample(it)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,6 @@ package androidx.compose.ui.interop
import androidx.compose.runtime.staticCompositionLocalOf
import platform.UIKit.UIView

internal val LocalLayerContainer = staticCompositionLocalOf<UIView> {
internal val LocalInteropContainer = staticCompositionLocalOf<UIView> {
error("CompositionLocal LayerContainer not provided")
}
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,6 @@ import androidx.compose.runtime.setValue
import androidx.compose.runtime.snapshots.SnapshotStateObserver
import androidx.compose.ui.*
import androidx.compose.ui.draw.drawBehind
import androidx.compose.ui.geometry.Offset
import androidx.compose.ui.graphics.BlendMode
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.input.pointer.InteropViewCatchPointerModifier
Expand Down Expand Up @@ -83,7 +82,7 @@ fun <T : UIView> UIKitView(
) {
// TODO: adapt UIKitView to reuse inside LazyColumn like in AndroidView:
// https://developer.android.com/reference/kotlin/androidx/compose/ui/viewinterop/package-summary#AndroidView(kotlin.Function1,kotlin.Function1,androidx.compose.ui.Modifier,kotlin.Function1,kotlin.Function1)
val rootView = LocalLayerContainer.current
val rootView = LocalInteropContainer.current
val embeddedInteropComponent = remember {
EmbeddedInteropView(
rootView = rootView,
Expand Down Expand Up @@ -179,7 +178,7 @@ fun <T : UIViewController> UIKitViewController(
) {
// TODO: adapt UIKitViewController to reuse inside LazyColumn like in AndroidView:
// https://developer.android.com/reference/kotlin/androidx/compose/ui/viewinterop/package-summary#AndroidView(kotlin.Function1,kotlin.Function1,androidx.compose.ui.Modifier,kotlin.Function1,kotlin.Function1)
val rootView = LocalLayerContainer.current
val rootView = LocalInteropContainer.current
val rootViewController = LocalUIViewController.current
val embeddedInteropComponent = remember {
EmbeddedInteropViewController(
Expand Down Expand Up @@ -296,7 +295,7 @@ private abstract class EmbeddedInteropComponent<T : Any>(
wrappingView = UIView().apply {
addSubview(view)
}
rootView.insertSubview(wrappingView, 0)
rootView.addSubview(wrappingView)
}

protected fun removeViewFromHierarchy(view: UIView) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import androidx.compose.ui.input.pointer.HistoricalChange
import androidx.compose.ui.input.pointer.PointerEventType
import androidx.compose.ui.input.pointer.PointerId
import androidx.compose.ui.input.pointer.PointerType
import androidx.compose.ui.interop.LocalInteropContainer
import androidx.compose.ui.interop.LocalUIKitInteropContext
import androidx.compose.ui.interop.UIKitInteropContext
import androidx.compose.ui.interop.UIKitInteropTransaction
Expand Down Expand Up @@ -146,6 +147,11 @@ internal class ComposeSceneMediator(
renderingUIViewFactory(renderDelegate)
}

/**
* Container for UIKitView and UIKitViewController
*/
private val interopViewContainer = UIView()

private val interactionView by lazy {
InteractionUIView(
keyboardEventHandler = keyboardEventHandler,
Expand Down Expand Up @@ -331,6 +337,7 @@ internal class ComposeSceneMediator(
this.onAttachedToWindow?.invoke()
focusStack?.pushAndFocus(interactionView)
}
container.addSubview(interopViewContainer)
container.addSubview(interactionView)
interactionView.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activateConstraints(
Expand Down Expand Up @@ -382,6 +389,7 @@ internal class ComposeSceneMediator(
LocalKeyboardOverlapHeight provides keyboardOverlapHeightState.value,
LocalSafeArea provides safeAreaState.value,
LocalLayoutMargins provides layoutMarginsState.value,
LocalInteropContainer provides interopViewContainer,
content = content
)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,6 @@ import androidx.compose.runtime.MutableState
import androidx.compose.runtime.mutableStateOf
import androidx.compose.ui.LocalSystemTheme
import androidx.compose.ui.SystemTheme
import androidx.compose.ui.interop.LocalLayerContainer
import androidx.compose.ui.interop.LocalUIViewController
import androidx.compose.ui.platform.PlatformContext
import androidx.compose.ui.platform.WindowInfoImpl
Expand Down Expand Up @@ -419,7 +418,6 @@ internal fun ProvideContainerCompositionLocals(
) = with(composeContainer) {
CompositionLocalProvider(
LocalUIViewController provides this,
LocalLayerContainer provides view,
LocalInterfaceOrientation provides interfaceOrientationState.value,
LocalSystemTheme provides systemThemeState.value,
content = content
Expand Down

0 comments on commit 6526397

Please sign in to comment.