UNPKG

@doku-dev/doku-fragment

Version:

A new Angular UI library that moving away from Bootstrap and built from scratch.

158 lines 26.8 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewEncapsulation, } from '@angular/core'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class DokuPagination { constructor(cdr) { this.cdr = cdr; /** * Total data items. * @default 0 */ this.totalData = 0; /** * Active page of the pagination. * * The value will be normalized to first or last page if out of total pages range. * * @default 0 */ this.activePage = 0; /** * Total items displayed per page. * @default 5 */ this.itemsPerPage = 5; /** * Listen for page change. */ this.pageChange = new EventEmitter(); /** * Use internally to inform other component for changes. * @internal */ this.triggerChange = new EventEmitter(); this.pages = []; this.firstPage = 0; this.lastPage = 0; this.totalPages = 0; this.totalPagesToShow = 7; } ngOnChanges(changes) { const totalDataChange = changes['totalData']?.previousValue !== changes['totalData']?.currentValue; const activePageChange = changes['activePage']?.previousValue !== changes['activePage']?.currentValue; const itemsPerPageChange = changes['itemsPerPage']?.previousValue !== changes['itemsPerPage']?.currentValue; if (totalDataChange || activePageChange || itemsPerPageChange) { this.calculateAll(); } } /** * Navigate to the specific page. * * If page value is out of total pages range, the action will be ignored. */ navigateToPage(page) { if (page < this.firstPage || page > this.lastPage || this.activePage === page) return; this.activePage = page; this.calculateAll(); this.emitPageChange(); } /** * Go to previous page. * * The action will be ignored if already on the first page. */ goToPreviousPage() { if (this.activePage <= this.firstPage) return; this.activePage -= 1; this.calculateAll(); this.emitPageChange(); } /** * Go to next page. * * The action will be ignored if already on the last page. */ goToNextPage() { if (this.activePage >= this.lastPage) return; this.activePage += 1; this.calculateAll(); this.emitPageChange(); } /** * @internal */ changeItemsPerPage(value) { this.itemsPerPage = value; this.cdr.markForCheck(); this.calculateAll(); this.emitPageChange(); } calculateAll() { this.calculateValues(); this.calculatePages(); this.triggerChange.emit(); } calculateValues() { this.totalPages = Math.ceil(this.totalData / this.itemsPerPage); this.firstPage = this.totalPages >= 1 ? 1 : 0; this.lastPage = this.totalPages >= 1 ? this.totalPages : 0; if (this.totalPages && this.activePage < 1) this.activePage = 1; if (this.totalPages && this.activePage > this.totalPages) this.activePage = this.totalPages; } calculatePages() { const totalPagesToShowInMiddle = this.totalPagesToShow - 4; const offsetPagesToShow = totalPagesToShowInMiddle + 1; const totalPagesToShowOnSide = offsetPagesToShow + 1; // Handle pages if `totalPages` is less than or equal to `totalPagesToShow`, then no need ellipsis. if (this.totalPages <= this.totalPagesToShow) { this.pages = Array.from(Array(this.totalPages)).map((_, idx) => idx + 1); return; } // Handle pages with ellipsis before the last page. if (this.activePage <= offsetPagesToShow) { const numbers = Array.from(Array(totalPagesToShowOnSide)).map((_, idx) => idx + 1); this.pages = [...numbers, '...', this.lastPage]; return; } // Handle pages with ellipsis after the first page. if (this.totalPages - this.activePage < offsetPagesToShow) { const numbers = Array.from(Array(totalPagesToShowOnSide)) .map((_, idx) => this.totalPages - idx) .reverse(); this.pages = [this.firstPage, '...', ...numbers]; return; } // Handle pages with ellipsis after the first page and before the last page. const startNumber = this.activePage - (totalPagesToShowInMiddle - 2); // const endNumber = this.activePage + (totalPagesToShowInMiddle - 2); const numbers = Array.from(Array(totalPagesToShowInMiddle)).map((_, idx) => startNumber + idx); this.pages = [this.firstPage, '...', ...numbers, '...', this.lastPage]; } emitPageChange() { this.pageChange.emit({ activePage: this.activePage, itemsPerPage: this.itemsPerPage, }); } } DokuPagination.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuPagination, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); DokuPagination.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuPagination, isStandalone: true, selector: "doku-pagination", inputs: { totalData: "totalData", activePage: "activePage", itemsPerPage: "itemsPerPage" }, outputs: { pageChange: "pageChange" }, exportAs: ["dokuPagination"], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"totalData\" class=\"d-pagination\">\n <div\n class=\"d-pagination-prev\"\n [class.disabled]=\"activePage <= firstPage\"\n (click)=\"goToPreviousPage()\"\n >\n <ng-template *ngTemplateOutlet=\"iconPrevious\"></ng-template>\n </div>\n\n <div *ngFor=\"let page of pages\">\n <div *ngIf=\"page === '...'; else pageNumber\" class=\"d-pagination-page-ellipsis\">...</div>\n <ng-template #pageNumber>\n <div\n class=\"d-pagination-page-number\"\n [class.active]=\"page === activePage\"\n (click)=\"navigateToPage($any(page))\"\n >\n {{ page }}\n </div>\n </ng-template>\n </div>\n\n <div class=\"d-pagination-next\" [class.disabled]=\"activePage >= lastPage\" (click)=\"goToNextPage()\">\n <ng-template *ngTemplateOutlet=\"iconNext\"></ng-template>\n </div>\n</div>\n\n<ng-template #iconPrevious>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"42\" height=\"46\" viewBox=\"0 0 42 46\" fill=\"none\">\n <path\n d=\"M23.5252 27.6092L18.8851 22.2096L23.5252 16.81C23.9916 16.2673 23.9916 15.3905 23.5252 14.8478C23.3018 14.5872 22.9984 14.4408 22.6821 14.4408C22.3658 14.4408 22.0624 14.5872 21.839 14.8478L16.3498 21.2354C15.8834 21.7782 15.8834 22.6549 16.3498 23.1977L21.839 29.5853C22.3054 30.128 23.0588 30.128 23.5252 29.5853C23.9796 29.0426 23.9916 28.1519 23.5252 27.6092Z\"\n fill=\"currentColor\"\n />\n </svg>\n</ng-template>\n\n<ng-template #iconNext>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"42\" height=\"46\" viewBox=\"0 0 42 46\" fill=\"none\">\n <path\n d=\"M18.4748 27.6092L23.1149 22.2096L18.4748 16.81C18.0084 16.2673 18.0084 15.3905 18.4748 14.8478C18.6982 14.5872 19.0016 14.4408 19.3179 14.4408C19.6342 14.4408 19.9376 14.5872 20.161 14.8478L25.6502 21.2354C26.1166 21.7782 26.1166 22.6549 25.6502 23.1977L20.161 29.5853C19.6946 30.128 18.9412 30.128 18.4748 29.5853C18.0204 29.0426 18.0084 28.1519 18.4748 27.6092Z\"\n fill=\"currentColor\"\n />\n </svg>\n</ng-template>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuPagination, decorators: [{ type: Component, args: [{ selector: 'doku-pagination', exportAs: 'dokuPagination', standalone: true, imports: [CommonModule], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div *ngIf=\"totalData\" class=\"d-pagination\">\n <div\n class=\"d-pagination-prev\"\n [class.disabled]=\"activePage <= firstPage\"\n (click)=\"goToPreviousPage()\"\n >\n <ng-template *ngTemplateOutlet=\"iconPrevious\"></ng-template>\n </div>\n\n <div *ngFor=\"let page of pages\">\n <div *ngIf=\"page === '...'; else pageNumber\" class=\"d-pagination-page-ellipsis\">...</div>\n <ng-template #pageNumber>\n <div\n class=\"d-pagination-page-number\"\n [class.active]=\"page === activePage\"\n (click)=\"navigateToPage($any(page))\"\n >\n {{ page }}\n </div>\n </ng-template>\n </div>\n\n <div class=\"d-pagination-next\" [class.disabled]=\"activePage >= lastPage\" (click)=\"goToNextPage()\">\n <ng-template *ngTemplateOutlet=\"iconNext\"></ng-template>\n </div>\n</div>\n\n<ng-template #iconPrevious>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"42\" height=\"46\" viewBox=\"0 0 42 46\" fill=\"none\">\n <path\n d=\"M23.5252 27.6092L18.8851 22.2096L23.5252 16.81C23.9916 16.2673 23.9916 15.3905 23.5252 14.8478C23.3018 14.5872 22.9984 14.4408 22.6821 14.4408C22.3658 14.4408 22.0624 14.5872 21.839 14.8478L16.3498 21.2354C15.8834 21.7782 15.8834 22.6549 16.3498 23.1977L21.839 29.5853C22.3054 30.128 23.0588 30.128 23.5252 29.5853C23.9796 29.0426 23.9916 28.1519 23.5252 27.6092Z\"\n fill=\"currentColor\"\n />\n </svg>\n</ng-template>\n\n<ng-template #iconNext>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"42\" height=\"46\" viewBox=\"0 0 42 46\" fill=\"none\">\n <path\n d=\"M18.4748 27.6092L23.1149 22.2096L18.4748 16.81C18.0084 16.2673 18.0084 15.3905 18.4748 14.8478C18.6982 14.5872 19.0016 14.4408 19.3179 14.4408C19.6342 14.4408 19.9376 14.5872 20.161 14.8478L25.6502 21.2354C26.1166 21.7782 26.1166 22.6549 25.6502 23.1977L20.161 29.5853C19.6946 30.128 18.9412 30.128 18.4748 29.5853C18.0204 29.0426 18.0084 28.1519 18.4748 27.6092Z\"\n fill=\"currentColor\"\n />\n </svg>\n</ng-template>\n" }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { totalData: [{ type: Input }], activePage: [{ type: Input }], itemsPerPage: [{ type: Input }], pageChange: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.component.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/pagination/pagination.component.ts","../../../../../../projects/doku-fragment/src/lib/pagination/pagination.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { DokuPageChangeProps } from './pagination.interface';\n\n@Component({\n  selector: 'doku-pagination',\n  exportAs: 'dokuPagination',\n  standalone: true,\n  imports: [CommonModule],\n  templateUrl: './pagination.component.html',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DokuPagination implements OnChanges {\n  /**\n   * Total data items.\n   * @default 0\n   */\n  @Input() totalData = 0;\n\n  /**\n   * Active page of the pagination.\n   *\n   * The value will be normalized to first or last page if out of total pages range.\n   *\n   * @default 0\n   */\n  @Input() activePage = 0;\n\n  /**\n   * Total items displayed per page.\n   * @default 5\n   */\n  @Input() itemsPerPage = 5;\n\n  /**\n   * Listen for page change.\n   */\n  @Output() pageChange = new EventEmitter<DokuPageChangeProps>();\n\n  /**\n   * Use internally to inform other component for changes.\n   * @internal\n   */\n  protected triggerChange = new EventEmitter();\n\n  protected pages: (number | string)[] = [];\n  protected firstPage = 0;\n  protected lastPage = 0;\n  private totalPages = 0;\n  private readonly totalPagesToShow = 7;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const totalDataChange =\n      changes['totalData']?.previousValue !== changes['totalData']?.currentValue;\n    const activePageChange =\n      changes['activePage']?.previousValue !== changes['activePage']?.currentValue;\n    const itemsPerPageChange =\n      changes['itemsPerPage']?.previousValue !== changes['itemsPerPage']?.currentValue;\n\n    if (totalDataChange || activePageChange || itemsPerPageChange) {\n      this.calculateAll();\n    }\n  }\n\n  /**\n   * Navigate to the specific page.\n   *\n   * If page value is out of total pages range, the action will be ignored.\n   */\n  navigateToPage(page: number) {\n    if (page < this.firstPage || page > this.lastPage || this.activePage === page) return;\n    this.activePage = page;\n    this.calculateAll();\n    this.emitPageChange();\n  }\n\n  /**\n   * Go to previous page.\n   *\n   * The action will be ignored if already on the first page.\n   */\n  goToPreviousPage() {\n    if (this.activePage <= this.firstPage) return;\n    this.activePage -= 1;\n    this.calculateAll();\n    this.emitPageChange();\n  }\n\n  /**\n   * Go to next page.\n   *\n   * The action will be ignored if already on the last page.\n   */\n  goToNextPage() {\n    if (this.activePage >= this.lastPage) return;\n    this.activePage += 1;\n    this.calculateAll();\n    this.emitPageChange();\n  }\n\n  /**\n   * @internal\n   */\n  protected changeItemsPerPage(value: number) {\n    this.itemsPerPage = value;\n    this.cdr.markForCheck();\n    this.calculateAll();\n    this.emitPageChange();\n  }\n\n  private calculateAll() {\n    this.calculateValues();\n    this.calculatePages();\n    this.triggerChange.emit();\n  }\n\n  private calculateValues() {\n    this.totalPages = Math.ceil(this.totalData / this.itemsPerPage);\n    this.firstPage = this.totalPages >= 1 ? 1 : 0;\n    this.lastPage = this.totalPages >= 1 ? this.totalPages : 0;\n    if (this.totalPages && this.activePage < 1) this.activePage = 1;\n    if (this.totalPages && this.activePage > this.totalPages) this.activePage = this.totalPages;\n  }\n\n  private calculatePages() {\n    const totalPagesToShowInMiddle = this.totalPagesToShow - 4;\n    const offsetPagesToShow = totalPagesToShowInMiddle + 1;\n    const totalPagesToShowOnSide = offsetPagesToShow + 1;\n\n    // Handle pages if `totalPages` is less than or equal to `totalPagesToShow`, then no need ellipsis.\n    if (this.totalPages <= this.totalPagesToShow) {\n      this.pages = Array.from(Array(this.totalPages)).map((_, idx) => idx + 1);\n      return;\n    }\n\n    // Handle pages with ellipsis before the last page.\n    if (this.activePage <= offsetPagesToShow) {\n      const numbers = Array.from(Array(totalPagesToShowOnSide)).map((_, idx) => idx + 1);\n      this.pages = [...numbers, '...', this.lastPage];\n      return;\n    }\n\n    // Handle pages with ellipsis after the first page.\n    if (this.totalPages - this.activePage < offsetPagesToShow) {\n      const numbers = Array.from(Array(totalPagesToShowOnSide))\n        .map((_, idx) => this.totalPages - idx)\n        .reverse();\n      this.pages = [this.firstPage, '...', ...numbers];\n      return;\n    }\n\n    // Handle pages with ellipsis after the first page and before the last page.\n    const startNumber = this.activePage - (totalPagesToShowInMiddle - 2);\n    // const endNumber = this.activePage + (totalPagesToShowInMiddle - 2);\n    const numbers = Array.from(Array(totalPagesToShowInMiddle)).map((_, idx) => startNumber + idx);\n    this.pages = [this.firstPage, '...', ...numbers, '...', this.lastPage];\n  }\n\n  private emitPageChange() {\n    this.pageChange.emit({\n      activePage: this.activePage,\n      itemsPerPage: this.itemsPerPage,\n    });\n  }\n}\n","<div *ngIf=\"totalData\" class=\"d-pagination\">\n  <div\n    class=\"d-pagination-prev\"\n    [class.disabled]=\"activePage <= firstPage\"\n    (click)=\"goToPreviousPage()\"\n  >\n    <ng-template *ngTemplateOutlet=\"iconPrevious\"></ng-template>\n  </div>\n\n  <div *ngFor=\"let page of pages\">\n    <div *ngIf=\"page === '...'; else pageNumber\" class=\"d-pagination-page-ellipsis\">...</div>\n    <ng-template #pageNumber>\n      <div\n        class=\"d-pagination-page-number\"\n        [class.active]=\"page === activePage\"\n        (click)=\"navigateToPage($any(page))\"\n      >\n        {{ page }}\n      </div>\n    </ng-template>\n  </div>\n\n  <div class=\"d-pagination-next\" [class.disabled]=\"activePage >= lastPage\" (click)=\"goToNextPage()\">\n    <ng-template *ngTemplateOutlet=\"iconNext\"></ng-template>\n  </div>\n</div>\n\n<ng-template #iconPrevious>\n  <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"42\" height=\"46\" viewBox=\"0 0 42 46\" fill=\"none\">\n    <path\n      d=\"M23.5252 27.6092L18.8851 22.2096L23.5252 16.81C23.9916 16.2673 23.9916 15.3905 23.5252 14.8478C23.3018 14.5872 22.9984 14.4408 22.6821 14.4408C22.3658 14.4408 22.0624 14.5872 21.839 14.8478L16.3498 21.2354C15.8834 21.7782 15.8834 22.6549 16.3498 23.1977L21.839 29.5853C22.3054 30.128 23.0588 30.128 23.5252 29.5853C23.9796 29.0426 23.9916 28.1519 23.5252 27.6092Z\"\n      fill=\"currentColor\"\n    />\n  </svg>\n</ng-template>\n\n<ng-template #iconNext>\n  <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"42\" height=\"46\" viewBox=\"0 0 42 46\" fill=\"none\">\n    <path\n      d=\"M18.4748 27.6092L23.1149 22.2096L18.4748 16.81C18.0084 16.2673 18.0084 15.3905 18.4748 14.8478C18.6982 14.5872 19.0016 14.4408 19.3179 14.4408C19.6342 14.4408 19.9376 14.5872 20.161 14.8478L25.6502 21.2354C26.1166 21.7782 26.1166 22.6549 25.6502 23.1977L20.161 29.5853C19.6946 30.128 18.9412 30.128 18.4748 29.5853C18.0204 29.0426 18.0084 28.1519 18.4748 27.6092Z\"\n      fill=\"currentColor\"\n    />\n  </svg>\n</ng-template>\n"]}