How to rotate an object around a arbitrary point?

后端 未结 4 1236
醉酒成梦
醉酒成梦 2020-12-05 08:32

I want to rotate an UILabel around an arbitrary point in a circular manner, not a straight line. This is my code.The final point is perfect but it goes through a straight li

相关标签:
4条回答
  • 2020-12-05 09:13

    You should set your own anchorPoint

    Its very much overkill to use a keyframe animation for what really is a change of the anchor point.

    The anchor point is the point where all transforms are applied from, the default anchor point is the center. By moving the anchor point to (0,0) you can instead make the layer rotate from the bottom most corner. By setting the anchor point to something where x or y is outside the range 0.0 - 1.0 you can have the layer rotate around a point that lies outside of its bounds.

    Please read the section about Layer Geometry and Transforms in the Core Animation Programming Guide for more information. It goes through this in detail with images to help you understand.


    EDIT: One thing to remember

    The frame of your layer (which is also the frame of your view) is calculated using the position, bounds and anchor point. Changing the anchorPoint will change where your view appears on screen. You can counter this by re-setting the frame after changing the anchor point (this will set the position for you). Otherwise you can set the position to the point you are rotating to yourself. The documentation (linked to above) also mentions this.


    Applied to you code

    The point you called "transportPoint" should be updated to calculate the difference between the rotation point and the lower left corner of the label divided by the width and height.

    // Pseudocode for the correct anchor point
    transportPoint = ( (rotationX - labelMinX)/labelWidth,
                       (rotationX - labelMinY)/labelHeight )
    

    I also made the rotation point an argument to your method. The full updated code is below:

    #define DEGREES_TO_RADIANS(angle) (angle/180.0*M_PI)
    
    - (void)rotateText:(UILabel *)label 
           aroundPoint:(CGPoint)rotationPoint 
              duration:(NSTimeInterval)duration 
               degrees:(CGFloat)degrees {
    
        /* Setup the animation */
        [UILabel beginAnimations:nil context:NULL];
        [UILabel setAnimationDuration:duration];
    
        // The anchor point is expressed in the unit coordinate
        // system ((0,0) to (1,1)) of the label. Therefore the 
        // x and y difference must be divided by the width and 
        // height of the label (divide x difference by width and 
        // y difference by height).
        CGPoint transportPoint = CGPointMake((rotationPoint.x - CGRectGetMinX(label.frame))/CGRectGetWidth(label.bounds),
                                             (rotationPoint.y - CGRectGetMinY(label.frame))/CGRectGetHeight(label.bounds));
    
        [label.layer setAnchorPoint:transportPoint];
        [label.layer setPosition:rotationPoint]; // change the position here to keep the frame 
        [label.layer setTransform:CATransform3DMakeRotation(DEGREES_TO_RADIANS(degrees), 0, 0, 1)];   
    
        /* Commit the changes */
        [UILabel commitAnimations];
    }
    
    0 讨论(0)
  • 2020-12-05 09:13

    translate, rotate around center, translate back.

    0 讨论(0)
  • 2020-12-05 09:14

    I decided to post my solution as an answer. It works fine accept it doesn't have the old solutions's curve animations (UIViewAnimationCurveEaseOut), but I can sort that out.

    #define DEGREES_TO_RADIANS(angle) (angle / 180.0 * M_PI)
    
    - (void)rotateText:(UILabel *)label duration:(NSTimeInterval)duration degrees:(CGFloat)degrees {
        CGMutablePathRef path = CGPathCreateMutable();
        CGPathAddArc(path,nil, 160, 236, 100, DEGREES_TO_RADIANS(0), DEGREES_TO_RADIANS(degrees), YES);
    
        CAKeyframeAnimation *theAnimation;
    
        // animation object for the key path
        theAnimation = [CAKeyframeAnimation animationWithKeyPath:@"position"];
        theAnimation.path=path;
        CGPathRelease(path);
    
        // set the animation properties
        theAnimation.duration=duration;
        theAnimation.removedOnCompletion = NO; 
        theAnimation.autoreverses = NO;
        theAnimation.rotationMode = kCAAnimationRotateAutoReverse;
        theAnimation.fillMode = kCAFillModeForwards;
    
        [label.layer addAnimation:theAnimation forKey:@"position"]; 
    }
    
    0 讨论(0)
  • 2020-12-05 09:14

    CAKeyframeAnimation is the right tool for this job. Most UIKit animations are between start and end points. The middle points are not considered. CAKeyframeAnimation allows you to define those middle points to provide a non-linear animation. You will have to provide the appropriate bezier path for your animation. You should look at this example and the one's provided in the Apple documentation to see how it works.

    0 讨论(0)
提交回复
热议问题