From 8c93519191fc26cc456e5481bf04090e37fd2f88 Mon Sep 17 00:00:00 2001 From: DreamPiggy Date: Thu, 25 Apr 2019 13:23:34 +0800 Subject: [PATCH 1/2] Fix the bug of SDAnimatedImageView, which can not render the normal NSImage (including animated image) at all. Only `SDAnimatedImage` can be rendered --- Examples/SDWebImage OSX Demo/ViewController.m | 9 ++++++--- SDWebImage/SDAnimatedImageView.m | 11 +++++++++++ 2 files changed, 17 insertions(+), 3 deletions(-) diff --git a/Examples/SDWebImage OSX Demo/ViewController.m b/Examples/SDWebImage OSX Demo/ViewController.m index 20477147..270f4cdf 100644 --- a/Examples/SDWebImage OSX Demo/ViewController.m +++ b/Examples/SDWebImage OSX Demo/ViewController.m @@ -29,13 +29,16 @@ // For animated GIF rendering, set `animates` to YES or will only show the first frame self.imageView2.animates = YES; // `SDAnimatedImageRep` can be used for built-in `NSImageView` to support better GIF & APNG rendering as well. No need `SDAnimatedImageView` - self.imageView3.animates = YES; self.imageView4.animates = YES; + + // NSImageView + Static Image self.imageView1.sd_imageIndicator = SDWebImageProgressIndicator.defaultIndicator; [self.imageView1 sd_setImageWithURL:[NSURL URLWithString:@"https://raw.githubusercontent.com/recurser/exif-orientation-examples/master/Landscape_2.jpg"] placeholderImage:nil options:SDWebImageProgressiveLoad]; + // NSImageView + Animated Image [self.imageView2 sd_setImageWithURL:[NSURL URLWithString:@"https:raw.githubusercontent.com/onevcat/APNGKit/master/TestImages/APNG-cube.apng"]]; - [self.imageView3 sd_setImageWithURL:[NSURL URLWithString:@"https://raw.githubusercontent.com/liyong03/YLGIFImage/master/YLGIFImageDemo/YLGIFImageDemo/joy.gif"]]; - self.imageView4.wantsLayer = YES; + // SDAnimatedImageView + Static Image + [self.imageView3 sd_setImageWithURL:[NSURL URLWithString:@"https://nr-platform.s3.amazonaws.com/uploads/platform/published_extension/branding_icon/275/AmazonS3.png"]]; + // SDAnimatedImageView + Animated Image self.imageView4.sd_imageTransition = SDWebImageTransition.fadeTransition; [self.imageView4 sd_setImageWithURL:[NSURL URLWithString:@"http://littlesvr.ca/apng/images/SteamEngine.webp"] placeholderImage:nil options:SDWebImageForceTransition]; diff --git a/SDWebImage/SDAnimatedImageView.m b/SDWebImage/SDAnimatedImageView.m index e898362b..5c863e74 100644 --- a/SDWebImage/SDAnimatedImageView.m +++ b/SDWebImage/SDAnimatedImageView.m @@ -678,6 +678,17 @@ static NSUInteger SDDeviceFreeMemory() { [super updateLayer]; } } + +- (BOOL)wantsUpdateLayer { + // AppKit is different from UIKit, it need extra check before the layer is updated + // When we use the custom animation, the layer.setNeedsDisplay is directly called from display link (See `displayDidRefresh:`). However, for normal image rendering, we must implements and return YES to mark it need display + if (_currentFrame) { + return NO; + } else { + return YES; + } +} + #endif From 9cce513a09d5ee4e5f655ae67d085736f7acbc79 Mon Sep 17 00:00:00 2001 From: DreamPiggy Date: Thu, 25 Apr 2019 14:13:56 +0800 Subject: [PATCH 2/2] Add a simple test case to ensure SDAnimatedImageView should display the static NSImage --- Tests/Tests/SDAnimatedImageTest.m | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/Tests/Tests/SDAnimatedImageTest.m b/Tests/Tests/SDAnimatedImageTest.m index 0135b4e4..dc8e91b6 100644 --- a/Tests/Tests/SDAnimatedImageTest.m +++ b/Tests/Tests/SDAnimatedImageTest.m @@ -137,6 +137,18 @@ static const NSUInteger kTestGIFFrameCount = 5; // local TestImage.gif loop coun expect(imageView.intrinsicContentSize).equal(image.size); } +- (void)test12AnimatedImageViewLayerContents { + // Test that SDAnimatedImageView with built-in UIImage/NSImage will actually setup the layer for display + SDAnimatedImageView *imageView = [SDAnimatedImageView new]; + UIImage *image = [[UIImage alloc] initWithData:[self testJPEGData]]; + imageView.image = image; +#if SD_MAC + expect(imageView.wantsUpdateLayer).beTruthy(); +#else + expect(imageView.layer.contents).notTo.beNil(); +#endif +} + - (void)test20AnimatedImageViewRendering { XCTestExpectation *expectation = [self expectationWithDescription:@"test SDAnimatedImageView rendering"]; SDAnimatedImageView *imageView = [[SDAnimatedImageView alloc] init];