1 | import { Directive, EventEmitter, Input, Output } from '@angular/core';
|
2 | import * as i0 from "@angular/core";
|
3 | import * as i1 from "./pagination.service";
|
4 | /**
|
5 | * This directive is what powers all pagination controls components, including the default one.
|
6 | * It exposes an API which is hooked up to the PaginationService to keep the PaginatePipe in sync
|
7 | * with the pagination controls.
|
8 | */
|
9 | export class PaginationControlsDirective {
|
10 | constructor(service, changeDetectorRef) {
|
11 | this.service = service;
|
12 | this.changeDetectorRef = changeDetectorRef;
|
13 | this.maxSize = 7;
|
14 | this.pageChange = new EventEmitter();
|
15 | this.pageBoundsCorrection = new EventEmitter();
|
16 | this.pages = [];
|
17 | this.changeSub = this.service.change
|
18 | .subscribe(id => {
|
19 | if (this.id === id) {
|
20 | this.updatePageLinks();
|
21 | this.changeDetectorRef.markForCheck();
|
22 | this.changeDetectorRef.detectChanges();
|
23 | }
|
24 | });
|
25 | }
|
26 | ngOnInit() {
|
27 | if (this.id === undefined) {
|
28 | this.id = this.service.defaultId();
|
29 | }
|
30 | this.updatePageLinks();
|
31 | }
|
32 | ngOnChanges(changes) {
|
33 | this.updatePageLinks();
|
34 | }
|
35 | ngOnDestroy() {
|
36 | this.changeSub.unsubscribe();
|
37 | }
|
38 | /**
|
39 | * Go to the previous page
|
40 | */
|
41 | previous() {
|
42 | this.checkValidId();
|
43 | this.setCurrent(this.getCurrent() - 1);
|
44 | }
|
45 | /**
|
46 | * Go to the next page
|
47 | */
|
48 | next() {
|
49 | this.checkValidId();
|
50 | this.setCurrent(this.getCurrent() + 1);
|
51 | }
|
52 | /**
|
53 | * Returns true if current page is first page
|
54 | */
|
55 | isFirstPage() {
|
56 | return this.getCurrent() === 1;
|
57 | }
|
58 | /**
|
59 | * Returns true if current page is last page
|
60 | */
|
61 | isLastPage() {
|
62 | return this.getLastPage() === this.getCurrent();
|
63 | }
|
64 | /**
|
65 | * Set the current page number.
|
66 | */
|
67 | setCurrent(page) {
|
68 | this.pageChange.emit(page);
|
69 | }
|
70 | /**
|
71 | * Get the current page number.
|
72 | */
|
73 | getCurrent() {
|
74 | return this.service.getCurrentPage(this.id);
|
75 | }
|
76 | /**
|
77 | * Returns the last page number
|
78 | */
|
79 | getLastPage() {
|
80 | let inst = this.service.getInstance(this.id);
|
81 | if (inst.totalItems < 1) {
|
82 | // when there are 0 or fewer (an error case) items, there are no "pages" as such,
|
83 | // but it makes sense to consider a single, empty page as the last page.
|
84 | return 1;
|
85 | }
|
86 | return Math.ceil(inst.totalItems / inst.itemsPerPage);
|
87 | }
|
88 | getTotalItems() {
|
89 | return this.service.getInstance(this.id).totalItems;
|
90 | }
|
91 | checkValidId() {
|
92 | if (this.service.getInstance(this.id).id == null) {
|
93 | console.warn(`PaginationControlsDirective: the specified id "${this.id}" does not match any registered PaginationInstance`);
|
94 | }
|
95 | }
|
96 | /**
|
97 | * Updates the page links and checks that the current page is valid. Should run whenever the
|
98 | * PaginationService.change stream emits a value matching the current ID, or when any of the
|
99 | * input values changes.
|
100 | */
|
101 | updatePageLinks() {
|
102 | let inst = this.service.getInstance(this.id);
|
103 | const correctedCurrentPage = this.outOfBoundCorrection(inst);
|
104 | if (correctedCurrentPage !== inst.currentPage) {
|
105 | setTimeout(() => {
|
106 | this.pageBoundsCorrection.emit(correctedCurrentPage);
|
107 | this.pages = this.createPageArray(inst.currentPage, inst.itemsPerPage, inst.totalItems, this.maxSize);
|
108 | });
|
109 | }
|
110 | else {
|
111 | this.pages = this.createPageArray(inst.currentPage, inst.itemsPerPage, inst.totalItems, this.maxSize);
|
112 | }
|
113 | }
|
114 | /**
|
115 | * Checks that the instance.currentPage property is within bounds for the current page range.
|
116 | * If not, return a correct value for currentPage, or the current value if OK.
|
117 | */
|
118 | outOfBoundCorrection(instance) {
|
119 | const totalPages = Math.ceil(instance.totalItems / instance.itemsPerPage);
|
120 | if (totalPages < instance.currentPage && 0 < totalPages) {
|
121 | return totalPages;
|
122 | }
|
123 | else if (instance.currentPage < 1) {
|
124 | return 1;
|
125 | }
|
126 | return instance.currentPage;
|
127 | }
|
128 | /**
|
129 | * Returns an array of Page objects to use in the pagination controls.
|
130 | */
|
131 | createPageArray(currentPage, itemsPerPage, totalItems, paginationRange) {
|
132 | // paginationRange could be a string if passed from attribute, so cast to number.
|
133 | paginationRange = +paginationRange;
|
134 | let pages = [];
|
135 | // Return 1 as default page number
|
136 | // Make sense to show 1 instead of empty when there are no items
|
137 | const totalPages = Math.max(Math.ceil(totalItems / itemsPerPage), 1);
|
138 | const halfWay = Math.ceil(paginationRange / 2);
|
139 | const isStart = currentPage <= halfWay;
|
140 | const isEnd = totalPages - halfWay < currentPage;
|
141 | const isMiddle = !isStart && !isEnd;
|
142 | let ellipsesNeeded = paginationRange < totalPages;
|
143 | let i = 1;
|
144 | while (i <= totalPages && i <= paginationRange) {
|
145 | let label;
|
146 | let pageNumber = this.calculatePageNumber(i, currentPage, paginationRange, totalPages);
|
147 | let openingEllipsesNeeded = (i === 2 && (isMiddle || isEnd));
|
148 | let closingEllipsesNeeded = (i === paginationRange - 1 && (isMiddle || isStart));
|
149 | if (ellipsesNeeded && (openingEllipsesNeeded || closingEllipsesNeeded)) {
|
150 | label = '...';
|
151 | }
|
152 | else {
|
153 | label = pageNumber;
|
154 | }
|
155 | pages.push({
|
156 | label: label,
|
157 | value: pageNumber
|
158 | });
|
159 | i++;
|
160 | }
|
161 | return pages;
|
162 | }
|
163 | /**
|
164 | * Given the position in the sequence of pagination links [i],
|
165 | * figure out what page number corresponds to that position.
|
166 | */
|
167 | calculatePageNumber(i, currentPage, paginationRange, totalPages) {
|
168 | let halfWay = Math.ceil(paginationRange / 2);
|
169 | if (i === paginationRange) {
|
170 | return totalPages;
|
171 | }
|
172 | else if (i === 1) {
|
173 | return i;
|
174 | }
|
175 | else if (paginationRange < totalPages) {
|
176 | if (totalPages - halfWay < currentPage) {
|
177 | return totalPages - paginationRange + i;
|
178 | }
|
179 | else if (halfWay < currentPage) {
|
180 | return currentPage - halfWay + i;
|
181 | }
|
182 | else {
|
183 | return i;
|
184 | }
|
185 | }
|
186 | else {
|
187 | return i;
|
188 | }
|
189 | }
|
190 | }
|
191 | PaginationControlsDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.9", ngImport: i0, type: PaginationControlsDirective, deps: [{ token: i1.PaginationService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Directive });
|
192 | PaginationControlsDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.9", type: PaginationControlsDirective, selector: "pagination-template,[pagination-template]", inputs: { id: "id", maxSize: "maxSize" }, outputs: { pageChange: "pageChange", pageBoundsCorrection: "pageBoundsCorrection" }, exportAs: ["paginationApi"], usesOnChanges: true, ngImport: i0 });
|
193 | i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.9", ngImport: i0, type: PaginationControlsDirective, decorators: [{
|
194 | type: Directive,
|
195 | args: [{
|
196 | selector: 'pagination-template,[pagination-template]',
|
197 | exportAs: 'paginationApi'
|
198 | }]
|
199 | }], ctorParameters: function () { return [{ type: i1.PaginationService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { id: [{
|
200 | type: Input
|
201 | }], maxSize: [{
|
202 | type: Input
|
203 | }], pageChange: [{
|
204 | type: Output
|
205 | }], pageBoundsCorrection: [{
|
206 | type: Output
|
207 | }] } });
|
208 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination-controls.directive.js","sourceRoot":"","sources":["../../../../projects/ngx-pagination/src/lib/pagination-controls.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectorRef, Directive, EventEmitter, Input, Output} from '@angular/core';\r\nimport {Subscription} from 'rxjs';\r\n\r\nimport {PaginationService} from './pagination.service';\r\nimport {PaginationInstance} from './pagination-instance';\r\n\r\nexport interface Page {\r\n    label: string;\r\n    value: any;\r\n}\r\n\r\n/**\r\n * This directive is what powers all pagination controls components, including the default one.\r\n * It exposes an API which is hooked up to the PaginationService to keep the PaginatePipe in sync\r\n * with the pagination controls.\r\n */\r\n@Directive({\r\n    selector: 'pagination-template,[pagination-template]',\r\n    exportAs: 'paginationApi'\r\n})\r\nexport class PaginationControlsDirective {\r\n    @Input() id: string;\r\n    @Input() maxSize: number = 7;\r\n    @Output() pageChange: EventEmitter<number> = new EventEmitter<number>();\r\n    @Output() pageBoundsCorrection: EventEmitter<number> = new EventEmitter<number>();\r\n    pages: Page[] = [];\r\n\r\n    private changeSub: Subscription;\r\n\r\n    constructor(private service: PaginationService,\r\n                private changeDetectorRef: ChangeDetectorRef) {\r\n        this.changeSub = this.service.change\r\n            .subscribe(id => {\r\n                if (this.id === id) {\r\n                    this.updatePageLinks();\r\n                    this.changeDetectorRef.markForCheck();\r\n                    this.changeDetectorRef.detectChanges();\r\n                }\r\n            });\r\n    }\r\n\r\n    ngOnInit() {\r\n        if (this.id === undefined) {\r\n            this.id = this.service.defaultId();\r\n        }\r\n        this.updatePageLinks();\r\n    }\r\n\r\n    ngOnChanges(changes: any) {\r\n        this.updatePageLinks();\r\n    }\r\n\r\n    ngOnDestroy() {\r\n        this.changeSub.unsubscribe();\r\n    }\r\n\r\n    /**\r\n     * Go to the previous page\r\n     */\r\n    previous() {\r\n        this.checkValidId();\r\n        this.setCurrent(this.getCurrent() - 1);\r\n    }\r\n\r\n    /**\r\n     * Go to the next page\r\n     */\r\n    next() {\r\n        this.checkValidId();\r\n        this.setCurrent(this.getCurrent() + 1);\r\n    }\r\n\r\n    /**\r\n     * Returns true if current page is first page\r\n     */\r\n    isFirstPage(): boolean {\r\n        return this.getCurrent() === 1;\r\n    }\r\n\r\n    /**\r\n     * Returns true if current page is last page\r\n     */\r\n    isLastPage(): boolean {\r\n        return this.getLastPage() === this.getCurrent();\r\n    }\r\n\r\n    /**\r\n     * Set the current page number.\r\n     */\r\n    setCurrent(page: number) {\r\n        this.pageChange.emit(page);\r\n    }\r\n\r\n    /**\r\n     * Get the current page number.\r\n     */\r\n    getCurrent(): number {\r\n        return this.service.getCurrentPage(this.id);\r\n    }\r\n\r\n    /**\r\n     * Returns the last page number\r\n     */\r\n    getLastPage(): number {\r\n        let inst = this.service.getInstance(this.id);\r\n        if (inst.totalItems < 1) {\r\n            // when there are 0 or fewer (an error case) items, there are no \"pages\" as such,\r\n            // but it makes sense to consider a single, empty page as the last page.\r\n            return 1;\r\n        }\r\n        return Math.ceil(inst.totalItems / inst.itemsPerPage);\r\n    }\r\n\r\n    getTotalItems(): number {\r\n        return this.service.getInstance(this.id).totalItems;\r\n    }\r\n\r\n    private checkValidId() {\r\n        if (this.service.getInstance(this.id).id == null) {\r\n            console.warn(`PaginationControlsDirective: the specified id \"${this.id}\" does not match any registered PaginationInstance`);\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Updates the page links and checks that the current page is valid. Should run whenever the\r\n     * PaginationService.change stream emits a value matching the current ID, or when any of the\r\n     * input values changes.\r\n     */\r\n    private updatePageLinks() {\r\n        let inst = this.service.getInstance(this.id);\r\n        const correctedCurrentPage = this.outOfBoundCorrection(inst);\r\n\r\n        if (correctedCurrentPage !== inst.currentPage) {\r\n            setTimeout(() => {\r\n                this.pageBoundsCorrection.emit(correctedCurrentPage);\r\n                this.pages = this.createPageArray(inst.currentPage, inst.itemsPerPage, inst.totalItems, this.maxSize);\r\n            });\r\n        } else {\r\n            this.pages = this.createPageArray(inst.currentPage, inst.itemsPerPage, inst.totalItems, this.maxSize);\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Checks that the instance.currentPage property is within bounds for the current page range.\r\n     * If not, return a correct value for currentPage, or the current value if OK.\r\n     */\r\n    private outOfBoundCorrection(instance: PaginationInstance): number {\r\n        const totalPages = Math.ceil(instance.totalItems / instance.itemsPerPage);\r\n        if (totalPages < instance.currentPage && 0 < totalPages) {\r\n            return totalPages;\r\n        } else if (instance.currentPage < 1) {\r\n            return 1;\r\n        }\r\n\r\n        return instance.currentPage;\r\n    }\r\n\r\n    /**\r\n     * Returns an array of Page objects to use in the pagination controls.\r\n     */\r\n    private createPageArray(currentPage: number, itemsPerPage: number, totalItems: number, paginationRange: number): Page[] {\r\n        // paginationRange could be a string if passed from attribute, so cast to number.\r\n        paginationRange = +paginationRange;\r\n        let pages = [];\r\n        \r\n        // Return 1 as default page number\r\n        // Make sense to show 1 instead of empty when there are no items\r\n        const totalPages = Math.max(Math.ceil(totalItems / itemsPerPage), 1);\r\n        const halfWay = Math.ceil(paginationRange / 2);\r\n\r\n        const isStart = currentPage <= halfWay;\r\n        const isEnd = totalPages - halfWay < currentPage;\r\n        const isMiddle = !isStart && !isEnd;\r\n\r\n        let ellipsesNeeded = paginationRange < totalPages;\r\n        let i = 1;\r\n\r\n        while (i <= totalPages && i <= paginationRange) {\r\n            let label;\r\n            let pageNumber = this.calculatePageNumber(i, currentPage, paginationRange, totalPages);\r\n            let openingEllipsesNeeded = (i === 2 && (isMiddle || isEnd));\r\n            let closingEllipsesNeeded = (i === paginationRange - 1 && (isMiddle || isStart));\r\n            if (ellipsesNeeded && (openingEllipsesNeeded || closingEllipsesNeeded)) {\r\n                label = '...';\r\n            } else {\r\n                label = pageNumber;\r\n            }\r\n            pages.push({\r\n                label: label,\r\n                value: pageNumber\r\n            });\r\n            i ++;\r\n        }\r\n        return pages;\r\n    }\r\n\r\n    /**\r\n     * Given the position in the sequence of pagination links [i],\r\n     * figure out what page number corresponds to that position.\r\n     */\r\n    private calculatePageNumber(i: number, currentPage: number, paginationRange: number, totalPages: number) {\r\n        let halfWay = Math.ceil(paginationRange / 2);\r\n        if (i === paginationRange) {\r\n            return totalPages;\r\n        } else if (i === 1) {\r\n            return i;\r\n        } else if (paginationRange < totalPages) {\r\n            if (totalPages - halfWay < currentPage) {\r\n                return totalPages - paginationRange + i;\r\n            } else if (halfWay < currentPage) {\r\n                return currentPage - halfWay + i;\r\n            } else {\r\n                return i;\r\n            }\r\n        } else {\r\n            return i;\r\n        }\r\n    }\r\n}\r\n"]} |
\ | No newline at end of file |