Skip to content

Commit

Permalink
fix(module:image): unsubscribe old src (#7102)
Browse files Browse the repository at this point in the history
Co-authored-by: jiangchunlin <[email protected]>
  • Loading branch information
qugemingzizhenmafan and jiangchunlin authored Jan 14, 2022
1 parent b12f43a commit 87a3e27
Show file tree
Hide file tree
Showing 2 changed files with 84 additions and 5 deletions.
11 changes: 8 additions & 3 deletions components/image/image.directive.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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<void> = new Subject();
private destroy$: Subject<void> = new Subject();

get previewable(): boolean {
Expand Down Expand Up @@ -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;
Expand All @@ -145,15 +150,15 @@ 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;
this.getElement().nativeElement.srcset = this.nzSrcset;
});

fromEvent(this.backLoadImage, 'error')
.pipe(takeUntil(this.destroy$))
.pipe(takeUntil(this.backLoadDestroy$), takeUntil(this.destroy$))
.subscribe(() => {
this.status = 'error';
if (this.nzFallback) {
Expand Down
78 changes: 76 additions & 2 deletions components/image/image.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,14 +39,74 @@ 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 =
'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png?x-oss-process=image/blur,r_50,s_50/quality,q_1/resize,m_mfit,h_200,w_200';
const FALLBACK =
'';

describe('Basics', () => {
let fixture: ComponentFixture<TestImageBasicsComponent>;
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<TestImagePlaceholderComponent>;
let context: TestImagePlaceholderComponent;
Expand Down Expand Up @@ -466,6 +526,15 @@ describe('Preview', () => {
});
});

@Component({
template: ` <img nz-image [nzSrc]="src" [nzPlaceholder]="placeholder" /> `
})
export class TestImageBasicsComponent {
@ViewChild(NzImageDirective) nzImage!: NzImageDirective;
src = '';
placeholder: string | null = '';
}

@Component({
template: ` <img nz-image [nzSrc]="src" [nzPlaceholder]="placeholder" [nzDisablePreview]="disablePreview" /> `
})
Expand Down Expand Up @@ -511,7 +580,12 @@ export class TestImagePreviewGroupComponent {
}
}

const TEST_COMPONENTS = [TestImageFallbackComponent, TestImagePlaceholderComponent, TestImagePreviewGroupComponent];
const TEST_COMPONENTS = [
TestImageBasicsComponent,
TestImageFallbackComponent,
TestImagePlaceholderComponent,
TestImagePreviewGroupComponent
];

@NgModule({
imports: [NzImageModule],
Expand Down

0 comments on commit 87a3e27

Please sign in to comment.