From 87a3e276fc1f1a50ecc6da9edc28dd4f77ac8482 Mon Sep 17 00:00:00 2001 From: qugemingzizhenmafan <45257656+qugemingzizhenmafan@users.noreply.github.com> Date: Fri, 14 Jan 2022 09:50:01 +0800 Subject: [PATCH] fix(module:image): unsubscribe old src (#7102) Co-authored-by: jiangchunlin --- components/image/image.directive.ts | 11 ++-- components/image/image.spec.ts | 78 ++++++++++++++++++++++++++++- 2 files changed, 84 insertions(+), 5 deletions(-) diff --git a/components/image/image.directive.ts b/components/image/image.directive.ts index 9035552605..9700e7f783 100644 --- a/components/image/image.directive.ts +++ b/components/image/image.directive.ts @@ -51,7 +51,8 @@ export class NzImageDirective implements OnInit, OnChanges, OnDestroy { dir?: Direction; backLoadImage!: HTMLImageElement; - private status: ImageStatusType = 'normal'; + status: ImageStatusType = 'normal'; + private backLoadDestroy$: Subject = new Subject(); private destroy$: Subject = new Subject(); get previewable(): boolean { @@ -129,6 +130,10 @@ export class NzImageDirective implements OnInit, OnChanges, OnDestroy { this.backLoadImage.srcset = this.nzSrcset; this.status = 'loading'; + // unsubscribe last backLoad + this.backLoadDestroy$.next(); + this.backLoadDestroy$.complete(); + this.backLoadDestroy$ = new Subject(); if (this.backLoadImage.complete) { this.status = 'normal'; this.getElement().nativeElement.src = this.nzSrc; @@ -145,7 +150,7 @@ export class NzImageDirective implements OnInit, OnChanges, OnDestroy { // The `nz-image` directive can be destroyed before the `load` or `error` event is dispatched, // so there's no sense to keep capturing `this`. fromEvent(this.backLoadImage, 'load') - .pipe(takeUntil(this.destroy$)) + .pipe(takeUntil(this.backLoadDestroy$), takeUntil(this.destroy$)) .subscribe(() => { this.status = 'normal'; this.getElement().nativeElement.src = this.nzSrc; @@ -153,7 +158,7 @@ export class NzImageDirective implements OnInit, OnChanges, OnDestroy { }); fromEvent(this.backLoadImage, 'error') - .pipe(takeUntil(this.destroy$)) + .pipe(takeUntil(this.backLoadDestroy$), takeUntil(this.destroy$)) .subscribe(() => { this.status = 'error'; if (this.nzFallback) { diff --git a/components/image/image.spec.ts b/components/image/image.spec.ts index 538e259bfa..c446ce6bd1 100644 --- a/components/image/image.spec.ts +++ b/components/image/image.spec.ts @@ -39,7 +39,7 @@ import { NzImageService } from 'ng-zorro-antd/image'; -const SRC = 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png`'; +const SRC = 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png'; const QUICK_SRC = ''; const PLACEHOLDER = @@ -47,6 +47,66 @@ const PLACEHOLDER = const FALLBACK = ''; +describe('Basics', () => { + let fixture: ComponentFixture; + let context: TestImageBasicsComponent; + let debugElement: DebugElement; + + beforeEach(fakeAsync(() => { + TestBed.configureTestingModule({ + imports: [NzImageModule, TestImageModule, NoopAnimationsModule], + providers: [{ provide: Overlay, useClass: Overlay }] + }); + TestBed.compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(TestImageBasicsComponent); + fixture.detectChanges(); + context = fixture.componentInstance; + debugElement = fixture.debugElement; + }); + + it('should only latest src work', fakeAsync(() => { + const ERROR_SRC = 'error.png'; + context.src = ERROR_SRC; + context.placeholder = null; + const image = debugElement.nativeElement.querySelector('img'); + fixture.detectChanges(); + const oldBackLoadImage = context.nzImage.backLoadImage; + context.src = SRC; + fixture.detectChanges(); + tick(1000); + context.nzImage.backLoadImage.dispatchEvent(new Event('load')); + fixture.detectChanges(); + expect(image.src).toBe(SRC); + tick(1000); + oldBackLoadImage.dispatchEvent(new ErrorEvent('error')); + fixture.detectChanges(); + expect(image.src).toBe(SRC); + expect(context.nzImage.status).toBe('normal'); + })); + + it('should keep placeholder when latest src is loading', fakeAsync(() => { + context.src = SRC; + context.placeholder = PLACEHOLDER; + const image = debugElement.nativeElement.querySelector('img'); + fixture.detectChanges(); + const oldBackLoadImage = context.nzImage.backLoadImage; + const SECOND_SRC = 'https://test.com/SECOND_SRC.png'; + context.src = SECOND_SRC; + fixture.detectChanges(); + tick(1000); + oldBackLoadImage.dispatchEvent(new Event('load')); + fixture.detectChanges(); + expect(image.src).toBe(PLACEHOLDER); + tick(1000); + context.nzImage.backLoadImage.dispatchEvent(new Event('load')); + fixture.detectChanges(); + expect(image.src).toBe(SECOND_SRC); + })); +}); + describe('Placeholder', () => { let fixture: ComponentFixture; let context: TestImagePlaceholderComponent; @@ -466,6 +526,15 @@ describe('Preview', () => { }); }); +@Component({ + template: ` ` +}) +export class TestImageBasicsComponent { + @ViewChild(NzImageDirective) nzImage!: NzImageDirective; + src = ''; + placeholder: string | null = ''; +} + @Component({ template: ` ` }) @@ -511,7 +580,12 @@ export class TestImagePreviewGroupComponent { } } -const TEST_COMPONENTS = [TestImageFallbackComponent, TestImagePlaceholderComponent, TestImagePreviewGroupComponent]; +const TEST_COMPONENTS = [ + TestImageBasicsComponent, + TestImageFallbackComponent, + TestImagePlaceholderComponent, + TestImagePreviewGroupComponent +]; @NgModule({ imports: [NzImageModule],