-
Notifications
You must be signed in to change notification settings - Fork 24.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
UIBlocks don't flush at the correct time #1365
Comments
Curious to see what the solution is here! |
I had a problem where a native action i was executing (animation related) was only happening after a while. The solution I found worked was to add - (dispatch_queue_t)methodQueue {
return dispatch_get_main_queue();
} To my native module implementation |
EDIT: The The issue is that when you call
Hope that explains the behavior 😃 |
@tadeuzagallo should we make |
@tadeuzagallo Thanks I will try that out soon :) |
@ide Hum, not sure, it'd introduce a 1 frame delay, that's probably not desired in most cases. I think it'd make more sense to just allow accessing the /cc @nicklockwood |
@tadeuzagallo If I correctly understand your explanation, then perhaps my case was actually bug, since no calls to UIManager were actually being made in the native method. In fact I attempted to just call a native Alert (not through the react alert manager) and it still displayed the same behaviour, only appearing after a significant amount of time. |
@JohnyDays that is expected behavior because you were calling into UIKit off of the main thread. Almost always |
Here's our internal POP module implementation. This will probably never be released as it will be made obsolete by the official animation API that @vjeux is working on, but if you're using it anyway, you may as well use something that works: RKPOPAnimationManager.h // Copyright 2004-present Facebook. All Rights Reserved.
#import "RCTBridgeModule.h"
#import "RCTInvalidating.h"
#import <POP/POPAnimation.h>
@interface RKPOPAnimationManager : NSObject <RCTInvalidating, RCTBridgeModule>
@end RKPOPAnimationManager.m // Copyright 2004-present Facebook. All Rights Reserved.
#import "RKPOPAnimationManager.h"
#import "RCTAssert.h"
#import "RCTBridge.h"
#import "RCTConvert.h"
#import "RCTLog.h"
#import "RCTSparseArray.h"
#import "RCTUIManager.h"
#import <POP/POP.h>
typedef NS_ENUM(NSInteger, RKPOPAnimationType) {
RKPOPAnimationTypeSpring = 0,
RKPOPAnimationTypeDecay,
RKPOPAnimationTypeLinear,
RKPOPAnimationTypeEaseIn,
RKPOPAnimationTypeEaseOut,
RKPOPAnimationTypeEaseInEaseOut,
};
/**
* Properties for all animations that can be specified in JavaScript via
* RKPOPAnimationManager.createAnimation(). These generally map directly
* to POPAnimation properties.
*/
static const struct {
__unsafe_unretained NSString *property;
__unsafe_unretained NSString *velocity;
__unsafe_unretained NSString *fromValue;
__unsafe_unretained NSString *toValue;
__unsafe_unretained NSString *springBounciness;
__unsafe_unretained NSString *dynamicsTension;
__unsafe_unretained NSString *dynamicsFriction;
__unsafe_unretained NSString *dynamicsMass;
__unsafe_unretained NSString *deceleration;
__unsafe_unretained NSString *duration;
} RKPOPAnimationManagerProperties = {
// Not all of these properties apply to all animation types
.property = @"property",
.velocity = @"velocity",
.fromValue = @"fromValue",
.toValue = @"toValue",
.springBounciness = @"springBounciness",
.dynamicsTension = @"dynamicsTension",
.dynamicsFriction = @"dynamicsFriction",
.dynamicsMass = @"dynamicsMass",
.deceleration = @"deceleration",
.duration = @"duration",
};
@implementation RKPOPAnimationManager
{
RCTSparseArray *_animationRegistry;
}
@synthesize bridge = _bridge;
RCT_EXPORT_MODULE()
- (instancetype)init
{
if ((self = [super init])) {
_animationRegistry = [[RCTSparseArray alloc] init];
}
return self;
}
- (dispatch_queue_t)methodQueue
{
return _bridge.uiManager.methodQueue;
}
- (NSDictionary *)constantsToExport
{
return @{
@"Types": @{
@"spring": @(RKPOPAnimationTypeSpring),
@"decay": @(RKPOPAnimationTypeDecay),
@"linear": @(RKPOPAnimationTypeLinear),
@"easeIn": @(RKPOPAnimationTypeEaseIn),
@"easeOut": @(RKPOPAnimationTypeEaseOut),
@"easeInEaseOut": @(RKPOPAnimationTypeEaseInEaseOut)
},
@"Properties": @{
@"bounds": kPOPLayerBounds,
@"position": kPOPLayerPosition,
@"positionX": kPOPLayerPositionX,
@"positionY": kPOPLayerPositionY,
@"opacity": kPOPLayerOpacity,
@"scaleX": kPOPLayerScaleX,
@"scaleY": kPOPLayerScaleY,
@"scaleXY": kPOPLayerScaleXY,
@"subscaleXY": kPOPLayerSubscaleXY,
@"translationX": kPOPLayerTranslationX,
@"translationY": kPOPLayerTranslationY,
@"translationZ": kPOPLayerTranslationZ,
@"translationXY": kPOPLayerTranslationXY,
@"subtranslationX": kPOPLayerSubtranslationX,
@"subtranslationY": kPOPLayerSubtranslationY,
@"subtranslationZ": kPOPLayerSubtranslationZ,
@"subtranslationXY": kPOPLayerSubtranslationXY,
@"zPosition": kPOPLayerZPosition,
@"size": kPOPLayerSize,
@"rotation": kPOPLayerRotation,
@"rotationX": kPOPLayerRotationX,
@"rotationY": kPOPLayerRotationY,
@"shadowColor": kPOPLayerShadowColor,
@"shadowOffset": kPOPLayerShadowOffset,
@"shadowOpacity": kPOPLayerShadowOpacity,
@"shadowRadius": kPOPLayerShadowRadius
}
};
}
- (void)dealloc
{
RCTAssert(!self.valid, @"must call -invalidate before -dealloc");
}
- (BOOL)isValid
{
return _animationRegistry != nil;
}
- (void)invalidate
{
_animationRegistry = nil;
_bridge = nil;
}
RCT_EXPORT_METHOD(createAnimationInternal:(NSNumber *)animationTag
type:(NSInteger)type
props:(NSDictionary *)props)
{
RCTAssert(props != nil, @"props should exist");
[_bridge.uiManager addUIBlock:^(RCTUIManager *uiManager, RCTSparseArray *viewRegistry) {
NSString *propertyName = [props objectForKey:RKPOPAnimationManagerProperties.property];
if (!propertyName) {
RCTLogError(@"Animation has no property to animate: %@", props);
}
POPAnimatableProperty *property = [POPAnimatableProperty propertyWithName:propertyName];
static NSDictionary *animationsByType;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
animationsByType = @{
@(RKPOPAnimationTypeSpring): ^{return [POPSpringAnimation animation];},
@(RKPOPAnimationTypeDecay): ^{return [POPDecayAnimation animation];},
@(RKPOPAnimationTypeLinear): ^{return [POPBasicAnimation linearAnimation];},
@(RKPOPAnimationTypeEaseIn): ^{return [POPBasicAnimation easeInAnimation];},
@(RKPOPAnimationTypeEaseOut): ^{return [POPBasicAnimation easeOutAnimation];},
@(RKPOPAnimationTypeEaseInEaseOut): ^{return [POPBasicAnimation easeInEaseOutAnimation];},
};
});
POPPropertyAnimation *(^animationBlock)() = animationsByType[@(type)];
if (!animationBlock) {
RCTLogError(@"Unknown animation type: %zd", type);
animationBlock = ^{return [POPBasicAnimation easeInEaseOutAnimation];};
}
POPPropertyAnimation *animation = animationBlock();
animation.property = property;
[props enumerateKeysAndObjectsUsingBlock:^(NSString *key, id jsonValue, BOOL *stop) {
// Specially handled these keys above.
if ([key isEqualToString:RKPOPAnimationManagerProperties.property] || [key isEqualToString:@"type"]) {
return;
}
if ([RKPOPAnimationManager _propertyKeyIsPOPAnimationKey:key]) {
id animationValue;
if ([jsonValue isKindOfClass:[NSNumber class]]) {
//
// JSON number -> NSNumber
//
animationValue = jsonValue;
// Make sure it's actually an NSNumber(float) or NSNumber(double); POPAnimation fails on NSNumber(int).
animationValue = @([animationValue doubleValue]);
} else if ([jsonValue isKindOfClass:[NSArray class]]) {
NSArray *jsonArray = (NSArray *)jsonValue;
if (jsonArray.count == 2) {
//
// JSON array -> NSValue(CGPoint)
//
CGPoint point = [RCTConvert CGPoint:jsonArray];
animationValue = [NSValue valueWithCGPoint:point];
} else if (jsonArray.count == 4) {
//
// JSON array -> NSValue(CGRect)
//
CGRect rect = [RCTConvert CGRect:jsonArray];
animationValue = [NSValue valueWithCGRect:rect];
} else {
RCTLogError(@"Only two-dimensional (point) values supported for %@ of property: %@", key, propertyName);
return;
}
}
// Various POPAnimation methods, such as -setToValue:, throw if the value is of a type not supported for the animated property (for example, if the animated property is 'scaleXY', the toValue should be a CGPoint, never an NSNumber or some other type). Catch and log errors instead of crashing.
@try {
[animation setValue:animationValue forKey:key];
} @catch (NSException *exception) {
RCTLogError(@"Error setting animation property \"%@\" to %@: %@", key, animationValue, exception);
}
} else {
RCTLogError(@"Unknown animation property: %@", key);
}
}];
_animationRegistry[animationTag] = animation;
}];
}
RCT_EXPORT_METHOD(addAnimation:(NSNumber *)viewTag
withTag:(NSNumber *)animationTag
completion:(RCTResponseSenderBlock)callback)
{
if (!animationTag || !viewTag) {
RCTLogError(@"addAnimation requires both animationTag and viewTag, received (#%@, #%@)", animationTag, viewTag);
return;
}
[_bridge.uiManager addUIBlock:^(RCTUIManager *uiManager, RCTSparseArray *viewRegistry) {
UIView *view = viewRegistry[viewTag];
POPAnimation *animation = _animationRegistry[animationTag];
if (!view) {
RCTLogError(@"addAnimation cannot find view with tag #%@, attempting to add animation:\n%@", viewTag, animation);
return;
}
if (!animation) {
RCTLogError(@"addAnimation cannot find animation with tag #%@, attempting to animate view %@ with tag #%@", animationTag, view, viewTag);
}
if (callback) {
animation.completionBlock = ^(POPAnimation *anim, BOOL finished) {
callback(@[@(finished)]);
};
}
CALayer *layer = view.layer;
[layer pop_addAnimation:animation forKey:[animationTag stringValue]];
}];
}
RCT_EXPORT_METHOD(removeAnimation:(NSNumber *)viewTag
withTag:(NSNumber *)animationTag)
{
if (!animationTag || !viewTag) {
RCTLogError(@"removeAnimation requires both animationTag and viewTag, received (%zd, %zd)", animationTag, viewTag);
return;
}
[_bridge.uiManager addUIBlock:^(RCTUIManager *uiManager, RCTSparseArray *viewRegistry) {
UIView *view = viewRegistry[viewTag];
// Simply getting the animation to verify that it exists.
POPAnimation *animation = _animationRegistry[animationTag];
if (!view) {
RCTLogError(@"removeAnimation cannot find view with tag #%@, attempting to remove animation:\n%@", viewTag, animation);
return;
}
if (!animation) {
RCTLogError(@"removeAnimation cannot find animation with tag #%@, attempting to remove from view %@ with tag #%@", animationTag, view, viewTag);
}
CALayer *layer = view.layer;
[layer pop_removeAnimationForKey:[animationTag stringValue]];
}];
}
+ (BOOL)_propertyKeyIsPOPAnimationKey:(NSString *)propertyKey
{
// These are RKAnimationProperties which correspond exactly to CGFloat or NSValue(CGPoint) properties of POPAnimation. Specifying one of these in the animation properties means that it'll be set directly on the underlying POPAnimation. This is for convenience so that -createAndRegisterAnimationWithTag:type:props: can parse various similar properties in the same way.
static NSSet *POPAnimationPropertyKeys = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
__unsafe_unretained NSString **objects = (__unsafe_unretained NSString **)(void *)&RKPOPAnimationManagerProperties;
POPAnimationPropertyKeys = [NSSet setWithObjects:objects count:sizeof(RKPOPAnimationManagerProperties) / sizeof(*objects)];
});
return [POPAnimationPropertyKeys containsObject:propertyKey];
}
@end |
Awesome thx :) I just reimplemented the API and it worked for me but this is more complete than mine. |
Thanks @ide, I did not know method calls to UIKit were limited to the main thread, very new to Obj-C and the Apple APIs(didn't actually know what UIKit was, had to check). |
@tadeuzagallo So I added this to my module. - (dispatch_queue_t)methodQueue {
return dispatch_get_main_queue();
} The end result is the same. The weirder part is some of my animations do multiple invokes to addAnimation some of them play instantly and others wait till next interaction or just after sometime. An example is fading in a backdrop and transforming a modal upwards. |
Actually using this from the FB implementation. - (dispatch_queue_t)methodQueue {
return _bridge.uiManager.methodQueue;
} Solved the problem |
Yes, that makes sense, sorry about that. I read the answer and forgot to check the actual queue, but the chain I explained remains the same. I'll update the answer accordingly for future references. :) |
@tadeuzagallo - what I was thinking is that it is confusing that the UI blocks do not run until some other unrelated piece of code or a UI interaction causes the UIManager to flush the blocks. To make this less surprising, here are two ideas: one idea is to schedule the the UI blocks to run ASAP (probably after the next vsync iteration as you suggested). The other idea is to print a warning message if |
Asserting that [UIManager addBlock:] is called on the shadowQueue seems like a good solution. |
…:] to _shadowQueue Summary: @public Add `RCTAssertThread` to `RCTAssert.h` for convenience when checking the current/queue, it accepts either a `NSString *`, `NSThread *` or `dispatch_queue_t` as the object to be checked Also add a check to `-[RCTUIManager addUIBlock:]` - There was a discussion on github (facebook#1365) due to the weird behavior caused by calling it from a different thread/queue (it might be added after `batchDidComplete` has been called and will just be dispatched on the next call from JS to objc) Test Plan: Change `-[RCTAnimationExperimentalManager methodQueue]` to return `dispatch_get_main_queue()` and run the 2048 example, it should dispatch with a helpful message (screenshot on the comments)
…:] to _shadowQueue Summary: @public Add `RCTAssertThread` to `RCTAssert.h` for convenience when checking the current/queue, it accepts either a `NSString *`, `NSThread *` or `dispatch_queue_t` as the object to be checked Also add a check to `-[RCTUIManager addUIBlock:]` - There was a discussion on github (facebook#1365) due to the weird behavior caused by calling it from a different thread/queue (it might be added after `batchDidComplete` has been called and will just be dispatched on the next call from JS to objc) Test Plan: Change `-[RCTAnimationExperimentalManager methodQueue]` to return `dispatch_get_main_queue()` and run the 2048 example, it should dispatch with a helpful message (screenshot on the comments)
@nicklockwood when do you guys think @vjeux 's animation library will land? |
@mcz: hopefully a month from now |
@vjeux React Europe deadline? 🎯 |
On a side note - having something with a POP similar API would be very beneficial to those who are already familiar with your animation library when writing native apps in the old way! |
@tadeuzagallo Still, sometimes the UIBlock won't execute until the next user interaction after this function was called. I met this situation in iPhone6s, but iPhone6, iPhone6sp are fine. |
I upgraded from a much lower version (I had a personal fork) to 0.4.4 and the POPAnimation native module I made is now having issues.
This is all I am doing to add animations to a view, but now sometimes the UIBlock won't execute until the next user interaction after this function was called or even more oddly if I just leave the simulator running within a minute or so it executes.
@vjeux told me to tag @tadeuzagallo on this 😄
Any ideas?
The text was updated successfully, but these errors were encountered: