UNPKG

ng-cw-v12

Version:

Angular UI Component Library

127 lines 21.6 kB
import { Component, Input, ViewChild } from '@angular/core'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class VirtualScrollComponent { constructor(ngZone) { this.ngZone = ngZone; /** 列表数据 */ this.ncItems = []; /** 缓冲item数量,前后各2项 */ this.ncBufferCount = 2; this.initialized = false; // 标记 itemHeight 是否已初始化(模板中 *ngIf 使用,需为 public) this.viewInitialized = false; // 标记 ngAfterViewInit 是否已执行(templateContainer 才可用) this.visibleItems = []; //显示items this.totalPadding = 0; //总填充高度 this.topPadding = 0; this.scrollTop = 0; //已滚动高度 this.itemHeight = 1; //若设为0,下方代码获取不到this.templateContainer this.viewportHeight = 0; //视窗可视高度 } /** * 场景 触发行 原因 * 初始就有值(同步) 第 49 行measureAndUpdate ngOnChanges 先执行但 viewInitialized=false 跳过,由 ngAfterViewInit 补调 * 初始无值,后续赋值(异步) 第 75 行measureAndUpdate ngAfterViewInit 时无数据跳过,数据到达时 ngOnChanges 直接调用 */ ngAfterViewInit() { this.viewportHeight = this.viewport.nativeElement.clientHeight; this.viewInitialized = true; // 若 ncItems 在 ngAfterViewInit 之前就已同步赋值,则在此补充执行初始化 if (!this.initialized && this.ncItems.length > 0) { this.measureAndUpdate(); } // 监听 viewport 容器高度变化,在 Angular 区域外观察以避免频繁触发变更检测 this.ngZone.runOutsideAngular(() => { this.resizeObserver = new ResizeObserver(entries => { const newHeight = entries[0].contentRect.height; if (newHeight !== this.viewportHeight) { this.ngZone.run(() => { this.viewportHeight = newHeight; this.updateVisibleItems(); }); } }); this.resizeObserver.observe(this.viewport.nativeElement); }); } ngOnDestroy() { var _a; (_a = this.resizeObserver) === null || _a === void 0 ? void 0 : _a.disconnect(); } ngOnChanges(changes) { if (changes['ncItems'] && this.ncItems.length > 0) { if (!this.initialized) { // templateContainer 仅在 ngAfterViewInit 后可用,未初始化则等待 if (this.viewInitialized) { this.measureAndUpdate(); } // 否则由 ngAfterViewInit 补调,无需处理 } else { this.updateTotalPadding(); this.updateVisibleItems(); } } } /** 测量 itemHeight 并更新列表(需在 ngAfterViewInit 之后调用) */ measureAndUpdate() { // setTimeout(0) 确保 Angular 已完成本次 DOM 渲染 setTimeout(() => { var _a, _b; const element = (_b = (_a = this.templateContainer) === null || _a === void 0 ? void 0 : _a.nativeElement) === null || _b === void 0 ? void 0 : _b.firstChild; if (element) { const styles = getComputedStyle(element); this.itemHeight = element.offsetHeight + parseFloat(styles.marginTop) + parseFloat(styles.marginBottom); this.initialized = true; } this.updateTotalPadding(); this.updateVisibleItems(); }); } onScroll() { this.ngZone.run(() => { this.scrollTop = this.viewport.nativeElement.scrollTop; this.updateVisibleItems(); }); } updateVisibleItems() { const firstVisibleIndex = Math.floor(this.scrollTop / this.itemHeight); const startIndex = Math.max(0, firstVisibleIndex - this.ncBufferCount); const visibleItemsCount = Math.ceil(this.viewportHeight / this.itemHeight); const endIndex = Math.min(firstVisibleIndex + visibleItemsCount + this.ncBufferCount, this.ncItems.length - 1); this.visibleItems = []; for (let i = startIndex; i <= endIndex; i++) { this.visibleItems.push({ index: i, data: this.ncItems[i] }); } this.topPadding = startIndex * this.itemHeight; } updateTotalPadding() { this.totalPadding = this.ncItems.length * this.itemHeight; } //采用trackBy提升性能,只渲染改变的dom trackByIndex(index, item) { return item.index; } } VirtualScrollComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.1.5", ngImport: i0, type: VirtualScrollComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); VirtualScrollComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.1.5", type: VirtualScrollComponent, selector: "nc-virtual-scroll", inputs: { ncItems: "ncItems", ncItemTemplate: "ncItemTemplate", ncBufferCount: "ncBufferCount" }, viewQueries: [{ propertyName: "viewport", first: true, predicate: ["viewport"], descendants: true, static: true }, { propertyName: "templateContainer", first: true, predicate: ["templateContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<!-- \u7528\u4E8E\u6D4B\u91CF itemHeight \u7684\u9690\u85CF\u63A2\u9488\uFF0C\u4EC5\u5728\u521D\u59CB\u5316\u524D\u4E14\u6709\u6570\u636E\u65F6\u6E32\u67D3\uFF0C\u4E0E visibleItems \u5B8C\u5168\u89E3\u8026 -->\r\n<div #templateContainer *ngIf=\"!initialized && ncItems.length > 0\"\r\n style=\"visibility:hidden;position:absolute;pointer-events:none;\">\r\n <ng-container [ngTemplateOutlet]=\"ncItemTemplate\"\r\n [ngTemplateOutletContext]=\"{$implicit: ncItems[0], index: 0}\"></ng-container>\r\n</div>\r\n\r\n<div class=\"viewport\" #viewport (scroll)=\"onScroll()\">\r\n <div class=\"total-padding\" [style.height.px]=\"totalPadding\">\r\n <div class=\"item-container\" [style.transform]=\"'translateY(' + topPadding + 'px)'\">\r\n <div *ngFor=\"let item of visibleItems; trackBy: trackByIndex\">\r\n <ng-container [ngTemplateOutlet]=\"ncItemTemplate\"\r\n [ngTemplateOutletContext]=\"{$implicit: item.data, index: item.index}\"></ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n</div>", styles: [".viewport{width:100%;height:100%;overflow-y:auto;position:relative}.viewport::-webkit-scrollbar{width:6px;height:6px}.viewport::-webkit-scrollbar-thumb{background-color:#cdc9cf;border-radius:10px;-webkit-transition:background-color .3s ease;transition:background-color .3s ease}.viewport::-webkit-scrollbar-thumb:hover{background-color:#766d7b}.viewport::-webkit-scrollbar-track{background:#0000;cursor:pointer}\n"], directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.1.5", ngImport: i0, type: VirtualScrollComponent, decorators: [{ type: Component, args: [{ selector: 'nc-virtual-scroll', templateUrl: './virtual-scroll.component.html', styleUrls: ['./virtual-scroll.component.less'] }] }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { viewport: [{ type: ViewChild, args: ['viewport', { static: true }] }], templateContainer: [{ type: ViewChild, args: ['templateContainer'] }], ncItems: [{ type: Input }], ncItemTemplate: [{ type: Input }], ncBufferCount: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmlydHVhbC1zY3JvbGwuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50cy92aXJ0dWFsLXNjcm9sbC92aXJ0dWFsLXNjcm9sbC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3ZpcnR1YWwtc2Nyb2xsL3ZpcnR1YWwtc2Nyb2xsLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUF3QyxTQUFTLEVBQW9DLE1BQU0sZUFBZSxDQUFDOzs7QUFZcEksTUFBTSxPQUFPLHNCQUFzQjtJQXNCakMsWUFBb0IsTUFBYztRQUFkLFdBQU0sR0FBTixNQUFNLENBQVE7UUFsQmxDLFdBQVc7UUFDRixZQUFPLEdBQVUsRUFBRSxDQUFDO1FBRzdCLHFCQUFxQjtRQUNaLGtCQUFhLEdBQVcsQ0FBQyxDQUFDO1FBRW5DLGdCQUFXLEdBQUcsS0FBSyxDQUFDLENBQUcsK0NBQStDO1FBQzlELG9CQUFlLEdBQUcsS0FBSyxDQUFDLENBQUMsa0RBQWtEO1FBR25GLGlCQUFZLEdBQWtCLEVBQUUsQ0FBQyxDQUFBLFNBQVM7UUFDMUMsaUJBQVksR0FBVyxDQUFDLENBQUMsQ0FBQSxPQUFPO1FBQ2hDLGVBQVUsR0FBVyxDQUFDLENBQUM7UUFDdkIsY0FBUyxHQUFXLENBQUMsQ0FBQyxDQUFBLE9BQU87UUFDN0IsZUFBVSxHQUFXLENBQUMsQ0FBQyxDQUFBLHFDQUFxQztRQUM1RCxtQkFBYyxHQUFXLENBQUMsQ0FBQyxDQUFBLFFBQVE7SUFFRyxDQUFDO0lBR3ZDOzs7O09BSUc7SUFFSCxlQUFlO1FBQ2IsSUFBSSxDQUFDLGNBQWMsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLGFBQWEsQ0FBQyxZQUFZLENBQUM7UUFDL0QsSUFBSSxDQUFDLGVBQWUsR0FBRyxJQUFJLENBQUM7UUFDNUIsa0RBQWtE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aUJBQy9DOzZGQUUwQyxRQUFRO3NCQUFoRCxTQUFTO3VCQUFDLFVBQVUsRUFBRSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUU7Z0JBQ1AsaUJBQWlCO3NCQUFoRCxTQUFTO3VCQUFDLG1CQUFtQjtnQkFHckIsT0FBTztzQkFBZixLQUFLO2dCQUVHLGNBQWM7c0JBQXRCLEtBQUs7Z0JBRUcsYUFBYTtzQkFBckIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIEFmdGVyVmlld0luaXQsIE9uRGVzdHJveSwgRWxlbWVudFJlZiwgVmlld0NoaWxkLCBOZ1pvbmUsIE9uQ2hhbmdlcywgU2ltcGxlQ2hhbmdlcyB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5cclxuaW50ZXJmYWNlIFZpcnR1YWxJdGVtIHtcclxuICBpbmRleDogbnVtYmVyO1xyXG4gIGRhdGE6IGFueTtcclxufVxyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICduYy12aXJ0dWFsLXNjcm9sbCcsXHJcbiAgdGVtcGxhdGVVcmw6ICcuL3ZpcnR1YWwtc2Nyb2xsLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybHM6IFsnLi92aXJ0dWFsLXNjcm9sbC5jb21wb25lbnQubGVzcyddXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBWaXJ0dWFsU2Nyb2xsQ29tcG9uZW50IGltcGxlbWVudHMgQWZ0ZXJWaWV3SW5pdCwgT25DaGFuZ2VzLCBPbkRlc3Ryb3kge1xyXG4gIEBWaWV3Q2hpbGQoJ3ZpZXdwb3J0JywgeyBzdGF0aWM6IHRydWUgfSkgdmlld3BvcnQhOiBFbGVtZW50UmVmOy8v6KeG56qXXHJcbiAgQFZpZXdDaGlsZCgndGVtcGxhdGVDb250YWluZXInKSB0ZW1wbGF0ZUNvbnRhaW5lciE6IEVsZW1lbnRSZWY7Ly/nlKjkuo7ojrflj5bkvKDlhaXnmoRpdGVt6auY5bqm77yM6K6+572uaXRlbUhlaWdodOWAvFxyXG5cclxuICAvKiog5YiX6KGo5pWw5o2uICovXHJcbiAgQElucHV0KCkgbmNJdGVtczogYW55W10gPSBbXTtcclxuICAvKiog5YiX6KGo6aG55qih5p2/ICovXHJcbiAgQElucHV0KCkgbmNJdGVtVGVtcGxhdGU6IGFueTsvL+S8oOWFpeeahGl0ZW3mqKHmnb9cclxuICAvKiog57yT5YayaXRlbeaVsOmHj++8jOWJjeWQjuWQhDLpobkgKi9cclxuICBASW5wdXQoKSBuY0J1ZmZlckNvdW50OiBudW1iZXIgPSAyO1xyXG5cclxuICBpbml0aWFsaXplZCA9IGZhbHNlOyAgIC8vIOagh+iusCBpdGVtSGVpZ2h0IOaYr+WQpuW3suWIneWni+WMlu+8iOaooeadv+S4rSAqbmdJZiDkvb/nlKjvvIzpnIDkuLogcHVibGlj77yJXHJcbiAgcHJpdmF0ZSB2aWV3SW5pdGlhbGl6ZWQgPSBmYWxzZTsgLy8g5qCH6K6wIG5nQWZ0ZXJWaWV3SW5pdCDmmK/lkKblt7LmiafooYzvvIh0ZW1wbGF0ZUNvbnRhaW5lciDmiY3lj6/nlKjvvIlcclxuICBwcml2YXRlIHJlc2l6ZU9ic2VydmVyITogUmVzaXplT2JzZXJ2ZXI7IC8vIOebkeWQrCB2aWV3cG9ydCDlrrnlmajlsLrlr7jlj5jljJZcclxuXHJcbiAgdmlzaWJsZUl0ZW1zOiBWaXJ0dWFsSXRlbVtdID0gW107Ly/mmL7npLppdGVtc1xyXG4gIHRvdGFsUGFkZGluZzogbnVtYmVyID0gMDsvL+aAu+Whq+WFhemrmOW6plxyXG4gIHRvcFBhZGRpbmc6IG51bWJlciA9IDA7XHJcbiAgc2Nyb2xsVG9wOiBudW1iZXIgPSAwOy8v5bey5rua5Yqo6auY5bqmXHJcbiAgaXRlbUhlaWdodDogbnVtYmVyID0gMTsvL+iLpeiuvuS4ujDvvIzkuIvmlrnku6PnoIHojrflj5bkuI3liLB0aGlzLnRlbXBsYXRlQ29udGFpbmVyXHJcbiAgdmlld3BvcnRIZWlnaHQ6IG51bWJlciA9IDA7Ly/op4bnqpflj6/op4bpq5jluqZcclxuXHJcbiAgY29uc3RydWN0b3IocHJpdmF0ZSBuZ1pvbmU6IE5nWm9uZSkgeyB9XHJcblxyXG5cclxuICAvKipcclxuICAgKiDlnLrmma9cdOinpuWPkeihjFx05Y6f5ZugXHJcbiAgICog5Yid5aeL5bCx5pyJ5YC877yI5ZCM5q2l77yJXHTnrKwgNDkg6KGMbWVhc3VyZUFuZFVwZGF0ZVx0bmdPbkNoYW5nZXMg5YWI5omn6KGM5L2GIHZpZXdJbml0aWFsaXplZD1mYWxzZSDot7Pov4fvvIznlLEgbmdBZnRlclZpZXdJbml0IOihpeiwg1xyXG4gICAqIOWIneWni+aXoOWAvO+8jOWQjue7rei1i+WAvO+8iOW8guatpe+8iVx056ysIDc1IOihjG1lYXN1cmVBbmRVcGRhdGVcdG5nQWZ0ZXJWaWV3SW5pdCDml7bml6DmlbDmja7ot7Pov4fvvIzmlbDmja7liLDovr7ml7YgbmdPbkNoYW5nZXMg55u05o6l6LCD55SoXHJcbiAgICovXHJcblxyXG4gIG5nQWZ0ZXJWaWV3SW5pdCgpIHtcclxuICAgIHRoaXMudmlld3BvcnRIZWlnaHQgPSB0aGlzLnZpZXdwb3J0Lm5hdGl2ZUVsZW1lbnQuY2xpZW50SGVpZ2h0O1xyXG4gICAgdGhpcy52aWV3SW5pdGlhbGl6ZWQgPSB0cnVlO1xyXG4gICAgLy8g6IulIG5jSXRlbXMg5ZyoIG5nQWZ0ZXJWaWV3SW5pdCDkuYvliY3lsLHlt7LlkIzmraXotYvlgLzvvIzliJnlnKjmraTooaXlhYXmiafooYzliJ3lp4vljJZcclxuICAgIGlmICghdGhpcy5pbml0aWFsaXplZCAmJiB0aGlzLm5jSXRlbXMubGVuZ3RoID4gMCkge1xyXG4gICAgICB0aGlzLm1lYXN1cmVBbmRVcGRhdGUoKTtcclxuICAgIH1cclxuICAgIC8vIOebkeWQrCB2aWV3cG9ydCDlrrnlmajpq5jluqblj5jljJbvvIzlnKggQW5ndWxhciDljLrln5/lpJbop4Llr5/ku6Xpgb/lhY3popHnuYHop6blj5Hlj5jmm7Tmo4DmtYtcclxuICAgIHRoaXMubmdab25lLnJ1bk91dHNpZGVBbmd1bGFyKCgpID0+IHtcclxuICAgICAgdGhpcy5yZXNpemVPYnNlcnZlciA9IG5ldyBSZXNpemVPYnNlcnZlcihlbnRyaWVzID0+IHtcclxuICAgICAgICBjb25zdCBuZXdIZWlnaHQgPSBlbnRyaWVzWzBdLmNvbnRlbnRSZWN0LmhlaWdodDtcclxuICAgICAgICBpZiAobmV3SGVpZ2h0ICE9PSB0aGlzLnZpZXdwb3J0SGVpZ2h0KSB7XHJcbiAgICAgICAgICB0aGlzLm5nWm9uZS5ydW4oKCkgPT4ge1xyXG4gICAgICAgICAgICB0aGlzLnZpZXdwb3J0SGVpZ2h0ID0gbmV3SGVpZ2h0O1xyXG4gICAgICAgICAgICB0aGlzLnVwZGF0ZVZpc2libGVJdGVtcygpO1xyXG4gICAgICAgICAgfSk7XHJcbiAgICAgICAgfVxyXG4gICAgICB9KTtcclxuICAgICAgdGhpcy5yZXNpemVPYnNlcnZlci5vYnNlcnZlKHRoaXMudmlld3BvcnQubmF0aXZlRWxlbWVudCk7XHJcbiAgICB9KTtcclxuICB9XHJcblxyXG4gIG5nT25EZXN0cm95KCkge1xyXG4gICAgdGhpcy5yZXNpemVPYnNlcnZlcj8uZGlzY29ubmVjdCgpO1xyXG4gIH1cclxuXHJcbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcykge1xyXG4gICAgaWYgKGNoYW5nZXNbJ25jSXRlbXMnXSAmJiB0aGlzLm5jSXRlbXMubGVuZ3RoID4gMCkge1xyXG4gICAgICBpZiAoIXRoaXMuaW5pdGlhbGl6ZWQpIHtcclxuICAgICAgICAvLyB0ZW1wbGF0ZUNvbnRhaW5lciDku4XlnKggbmdBZnRlclZpZXdJbml0IOWQjuWPr+eUqO+8jOacquWIneWni+WMluWImeetieW+hVxyXG4gICAgICAgIGlmICh0aGlzLnZpZXdJbml0aWFsaXplZCkge1xyXG4gICAgICAgICAgdGhpcy5tZWFzdXJlQW5kVXBkYXRlKCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIC8vIOWQpuWImeeUsSBuZ0FmdGVyVmlld0luaXQg6KGl6LCD77yM5peg6ZyA5aSE55CGXHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgdGhpcy51cGRhdGVUb3RhbFBhZGRpbmcoKTtcclxuICAgICAgICB0aGlzLnVwZGF0ZVZpc2libGVJdGVtcygpO1xyXG4gICAgICB9XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvKiog5rWL6YePIGl0ZW1IZWlnaHQg5bm25pu05paw5YiX6KGo77yI6ZyA5ZyoIG5nQWZ0ZXJWaWV3SW5pdCDkuYvlkI7osIPnlKjvvIkgKi9cclxuICBwcml2YXRlIG1lYXN1cmVBbmRVcGRhdGUoKSB7XHJcbiAgICAvLyBzZXRUaW1lb3V0KDApIOehruS/nSBBbmd1bGFyIOW3suWujOaIkOacrOasoSBET00g5riy5p+TXHJcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcclxuICAgICAgY29uc3QgZWxlbWVudCA9IHRoaXMudGVtcGxhdGVDb250YWluZXI/Lm5hdGl2ZUVsZW1lbnQ/LmZpcnN0Q2hpbGQ7XHJcbiAgICAgIGlmIChlbGVtZW50KSB7XHJcbiAgICAgICAgY29uc3Qgc3R5bGVzID0gZ2V0Q29tcHV0ZWRTdHlsZShlbGVtZW50KTtcclxuICAgICAgICB0aGlzLml0ZW1IZWlnaHQgPSBlbGVtZW50Lm9mZnNldEhlaWdodCArIHBhcnNlRmxvYXQoc3R5bGVzLm1hcmdpblRvcCkgKyBwYXJzZUZsb2F0KHN0eWxlcy5tYXJnaW5Cb3R0b20pO1xyXG4gICAgICAgIHRoaXMuaW5pdGlhbGl6ZWQgPSB0cnVlO1xyXG4gICAgICB9XHJcbiAgICAgIHRoaXMudXBkYXRlVG90YWxQYWRkaW5nKCk7XHJcbiAgICAgIHRoaXMudXBkYXRlVmlzaWJsZUl0ZW1zKCk7XHJcbiAgICB9KTtcclxuICB9XHJcblxyXG4gIG9uU2Nyb2xsKCkge1xyXG4gICAgdGhpcy5uZ1pvbmUucnVuKCgpID0+IHtcclxuICAgICAgdGhpcy5zY3JvbGxUb3AgPSB0aGlzLnZpZXdwb3J0Lm5hdGl2ZUVsZW1lbnQuc2Nyb2xsVG9wO1xyXG4gICAgICB0aGlzLnVwZGF0ZVZpc2libGVJdGVtcygpO1xyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuICB1cGRhdGVWaXNpYmxlSXRlbXMoKSB7XHJcbiAgICBjb25zdCBmaXJzdFZpc2libGVJbmRleCA9IE1hdGguZmxvb3IodGhpcy5zY3JvbGxUb3AgLyB0aGlzLml0ZW1IZWlnaHQpO1xyXG4gICAgY29uc3Qgc3RhcnRJbmRleCA9IE1hdGgubWF4KDAsIGZpcnN0VmlzaWJsZUluZGV4IC0gdGhpcy5uY0J1ZmZlckNvdW50KTtcclxuICAgIGNvbnN0IHZpc2libGVJdGVtc0NvdW50ID0gTWF0aC5jZWlsKHRoaXMudmlld3BvcnRIZWlnaHQgLyB0aGlzLml0ZW1IZWlnaHQpO1xyXG4gICAgY29uc3QgZW5kSW5kZXggPSBNYXRoLm1pbihmaXJzdFZpc2libGVJbmRleCArIHZpc2libGVJdGVtc0NvdW50ICsgdGhpcy5uY0J1ZmZlckNvdW50LCB0aGlzLm5jSXRlbXMubGVuZ3RoIC0gMSk7XHJcblxyXG4gICAgdGhpcy52aXNpYmxlSXRlbXMgPSBbXTtcclxuICAgIGZvciAobGV0IGkgPSBzdGFydEluZGV4OyBpIDw9IGVuZEluZGV4OyBpKyspIHtcclxuICAgICAgdGhpcy52aXNpYmxlSXRlbXMucHVzaCh7IGluZGV4OiBpLCBkYXRhOiB0aGlzLm5jSXRlbXNbaV0gfSk7XHJcbiAgICB9XHJcblxyXG4gICAgdGhpcy50b3BQYWRkaW5nID0gc3RhcnRJbmRleCAqIHRoaXMuaXRlbUhlaWdodDtcclxuICB9XHJcblxyXG4gIHVwZGF0ZVRvdGFsUGFkZGluZygpIHtcclxuICAgIHRoaXMudG90YWxQYWRkaW5nID0gdGhpcy5uY0l0ZW1zLmxlbmd0aCAqIHRoaXMuaXRlbUhlaWdodDtcclxuICB9XHJcblxyXG4gIC8v6YeH55SodHJhY2tCeeaPkOWNh+aAp+iDve+8jOWPqua4suafk+aUueWPmOeahGRvbVxyXG4gIHRyYWNrQnlJbmRleChpbmRleDogbnVtYmVyLCBpdGVtOiBWaXJ0dWFsSXRlbSk6IG51bWJlciB7XHJcbiAgICByZXR1cm4gaXRlbS5pbmRleDtcclxuICB9XHJcbn1cclxuIiwiPCEtLSDnlKjkuo7mtYvph48gaXRlbUhlaWdodCDnmoTpmpDol4/mjqLpkojvvIzku4XlnKjliJ3lp4vljJbliY3kuJTmnInmlbDmja7ml7bmuLLmn5PvvIzkuI4gdmlzaWJsZUl0ZW1zIOWujOWFqOino+iApiAtLT5cclxuPGRpdiAjdGVtcGxhdGVDb250YWluZXIgKm5nSWY9XCIhaW5pdGlhbGl6ZWQgJiYgbmNJdGVtcy5sZW5ndGggPiAwXCJcclxuICAgIHN0eWxlPVwidmlzaWJpbGl0eTpoaWRkZW47cG9zaXRpb246YWJzb2x1dGU7cG9pbnRlci1ldmVudHM6bm9uZTtcIj5cclxuICAgIDxuZy1jb250YWluZXIgW25nVGVtcGxhdGVPdXRsZXRdPVwibmNJdGVtVGVtcGxhdGVcIlxyXG4gICAgICAgIFtuZ1RlbXBsYXRlT3V0bGV0Q29udGV4dF09XCJ7JGltcGxpY2l0OiBuY0l0ZW1zWzBdLCBpbmRleDogMH1cIj48L25nLWNvbnRhaW5lcj5cclxuPC9kaXY+XHJcblxyXG48ZGl2IGNsYXNzPVwidmlld3BvcnRcIiAjdmlld3BvcnQgKHNjcm9sbCk9XCJvblNjcm9sbCgpXCI+XHJcbiAgICA8ZGl2IGNsYXNzPVwidG90YWwtcGFkZGluZ1wiIFtzdHlsZS5oZWlnaHQucHhdPVwidG90YWxQYWRkaW5nXCI+XHJcbiAgICAgICAgPGRpdiBjbGFzcz1cIml0ZW0tY29udGFpbmVyXCIgW3N0eWxlLnRyYW5zZm9ybV09XCIndHJhbnNsYXRlWSgnICsgdG9wUGFkZGluZyArICdweCknXCI+XHJcbiAgICAgICAgICAgIDxkaXYgKm5nRm9yPVwibGV0IGl0ZW0gb2YgdmlzaWJsZUl0ZW1zOyB0cmFja0J5OiB0cmFja0J5SW5kZXhcIj5cclxuICAgICAgICAgICAgICAgIDxuZy1jb250YWluZXIgW25nVGVtcGxhdGVPdXRsZXRdPVwibmNJdGVtVGVtcGxhdGVcIlxyXG4gICAgICAgICAgICAgICAgICAgIFtuZ1RlbXBsYXRlT3V0bGV0Q29udGV4dF09XCJ7JGltcGxpY2l0OiBpdGVtLmRhdGEsIGluZGV4OiBpdGVtLmluZGV4fVwiPjwvbmctY29udGFpbmVyPlxyXG4gICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICA8L2Rpdj5cclxuICAgIDwvZGl2PlxyXG48L2Rpdj4iXX0=