It is more a math question than a Swift question. Regarding the math part you need to check if the result of your method that calculates the rotation is negative and return the absolute value, otherwise return 360 minus the angle:
func angle(from location: CGPoint) -> CGFloat {
let deltaY = location.y - view.center.y
let deltaX = location.x - view.center.x
let angle = atan2(deltaY, deltaX) * 180 / .pi
return angle < 0 ? abs(angle) : 360 - angle
}
Regarding the animation part I suggest using CABasicAnimation, setting isRemovedOnCompletion to false, fillMode to kCAFillModeForwards and timingFunction to linear. Other important setting is the from and to which you should use the same value for both of them with a duration of 0:
fileprivate let rotateAnimation = CABasicAnimation()
func rotate(to: CGFloat, duration: Double = 0) {
rotateAnimation.fromValue = to
rotateAnimation.toValue = to
rotateAnimation.duration = duration
rotateAnimation.repeatCount = 0
rotateAnimation.isRemovedOnCompletion = false
rotateAnimation.fillMode = kCAFillModeForwards
rotateAnimation.timingFunction = CAMediaTimingFunction(name: kCAMediaTimingFunctionLinear)
imageView.layer.add(rotateAnimation, forKey: "transform.rotation.z")
}
To convert from degrees to radians you can use the extension from this answer:
extension FloatingPoint {
var degreesToRadians: Self { return self * .pi / 180 }
var radiansToDegrees: Self { return self * 180 / .pi }
}
To preserve the rotation between launches you can add a computed property to UserDefault:
extension UserDefaults {
/// rotation persistant computed property
var rotation: CGFloat {
get {
return CGFloat(double(forKey: "rotation"))
}
set {
set(Double(newValue), forKey: "rotation")
}
}
}
Regarding the gesture recognizer you need to switch the gesture state to detect the begin, change and end of it and act accordingly:
fileprivate var rotation: CGFloat = UserDefaults.standard.rotation
fileprivate var startRotationAngle: CGFloat = 0
@objc func pan(_ gesture: UIPanGestureRecognizer) {
let location = gesture.location(in: view)
let gestureRotation = CGFloat(angle(from: location)) - startRotationAngle
switch gesture.state {
case .began:
// set the start angle of rotation
startRotationAngle = angle(from: location)
case .changed:
rotate(to: rotation - gestureRotation.degreesToRadians)
case .ended:
// update the amount of rotation
rotation -= gestureRotation.degreesToRadians
default :
break
}
// save the final position of the rotation to defaults
UserDefaults.standard.rotation = rotation
}
And add the gesture recognizer to your view:
class ViewController: UIViewController, UIGestureRecognizerDelegate {
@IBOutlet weak var imageView: UIImageView!
override func viewDidLoad() {
super.viewDidLoad()
let address = "https://i.stack.imgur.com/xnZXF.jpg"
let url = URL(string: address)!
rotate(to: rotation)
URLSession.shared.dataTask(with: url) { data, response, error in
guard let data = data else { return }
DispatchQueue.main.async {
self.imageView.image = UIImage(data: data)
let pan = UIPanGestureRecognizer(target: self, action:#selector(self.pan))
pan.minimumNumberOfTouches = 1
pan.maximumNumberOfTouches = 1
pan.delegate = self
self.view.addGestureRecognizer(pan)
}
}.resume()
}
// rest of the view controller code
}
Sample project