Merge pull request #2931 from dreampiggy/fix_progressive_animation_animatedView_bug

Fix the SDAnimatedImageView's progressive animation bug, which reset the frame index to 0 each time new frames available
This commit is contained in:
DreamPiggy 2020-01-18 18:54:25 +08:00 committed by GitHub
commit 7b0941407e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 74 additions and 2 deletions

View File

@ -166,12 +166,13 @@
}; };
self.player.animationLoopHandler = ^(NSUInteger loopCount) { self.player.animationLoopHandler = ^(NSUInteger loopCount) {
@strongify(self); @strongify(self);
self.currentLoopCount = loopCount;
// Progressive image reach the current last frame index. Keep the state and pause animating. Wait for later restart // Progressive image reach the current last frame index. Keep the state and pause animating. Wait for later restart
if (self.isProgressive) { if (self.isProgressive) {
NSUInteger lastFrameIndex = self.player.totalFrameCount; NSUInteger lastFrameIndex = self.player.totalFrameCount - 1;
[self.player seekToFrameAtIndex:lastFrameIndex loopCount:0]; [self.player seekToFrameAtIndex:lastFrameIndex loopCount:0];
[self.player pausePlaying]; [self.player pausePlaying];
} else {
self.currentLoopCount = loopCount;
} }
}; };

View File

@ -8,6 +8,7 @@
*/ */
#import "SDTestCase.h" #import "SDTestCase.h"
#import "SDInternalMacros.h"
#import <KVOController/KVOController.h> #import <KVOController/KVOController.h>
static const NSUInteger kTestGIFFrameCount = 5; // local TestImage.gif loop count static const NSUInteger kTestGIFFrameCount = 5; // local TestImage.gif loop count
@ -397,6 +398,76 @@ static const NSUInteger kTestGIFFrameCount = 5; // local TestImage.gif loop coun
[self waitForExpectationsWithCommonTimeout]; [self waitForExpectationsWithCommonTimeout];
} }
- (void)test27AnimatedImageProgressiveAnimation {
XCTestExpectation *expectation = [self expectationWithDescription:@"test SDAnimatedImageView progressive animation rendering"];
// Simulate progressive download
NSData *fullData = [self testAPNGPData];
NSUInteger length = fullData.length;
SDAnimatedImageView *imageView = [SDAnimatedImageView new];
#if SD_UIKIT
[self.window addSubview:imageView];
#else
[self.window.contentView addSubview:imageView];
#endif
__block NSUInteger previousFrameIndex = 0;
@weakify(imageView);
// Observe to check rendering behavior using frame index
[self.KVOController observe:imageView keyPath:NSStringFromSelector(@selector(currentFrameIndex)) options:NSKeyValueObservingOptionNew block:^(id _Nullable observer, id _Nonnull object, NSDictionary<NSString *,id> * _Nonnull change) {
@strongify(imageView);
NSUInteger currentFrameIndex = [change[NSKeyValueChangeNewKey] unsignedIntegerValue];
printf("Animation Frame Index: %lu\n", (unsigned long)currentFrameIndex);
// The last time should not be progressive
if (currentFrameIndex == 0 && !imageView.isProgressive) {
[self.KVOController unobserve:imageView];
[expectation fulfill];
} else {
// Each progressive rendering should render new frame index, no backward and should stop at last frame index
expect(currentFrameIndex - previousFrameIndex).beGreaterThanOrEqualTo(0);
previousFrameIndex = currentFrameIndex;
}
}];
SDImageAPNGCoder *coder = [[SDImageAPNGCoder alloc] initIncrementalWithOptions:nil];
// Setup Data
NSData *setupData = [fullData subdataWithRange:NSMakeRange(0, length / 3.0)];
[coder updateIncrementalData:setupData finished:NO];
imageView.shouldIncrementalLoad = YES;
__block SDAnimatedImage *progressiveImage = [[SDAnimatedImage alloc] initWithAnimatedCoder:coder scale:1];
progressiveImage.sd_isIncremental = YES;
imageView.image = progressiveImage;
expect(imageView.isProgressive).beTruthy();
__block NSUInteger partialFrameCount;
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// Partial Data
NSData *partialData = [fullData subdataWithRange:NSMakeRange(0, length * 2.0 / 3.0)];
[coder updateIncrementalData:partialData finished:NO];
partialFrameCount = [coder animatedImageFrameCount];
expect(partialFrameCount).beGreaterThan(1);
progressiveImage = [[SDAnimatedImage alloc] initWithAnimatedCoder:coder scale:1];
progressiveImage.sd_isIncremental = YES;
imageView.image = progressiveImage;
expect(imageView.isProgressive).beTruthy();
});
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(4 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// Full Data
[coder updateIncrementalData:fullData finished:YES];
progressiveImage = [[SDAnimatedImage alloc] initWithAnimatedCoder:coder scale:1];
progressiveImage.sd_isIncremental = NO;
imageView.image = progressiveImage;
NSUInteger fullFrameCount = [coder animatedImageFrameCount];
expect(fullFrameCount).beGreaterThan(partialFrameCount);
expect(imageView.isProgressive).beFalsy();
});
[self waitForExpectationsWithCommonTimeout];
}
#pragma mark - Helper #pragma mark - Helper
- (UIWindow *)window { - (UIWindow *)window {
if (!_window) { if (!_window) {