I have a square containerView with a roundImageView inside of it. The containerView is added to the UIDynamicAnimator. When the corners of the containerViews collide off of each other I need them to bounce off of the roundImageView, same as this question. Inside the the customContainerView I override collisionBoundsType ... return .ellipse
but the collision is still occurs from the square and not the circle, and the views are overlapping each other.
customView:
class CustomContainerView: UIView {
override public var collisionBoundsType: UIDynamicItemCollisionBoundsType {
return .ellipse
}
}
code:
var arr = [CustomContainerView]()
var animator: UIDynamicAnimator!
var gravity: UIGravityBehavior!
var collider: UICollisionBehavior!
var bouncingBehavior : UIDynamicItemBehavior!
override func viewDidLoad() {
super.viewDidLoad()
addSubViews()
addAnimatorAndBehaviors()
}
func addAnimatorAndBehaviors() {
animator = UIDynamicAnimator(referenceView: self.view)
gravity = UIGravityBehavior(items: arr)
animator.addBehavior(gravity)
collider = UICollisionBehavior(items: arr)
collider.translatesReferenceBoundsIntoBoundary = true
animator.addBehavior(collider)
bouncingBehavior = UIDynamicItemBehavior(items: arr)
bouncingBehavior.elasticity = 0.05
animator.addBehavior(bouncingBehavior)
}
func addSubViews() {
let redView = createContainerView(with: .red)
let blueView = createContainerView(with: .blue)
let yellowView = createContainerView(with: .yellow)
let purpleView = createContainerView(with: .purple)
let greenView = createContainerView(with: .green)
view.addSubview(redView)
view.addSubview(blueView)
view.addSubview(yellowView)
view.addSubview(purpleView)
view.addSubview(greenView)
arr = [redView, blueView, yellowView, purpleView, greenView]
}
func createContainerView(with color: UIColor) -> UIView {
let containerView = CustomContainerView()
containerView.backgroundColor = .brown
let size = CGSize(width: 50, height: 50)
containerView.frame.size = size
containerView.center = view.center
let roundImageView = UIImageView()
roundImageView.translatesAutoresizingMaskIntoConstraints = false
roundImageView.backgroundColor = color
containerView.addSubview(roundImageView)
roundImageView.topAnchor.constraint(equalTo: containerView.topAnchor, constant: 10).isActive = true
roundImageView.leadingAnchor.constraint(equalTo: containerView.leadingAnchor, constant: 10).isActive = true
roundImageView.trailingAnchor.constraint(equalTo: containerView.trailingAnchor, constant: -10).isActive = true
roundImageView.bottomAnchor.constraint(equalTo: containerView.bottomAnchor, constant: -10).isActive = true
roundImageView.layer.masksToBounds = true
roundImageView.layoutIfNeeded()
roundImageView.layer.cornerRadius = roundImageView.frame.height / 2
roundImageView.layer.borderWidth = 1
roundImageView.layer.borderColor = UIColor.white.cgColor
return containerView
}
Looks like collision behavior doesn't like
.ellipse
type when the views are positioned exactly on top of each other.Running your code a few times gives different results (as expected)... sometimes, all 5 views end up in a full vertical stack, other times it ends up with some overlap, and other times (after waiting a few seconds) the views settle with a couple visible and the others way below the bottom of the view - I've seen their y-positions get to > 40,000.
I made a few modifications to your code to see what's happening...
I added more views and gave each one a shape layer showing the ellipse bounds.
Then, instead of starting with them all at identical positions, I created a couple "rows" so it looks like this:
Then, on each tap, I reset the original positions and toggle the
UIDynamicItemCollisionBoundsType
between ellipse and rectangle, and then calladdAnimatorAndBehaviors()
again.Here's how it looks on sample
.ellipse
run:and on sample
.rectangle
run:As we can see, the
.ellipse
bounds are being used.Here's the code I used to play with this:
Edit
Change the
while
loop inpositionViews()
to this... tap to reset and run the animation a number of times and see what happens when all the views start with the same frame:Then, use this while loop, where we start the views at the same x-position, but increment the y-position for each view (just by
0.1
points):Another Edit
Worth noting, the fact that the ellipse collision bounds is round (
1:1
ratio), also affects things.If we change the size of the view frames just slightly, we get very different results.
Try it with:
and start them all with the exact same center point:
and you'll see the views spread out immediately.
One more Edit - to help visualize the differences
Here's another example - this time, I've numbered the views and set a "every 1/10th second" timer to update a label with the current center of each view.
Also added segmented controls to select
collisionBoundsType
and overlaying the views exactly on top of each other or offsetting them slightly:Worth noting: when the
collisionBoundsType == .ellipse
and the views start exactly on top of each other, the collision algorithm can (and usually does) end up pushing a couple views off the bottom, which puts them outside the reference system’s bounds. At that point, the algorithm continues trying to collide those views, pushing them further and further down o the Y axis.Here is the output after letting it run for a few seconds:
Views 5, 7 and 8 are way out of bounds, and the animator is still running. Those views will continue to be pushed further and further down, presumably until we get an invalid point crash (I haven't let it run long enough to find out).
Also, because the animator ends up doing so much processing on those out-of-bounds views, the collision detection on the remaining views suffers.