Skip to content

Commit

Permalink
Add UIViewBox Implementation
Browse files Browse the repository at this point in the history
  • Loading branch information
underscoretang committed Sep 30, 2023
1 parent 32eb24b commit ca1657a
Show file tree
Hide file tree
Showing 3 changed files with 218 additions and 4 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,216 @@
/*
* Copyright (C) 2022 Square, Inc.
*
* 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 app.cash.redwood.layout.uiview

import app.cash.redwood.Modifier
import app.cash.redwood.layout.api.Constraint
import app.cash.redwood.layout.api.CrossAxisAlignment
import app.cash.redwood.layout.modifier.HorizontalAlignment
import app.cash.redwood.layout.modifier.VerticalAlignment
import app.cash.redwood.layout.widget.Box
import app.cash.redwood.ui.Dp
import app.cash.redwood.ui.Margin
import app.cash.redwood.ui.dp
import app.cash.redwood.widget.UIViewChildren
import kotlinx.cinterop.CValue
import kotlinx.cinterop.convert
import kotlinx.cinterop.readValue
import kotlinx.cinterop.useContents
import platform.CoreGraphics.CGFloat
import platform.CoreGraphics.CGRectMake
import platform.CoreGraphics.CGRectZero
import platform.CoreGraphics.CGSize
import platform.CoreGraphics.CGSizeMake
import platform.UIKit.NSLayoutConstraint
import platform.UIKit.UIColor
import platform.UIKit.UIView
import platform.darwin.NSInteger
import platform.posix.INT_MAX

internal class UIViewBox : Box<UIView> {
override val value: View = View()

override var modifier: Modifier = Modifier

override val children = value.children

override fun width(width: Constraint) {
value.widthConstraint = width
}

override fun minWidth(minWidth: Dp) {
value.minWidth = minWidth
}

override fun maxWidth(maxWidth: Dp) {
value.maxWidth = maxWidth
}

override fun height(height: Constraint) {
value.heightConstraint = height
}

override fun minHeight(minHeight: Dp) {
value.minHeight = minHeight
}

override fun maxHeight(maxHeight: Dp) {
value.maxHeight = maxHeight
}

override fun margin(margin: Margin) {
}

override fun horizontalAlignment(horizontalAlignment: CrossAxisAlignment) {
value.horizontalAlignment = horizontalAlignment
}

override fun verticalAlignment(verticalAlignment: CrossAxisAlignment) {
value.verticalAlignment = verticalAlignment
}

internal class View() : UIView(CGRectZero.readValue()) {
var widthConstraint = Constraint.Wrap
var heightConstraint = Constraint.Wrap

var minWidth = 0.dp
var maxWidth = INT_MAX.dp

var minHeight = 0.dp
var maxHeight = INT_MAX.dp

var horizontalAlignment = CrossAxisAlignment.Start
var verticalAlignment = CrossAxisAlignment.Start

val children = UIViewChildren(
this,
insert = { view, index ->
insertSubview(view, index.convert<NSInteger>())
view.setNeedsLayout()
},
remove = { index, count ->
val views = Array(count) {
typedSubviews[index].also(UIView::removeFromSuperview)
}
setNeedsLayout()
return@UIViewChildren views
},
)

override fun layoutSubviews() {
super.layoutSubviews()

children.widgets.forEach {
val view = it.value
view.sizeToFit()
var childWidth: CGFloat = view.frame.useContents { this.size.width }
var childHeight: CGFloat = view.frame.useContents { this.size.height }

// Check for modifier overrides in the children, otherwise default to the Box's alignment values
var itemHorizontalAlignment = horizontalAlignment
var itemVerticalAlignment = verticalAlignment
it.modifier.forEach { childModifier ->
when (childModifier) {
is HorizontalAlignment -> {
itemHorizontalAlignment = childModifier.alignment
}
is VerticalAlignment -> {
itemVerticalAlignment = childModifier.alignment
}
}
}

// Compute origin and stretch if needed
var x: CGFloat = 0.0
var y: CGFloat = 0.0
when (itemHorizontalAlignment) {
CrossAxisAlignment.Stretch -> {
x = 0.0
childWidth = frame.useContents { this.size.width }
}
CrossAxisAlignment.Start -> x = 0.0
CrossAxisAlignment.Center -> x = (frame.useContents { this.size.width } - childWidth) / 2.0
CrossAxisAlignment.End -> x = frame.useContents { this.size.width } - childWidth
}
when (itemVerticalAlignment) {
CrossAxisAlignment.Stretch -> {
y = 0.0
childHeight = frame.useContents { this.size.height }
}
CrossAxisAlignment.Start -> y = 0.0
CrossAxisAlignment.Center -> y = (frame.useContents { this.size.height } - childHeight) / 2.0
CrossAxisAlignment.End -> y = frame.useContents { this.size.height } - childHeight
}

// Position the view
view.setFrame(CGRectMake(x, y, childWidth, childHeight))
}
}

override fun sizeThatFits(size: CValue<CGSize>): CValue<CGSize> {
var width: CGFloat = 0.0
var height: CGFloat = 0.0

// Calculate the size based on Constraint Values
when (widthConstraint) {
Constraint.Fill -> {
when (heightConstraint) {
Constraint.Fill -> {
width = size.useContents { this.width.coerceIn(minWidth.value, maxWidth.value) }
height = size.useContents { this.height.coerceIn(minHeight.value, maxHeight.value) }
}
Constraint.Wrap -> {
height = size.useContents { this.height.coerceIn(minWidth.value, maxWidth.value) }

// calculate the width of the biggest item
val sizeToFit = CGSizeMake(maxWidth.value, height)
width = typedSubviews
.map { it.sizeThatFits(sizeToFit).useContents { this.width } }
.max()
}
}
}
Constraint.Wrap -> {
when (heightConstraint) {
Constraint.Fill -> {
width = size.useContents { this.width.coerceIn(minWidth.value, maxWidth.value) }

// calculate the height of the biggest item
val sizeToFit = CGSizeMake(width, maxHeight.value)
height = typedSubviews
.map { it.sizeThatFits(sizeToFit).useContents { this.height } }
.max()
}
Constraint.Wrap -> {
val unconstrainedSizes = typedSubviews
.map { it.sizeThatFits(CGSizeMake(maxWidth.value, maxHeight.value)) }

width = unconstrainedSizes
.map { it.useContents { this.width } }
.max()

height = unconstrainedSizes
.map { it.useContents { this.height } }
.max()
}
}
}
}
return CGSizeMake(width, height)
}
}
}

Original file line number Diff line number Diff line change
Expand Up @@ -25,9 +25,7 @@ import platform.UIKit.UIView

@ObjCName("UIViewRedwoodLayoutWidgetFactory", exact = true)
public class UIViewRedwoodLayoutWidgetFactory : RedwoodLayoutWidgetFactory<UIView> {
override fun Box(): Box<UIView> {
TODO("Not yet implemented")
}
override fun Box(): Box<UIView> = UIViewBox()
override fun Column(): Column<UIView> = UIViewFlexContainer(FlexDirection.Column)
override fun Row(): Row<UIView> = UIViewFlexContainer(FlexDirection.Row)
override fun Spacer(): Spacer<UIView> = UIViewSpacer()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ import com.example.redwood.testing.compose.Button2
import com.example.redwood.testing.compose.Text

@Composable
fun BoxSandbox(modifier: Modifier = Modifier) {
fun BoxSandbox() {
Column(
width = Constraint.Fill,
height = Constraint.Fill,
Expand Down

0 comments on commit ca1657a

Please sign in to comment.