Skip to content
This repository has been archived by the owner on Aug 30, 2023. It is now read-only.

Commit

Permalink
Migrate to the Objective-C interchange format (#88)
Browse files Browse the repository at this point in the history
This change introduces new APIs that are compatible with the new interchange format introduced in MotionInterchange [v1.5.0](https://github.com/material-motion/motion-interchange-objc/releases/tag/v1.5.0). The API arguments have also been reordered to be somewhat more intuitive.

Note that this is not intended to be a breaking change.
  • Loading branch information
jverkoey authored Dec 13, 2017
1 parent 45d43aa commit 573b192
Show file tree
Hide file tree
Showing 26 changed files with 558 additions and 496 deletions.
2 changes: 1 addition & 1 deletion MotionAnimator.podspec
Original file line number Diff line number Diff line change
Expand Up @@ -12,5 +12,5 @@ Pod::Spec.new do |s|
s.public_header_files = "src/*.h"
s.source_files = "src/*.{h,m,mm}", "src/private/*.{h,m,mm}"

s.dependency "MotionInterchange", "~> 1.3"
s.dependency "MotionInterchange", "~> 1.6"
end
9 changes: 6 additions & 3 deletions Podfile.lock
Original file line number Diff line number Diff line change
Expand Up @@ -2,21 +2,24 @@ PODS:
- CatalogByConvention (2.2.0)
- MotionAnimator (2.6.0):
- MotionInterchange (~> 1.3)
- MotionInterchange (1.3.0)
- MotionInterchange (1.6.0)

DEPENDENCIES:
- CatalogByConvention
- MotionAnimator (from `./`)
- MotionInterchange (from `../motion-interchange-objc/`)

EXTERNAL SOURCES:
MotionAnimator:
:path: ./
MotionInterchange:
:path: ../motion-interchange-objc/

SPEC CHECKSUMS:
CatalogByConvention: 5df5831e48b8083b18570dcb804f20fd1c90694f
MotionAnimator: a4b0ba87a674bb3e89e25f0530b7e80a204ac1c1
MotionInterchange: 988fc0011e4b806cc33f2fb4f9566f5eeb4159e8
MotionInterchange: ead0e3ae1f3a5fb539e289debbc7ae036160a10d

PODFILE CHECKSUM: 3537bf01c11174928ac008c20fec4738722e96f3
PODFILE CHECKSUM: f354f45cd3f9eb0e6ac9a2bfd9429945eae8c0ad

COCOAPODS: 1.3.1
2 changes: 1 addition & 1 deletion WORKSPACE
Original file line number Diff line number Diff line change
Expand Up @@ -27,5 +27,5 @@ git_repository(
git_repository(
name = "motion_interchange_objc",
remote = "https://github.com/material-motion/motion-interchange-objc.git",
tag = "v1.3.0",
tag = "v1.6.0",
)
80 changes: 41 additions & 39 deletions examples/CalendarCardExpansionExample.m
Original file line number Diff line number Diff line change
Expand Up @@ -20,12 +20,12 @@

#import "MotionAnimator.h"

// This example demonstrates how to use a motion timing specification to build a complex
// This example demonstrates how to use a motion traits specification to build a complex
// bi-directional animation using the MDMMotionAnimator object. MDMMotionAnimator is designed for
// building fine-tuned explicit animations. Unlike UIView's implicit animation API, which can be
// used to cause cascading animations on a variety of properties, MDMMotionAnimator will always add
// exactly one animation per key path to the layer. This means you don't get as much for "free", but
// you do gain more control over the timing and motion of the animation.
// you do gain more control over the traits and motion of the animation.

@implementation CalendarCardExpansionExampleViewController {
// In a real-world scenario we'd likely create a separate view to manage all of these subviews so
Expand All @@ -40,81 +40,83 @@ @implementation CalendarCardExpansionExampleViewController {
- (void)didTap {
_expanded = !_expanded;

CalendarChipTiming timing = (_expanded
? CalendarChipMotionSpec.expansion
: CalendarChipMotionSpec.collapse);
id<CalendarChipTiming> traits = (_expanded
? CalendarChipMotionSpec.expansion
: CalendarChipMotionSpec.collapse);

MDMMotionAnimator *animator = [[MDMMotionAnimator alloc] init];
animator.shouldReverseValues = !_expanded;
animator.beginFromCurrentState = YES;

[animator animateWithTiming:timing.navigationBarY animations:^{
[animator animateWithTraits:traits.navigationBarY animations:^{
[self.navigationController setNavigationBarHidden:_expanded animated:YES];
}];

CGRect chipFrame = [self frameForChip];
CGRect headerFrame = [self frameForHeader];

// Animate the chip itself.
[animator animateWithTiming:timing.chipHeight
toLayer:_chipView.layer
withValues:@[ @(chipFrame.size.height), @(headerFrame.size.height) ]
[animator animateWithTraits:traits.chipHeight
between:@[ @(chipFrame.size.height), @(headerFrame.size.height) ]
layer:_chipView.layer
keyPath:MDMKeyPathHeight];
[animator animateWithTiming:timing.chipWidth
toLayer:_chipView.layer
withValues:@[ @(chipFrame.size.width), @(headerFrame.size.width) ]
[animator animateWithTraits:traits.chipWidth
between:@[ @(chipFrame.size.width), @(headerFrame.size.width) ]
layer:_chipView.layer
keyPath:MDMKeyPathWidth];
[animator animateWithTiming:timing.chipWidth
toLayer:_chipView.layer
withValues:@[ @(CGRectGetMidX(chipFrame)), @(CGRectGetMidX(headerFrame)) ]
[animator animateWithTraits:traits.chipWidth
between:@[ @(CGRectGetMidX(chipFrame)), @(CGRectGetMidX(headerFrame)) ]
layer:_chipView.layer
keyPath:MDMKeyPathX];
[animator animateWithTiming:timing.chipY
toLayer:_chipView.layer
withValues:@[ @(CGRectGetMidY(chipFrame)), @(CGRectGetMidY(headerFrame)) ]
[animator animateWithTraits:traits.chipY
between:@[ @(CGRectGetMidY(chipFrame)), @(CGRectGetMidY(headerFrame)) ]
layer:_chipView.layer
keyPath:MDMKeyPathY];
[animator animateWithTiming:timing.chipHeight
toLayer:_chipView.layer
withValues:@[ @([self chipCornerRadius]), @0 ]
[animator animateWithTraits:traits.chipHeight
between:@[ @([self chipCornerRadius]), @0 ]
layer:_chipView.layer
keyPath:MDMKeyPathCornerRadius];

// Cross-fade the chip's contents.
[animator animateWithTiming:timing.chipContentOpacity
toLayer:_collapsedContent.layer
withValues:@[ @1, @0 ]
[animator animateWithTraits:traits.chipContentOpacity
between:@[ @1, @0 ]
layer:_collapsedContent.layer
keyPath:MDMKeyPathOpacity];
[animator animateWithTiming:timing.headerContentOpacity
toLayer:_expandedContent.layer
withValues:@[ @0, @1 ]
[animator animateWithTraits:traits.headerContentOpacity
between:@[ @0, @1 ]
layer:_expandedContent.layer
keyPath:MDMKeyPathOpacity];

// Keeps the expandec content aligned to the bottom of the card by taking into consideration the
// extra height.
CGFloat excessTopMargin = chipFrame.size.height - headerFrame.size.height;
[animator animateWithTiming:timing.chipHeight
toLayer:_expandedContent.layer
withValues:@[ @(CGRectGetMidY([self expandedContentFrame]) + excessTopMargin),
[animator animateWithTraits:traits.chipHeight
between:@[ @(CGRectGetMidY([self expandedContentFrame]) + excessTopMargin),
@(CGRectGetMidY([self expandedContentFrame])) ]
layer:_expandedContent.layer
keyPath:MDMKeyPathY];

// Keeps the collapsed content aligned to its position on screen by taking into consideration the
// excess left margin.
CGFloat excessLeftMargin = chipFrame.origin.x - headerFrame.origin.x;
[animator animateWithTiming:timing.chipWidth
toLayer:_collapsedContent.layer
withValues:@[ @(CGRectGetMidX([self collapsedContentFrame])),
[animator animateWithTraits:traits.chipWidth
between:@[ @(CGRectGetMidX([self collapsedContentFrame])),
@(CGRectGetMidX([self collapsedContentFrame]) + excessLeftMargin) ]
layer:_collapsedContent.layer
keyPath:MDMKeyPathX];

// Keeps the shape anchored to the bottom right of the chip.
CGRect shapeFrameInChip = [self shapeFrameInRect:chipFrame];
CGRect shapeFrameInHeader = [self shapeFrameInRect:headerFrame];
[animator animateWithTiming:timing.chipWidth
toLayer:_shapeView.layer
withValues:@[ @(CGRectGetMidX(shapeFrameInChip)), @(CGRectGetMidX(shapeFrameInHeader)) ]
[animator animateWithTraits:traits.chipWidth
between:@[ @(CGRectGetMidX(shapeFrameInChip)),
@(CGRectGetMidX(shapeFrameInHeader)) ]
layer:_shapeView.layer
keyPath:MDMKeyPathX];
[animator animateWithTiming:timing.chipHeight
toLayer:_shapeView.layer
withValues:@[ @(CGRectGetMidY(shapeFrameInChip)), @(CGRectGetMidY(shapeFrameInHeader)) ]
[animator animateWithTraits:traits.chipHeight
between:@[ @(CGRectGetMidY(shapeFrameInChip)),
@(CGRectGetMidY(shapeFrameInHeader)) ]
layer:_shapeView.layer
keyPath:MDMKeyPathY];
}

Expand Down
24 changes: 13 additions & 11 deletions examples/CalendarChipMotionSpec.h
Original file line number Diff line number Diff line change
Expand Up @@ -17,24 +17,26 @@
#import <Foundation/Foundation.h>
#import <MotionInterchange/MotionInterchange.h>

typedef struct CalendarChipTiming {
MDMMotionTiming chipWidth;
MDMMotionTiming chipHeight;
MDMMotionTiming chipY;
@protocol CalendarChipTiming <NSObject>

MDMMotionTiming chipContentOpacity;
MDMMotionTiming headerContentOpacity;
@property(nonatomic, strong, nonnull, readonly) MDMAnimationTraits *chipWidth;
@property(nonatomic, strong, nonnull, readonly) MDMAnimationTraits *chipHeight;
@property(nonatomic, strong, nonnull, readonly) MDMAnimationTraits *chipY;

MDMMotionTiming navigationBarY;
} CalendarChipTiming;
@property(nonatomic, strong, nonnull, readonly) MDMAnimationTraits *chipContentOpacity;
@property(nonatomic, strong, nonnull, readonly) MDMAnimationTraits *headerContentOpacity;

@property(nonatomic, strong, nonnull, readonly) MDMAnimationTraits *navigationBarY;

@end

@interface CalendarChipMotionSpec: NSObject

@property(nonatomic, class, readonly) CalendarChipTiming expansion;
@property(nonatomic, class, readonly) CalendarChipTiming collapse;
@property(nonatomic, class, strong, nonnull, readonly) id<CalendarChipTiming> expansion;
@property(nonatomic, class, strong, nonnull, readonly) id<CalendarChipTiming> collapse;

// This object is not meant to be instantiated.
- (instancetype)init NS_UNAVAILABLE;
- (nonnull instancetype)init NS_UNAVAILABLE;

@end

126 changes: 76 additions & 50 deletions examples/CalendarChipMotionSpec.m
Original file line number Diff line number Diff line change
Expand Up @@ -16,58 +16,84 @@

#import "CalendarChipMotionSpec.h"

static id<MDMTimingCurve> StandardTimingCurve(void) {
return [CAMediaTimingFunction functionWithControlPoints:0.4f :0.0f :0.2f :1.0f];
}

static id<MDMTimingCurve> LinearTimingCurve(void) {
return [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionLinear];
}

@interface CalendarChipExpansionTiming: NSObject <CalendarChipTiming>
@end

@implementation CalendarChipExpansionTiming

- (MDMAnimationTraits *)chipWidth {
return [[MDMAnimationTraits alloc] initWithDelay:0.000 duration:0.285 timingCurve:StandardTimingCurve()];
}

- (MDMAnimationTraits *)chipHeight {
return [[MDMAnimationTraits alloc] initWithDelay:0.015 duration:0.360 timingCurve:StandardTimingCurve()];
}

- (MDMAnimationTraits *)chipY {
return [[MDMAnimationTraits alloc] initWithDelay:0.015 duration:0.360 timingCurve:StandardTimingCurve()];
}

- (MDMAnimationTraits *)chipContentOpacity {
return [[MDMAnimationTraits alloc] initWithDelay:0.000 duration:0.075 timingCurve:LinearTimingCurve()];
}

- (MDMAnimationTraits *)headerContentOpacity {
return [[MDMAnimationTraits alloc] initWithDelay:0.075 duration:0.150 timingCurve:LinearTimingCurve()];
}

- (MDMAnimationTraits *)navigationBarY {
return [[MDMAnimationTraits alloc] initWithDelay:0.015 duration:0.360 timingCurve:StandardTimingCurve()];
}

@end

@interface CalendarChipCollapseTiming: NSObject <CalendarChipTiming>
@end

@implementation CalendarChipCollapseTiming

- (MDMAnimationTraits *)chipWidth {
return [[MDMAnimationTraits alloc] initWithDelay:0.045 duration:0.330 timingCurve:StandardTimingCurve()];
}

- (MDMAnimationTraits *)chipHeight {
return [[MDMAnimationTraits alloc] initWithDelay:0.000 duration:0.330 timingCurve:StandardTimingCurve()];
}

- (MDMAnimationTraits *)chipY {
return [[MDMAnimationTraits alloc] initWithDelay:0.015 duration:0.330 timingCurve:StandardTimingCurve()];
}

- (MDMAnimationTraits *)chipContentOpacity {
return [[MDMAnimationTraits alloc] initWithDelay:0.150 duration:0.150 timingCurve:LinearTimingCurve()];
}

- (MDMAnimationTraits *)headerContentOpacity {
return [[MDMAnimationTraits alloc] initWithDelay:0.000 duration:0.075 timingCurve:LinearTimingCurve()];
}

- (MDMAnimationTraits *)navigationBarY {
return [[MDMAnimationTraits alloc] initWithDelay:0.045 duration:0.150 timingCurve:StandardTimingCurve()];
}

@end

@implementation CalendarChipMotionSpec

+ (MDMMotionCurve)eightyForty {
return MDMMotionCurveMakeBezier(0.4f, 0.0f, 0.2f, 1.0f);
}

+ (CalendarChipTiming)expansion {
MDMMotionCurve eightyForty = [self eightyForty];
return (CalendarChipTiming){
.chipWidth = {
.delay = 0.000, .duration = 0.285, .curve = eightyForty,
},
.chipHeight = {
.delay = 0.015, .duration = 0.360, .curve = eightyForty,
},
.chipY = {
.delay = 0.015, .duration = 0.360, .curve = eightyForty,
},
.chipContentOpacity = {
.delay = 0.000, .duration = 0.075, .curve = MDMLinearMotionCurve,
},
.headerContentOpacity = {
.delay = 0.075, .duration = 0.150, .curve = MDMLinearMotionCurve,
},
.navigationBarY = {
.delay = 0.015, .duration = 0.360, .curve = eightyForty,
},
};
}

+ (CalendarChipTiming)collapse {
MDMMotionCurve eightyForty = [self eightyForty];
return (CalendarChipTiming){
.chipWidth = {
.delay = 0.045, .duration = 0.330, .curve = eightyForty,
},
.chipHeight = {
.delay = 0.000, .duration = 0.330, .curve = eightyForty,
},
.chipY = {
.delay = 0.015, .duration = 0.330, .curve = eightyForty,
},
.chipContentOpacity = {
.delay = 0.150, .duration = 0.150, .curve = MDMLinearMotionCurve,
},
.headerContentOpacity = {
.delay = 0.000, .duration = 0.075, .curve = MDMLinearMotionCurve,
},
.navigationBarY = {
.delay = 0.045, .duration = 0.150, .curve = eightyForty,
}
};
+ (id<CalendarChipTiming>)expansion {
return [[CalendarChipExpansionTiming alloc] init];
}

+ (id<CalendarChipTiming>)collapse {
return [[CalendarChipCollapseTiming alloc] init];
}

@end
Expand Down
13 changes: 7 additions & 6 deletions examples/TapToBounceExample.swift
Original file line number Diff line number Diff line change
Expand Up @@ -40,21 +40,22 @@ class TapToBounceExampleViewController: UIViewController {
for: [.touchUpInside, .touchUpOutside, .touchDragExit])
}

let timing = MotionTiming(delay: 0,
duration: 0.5,
curve: MotionCurveMakeSpring(mass: 1, tension: 100, friction: 10),
repetition: .init())
let traits = MDMAnimationTraits(delay: 0,
duration: 0.5,
timingCurve: MDMSpringTimingCurve(mass: 1,
tension: 100,
friction: 10))

func didFocus(_ sender: UIButton) {
let animator = MotionAnimator()
animator.animate(with: timing) {
animator.animate(with: traits) {
sender.transform = CGAffineTransform(scaleX: 1.5, y: 1.5)
}
}

func didUnfocus(_ sender: UIButton) {
let animator = MotionAnimator()
animator.animate(with: timing) {
animator.animate(with: traits) {
sender.transform = .identity
}
}
Expand Down
Loading

0 comments on commit 573b192

Please sign in to comment.