src/PaginatedResourceListing.js
/*
* BSD 3-Clause License
*
* Copyright (c) 2020, Mapcreator
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of the copyright holder nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
import Mapcreator from './Mapcreator';
import RequestParameters from './RequestParameters';
import { isParentOf } from './utils/reflection';
import { makeCancelable } from './utils/helpers';
/**
* Proxy for accessing paginated resources
*/
export default class PaginatedResourceListing {
/**
* @param {Mapcreator} api - Instance of the api
* @param {String} route - Resource route
* @param {Class<ResourceBase>} Target - Wrapper target
* @param {RequestParameters} parameters - Request parameters
* @param {Number} pageCount - Resolved page count
* @param {Number} rowCount - Resolved rowCount
* @param {Array<ResourceBase>} data - Resolved data
* @private
*/
constructor (api, route, Target, parameters, pageCount = null, rowCount = 0, data = []) {
if (!isParentOf(Mapcreator, api)) {
throw new TypeError('Expected api to be of type Mapcreator');
}
if (!isParentOf(RequestParameters, parameters)) {
parameters = new RequestParameters(parameters);
}
this._api = api;
this.route = route;
this._Target = Target;
this._parameters = parameters;
this._pageCount = pageCount;
this._rows = rowCount;
this._data = data;
}
/**
* Get api instance
* @returns {Mapcreator} - Api instance
*/
get api () {
return this._api;
}
/**
* Target route
* @returns {String} - Url
*/
get route () {
return this._route;
}
/**
* Override the target route
* @param {String} value - route
*/
set route (value) {
if (!value.startsWith('https://') && !value.startsWith('http://')) {
if (!value.startsWith('/')) {
value = `/${value}`;
}
value = `${this._api.url}${value}`;
}
this._route = value;
}
/**
* Target to wrap results in
* @returns {Class<ResourceBase>} - Target constructor
*/
get Target () {
return this._Target;
}
/**
* Request parameters
* @returns {RequestParameters} - Request parameters
*/
get parameters () {
return this._parameters;
}
/**
* Request parameters
* @param {RequestParameters} value - Request parameters
*/
set parameters (value) {
this._parameters = value;
}
/**
* Current page number
* @returns {Number} - Current page
*/
get page () {
return this.parameters.page;
}
/**
* Maximum amount of items per page
* @returns {Number} - Amount of items
*/
get perPage () {
return this.parameters.perPage;
}
/**
* Set sort direction
* @returns {Array<String>} - Sort
* @example
* const sort = ['-name', 'id']
*/
get sort () {
return this.parameters.sort;
}
/**
* Current sorting value
* @param {Array<String>} value - Sort
*/
set sort (value) {
this.parameters.sort = value;
}
/**
* Deleted items filter state
* @returns {String} value - Deleted items filter state
* @see {@link DeletedState}
*/
get deleted () {
return this.parameters.deleted;
}
/**
* Deleted items filter state
* @param {String} value - Deleted items filter state
* @see {@link DeletedState}
*/
set deleted (value) {
this.parameters.deleted = value;
}
/**
* Amount of pages available
* @returns {Number} - Page count
*/
get pageCount () {
return this._pageCount;
}
/**
* Page data
* @returns {Array<ResourceBase>} - Wrapped data
*/
get data () {
return this._data;
}
/**
* Row count
* @returns {Number} - Row count
*/
get rows () {
return this._rows;
}
/**
* Optional search query
* @default {}
* @return {Object<String, String|Array<String>>} - Query
*/
get query () {
return this.parameters.search;
}
/**
* Optional search query
* @param {Object<String, String|Array<String>>} value - Query
* @throws {TypeError}
* @default {}
* @see {@link ResourceProxy#search}
*/
set query (value) {
this.parameters.search = value;
}
/**
* Get target page
* @param {Number} page - Page number
* @param {Number} perPage - Amount of items per page (max 50)
* @returns {CancelablePromise<PaginatedResourceListing>} - Target page
* @throws {ApiError} - If the api returns errors
*/
getPage (page = this.page, perPage = this.perPage) {
const query = this.parameters.copy();
query.page = page;
query.perPage = perPage;
const glue = this.route.includes('?') ? '&' : '?';
const url = this.route + glue + query.encode();
return makeCancelable(async signal => {
const response = await this.api.ky.get(url, { signal });
const { data } = await response.json();
const rowCount = Number(response.headers.get('x-paginate-total') || data.length);
const totalPages = Number(response.headers.get('x-paginate-pages') || 1);
const parameters = this.parameters.copy();
parameters.page = page;
return new PaginatedResourceListing(
this.api, this.route, this.Target,
parameters, totalPages, rowCount,
data.map(row => new this.Target(this.api, row)),
);
});
}
/**
* If there is a next page
* @returns {boolean} - If there is a next page
*/
get hasNext () {
return this.page < this.pageCount;
}
/**
* If there is a previous page
* @returns {boolean} - If there is a previous page
*/
get hasPrevious () {
return this.page > 1;
}
/**
* Used for caching pages internally
* @returns {string} - Cache token
* @see {@link PaginatedResourceWrapper}
* @see {@link ResourceCache}
*/
get cacheToken () {
return this.parameters.token();
}
/**
* Get next page
* @returns {CancelablePromise<PaginatedResourceListing>} - Paginated resource
* @throws {ApiError} - If the api returns errors
*/
next () {
return this.getPage(this.page + 1);
}
/**
* Get previous page
* @returns {CancelablePromise<PaginatedResourceListing>} - Paginated resource
* @throws {ApiError} - If the api returns errors
*/
previous () {
return this.getPage(this.page - 1);
}
}