@verisoft/core 18.6.0 → 18.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.eslintrc.json +43 -0
- package/README.md +52 -3
- package/jest.config.ts +22 -0
- package/ng-package.json +7 -0
- package/package.json +14 -29
- package/project.json +36 -0
- package/{index.d.ts → src/index.ts} +3 -3
- package/src/lib/index.ts +1 -0
- package/{lib/models/all-item.datasource.d.ts → src/lib/models/all-item.datasource.ts} +5 -4
- package/src/lib/models/base-http.models.ts +144 -0
- package/src/lib/models/constants.ts +8 -0
- package/src/lib/models/datasource.model.ts +77 -0
- package/{lib/models/environment.model.d.ts → src/lib/models/environment.model.ts} +5 -5
- package/src/lib/models/error-provider.model.ts +20 -0
- package/{lib/models/event.models.d.ts → src/lib/models/event.models.ts} +7 -5
- package/{lib/models/index.d.ts → src/lib/models/index.ts} +7 -7
- package/src/lib/services/base-http.service.ts +114 -0
- package/src/lib/services/error-provider.service.ts +29 -0
- package/{lib/services/index.d.ts → src/lib/services/index.ts} +3 -3
- package/src/lib/services/local-storage.service.ts +13 -0
- package/src/lib/services/storage.service.ts +13 -0
- package/src/lib/utils/array.utils.spec.ts +49 -0
- package/src/lib/utils/array.utils.ts +54 -0
- package/src/lib/utils/clear.utils.ts +53 -0
- package/src/lib/utils/data.utils.ts +34 -0
- package/src/lib/utils/date.utils.ts +30 -0
- package/{lib/utils/index.d.ts → src/lib/utils/index.ts} +6 -6
- package/src/lib/utils/keyOrFn.utils.ts +15 -0
- package/src/lib/utils/object.utils.spec.ts +69 -0
- package/src/lib/utils/object.utils.ts +47 -0
- package/src/test-setup.ts +8 -0
- package/tsconfig.json +29 -0
- package/tsconfig.lib.json +20 -0
- package/tsconfig.lib.prod.json +9 -0
- package/tsconfig.spec.json +16 -0
- package/esm2022/index.mjs +0 -4
- package/esm2022/lib/models/all-item.datasource.mjs +0 -2
- package/esm2022/lib/models/base-http.models.mjs +0 -83
- package/esm2022/lib/models/constants.mjs +0 -6
- package/esm2022/lib/models/datasource.model.mjs +0 -44
- package/esm2022/lib/models/environment.model.mjs +0 -2
- package/esm2022/lib/models/error-provider.model.mjs +0 -12
- package/esm2022/lib/models/event.models.mjs +0 -2
- package/esm2022/lib/models/index.mjs +0 -8
- package/esm2022/lib/services/base-http.service.mjs +0 -86
- package/esm2022/lib/services/error-provider.service.mjs +0 -34
- package/esm2022/lib/services/index.mjs +0 -4
- package/esm2022/lib/services/local-storage.service.mjs +0 -12
- package/esm2022/lib/utils/array.utils.mjs +0 -43
- package/esm2022/lib/utils/clear.utils.mjs +0 -35
- package/esm2022/lib/utils/data.utils.mjs +0 -32
- package/esm2022/lib/utils/date.utils.mjs +0 -26
- package/esm2022/lib/utils/index.mjs +0 -7
- package/esm2022/lib/utils/keyOrFn.utils.mjs +0 -16
- package/esm2022/lib/utils/object.utils.mjs +0 -35
- package/esm2022/verisoft-core.mjs +0 -5
- package/fesm2022/verisoft-core.mjs +0 -462
- package/fesm2022/verisoft-core.mjs.map +0 -1
- package/lib/models/base-http.models.d.ts +0 -41
- package/lib/models/constants.d.ts +0 -3
- package/lib/models/datasource.model.d.ts +0 -8
- package/lib/models/error-provider.model.d.ts +0 -11
- package/lib/services/base-http.service.d.ts +0 -21
- package/lib/services/error-provider.service.d.ts +0 -12
- package/lib/services/local-storage.service.d.ts +0 -5
- package/lib/utils/array.utils.d.ts +0 -5
- package/lib/utils/clear.utils.d.ts +0 -7
- package/lib/utils/data.utils.d.ts +0 -3
- package/lib/utils/date.utils.d.ts +0 -1
- package/lib/utils/keyOrFn.utils.d.ts +0 -1
- package/lib/utils/object.utils.d.ts +0 -2
package/.eslintrc.json
ADDED
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
{
|
|
2
|
+
"extends": ["../../../.eslintrc.base.json"],
|
|
3
|
+
"ignorePatterns": ["!**/*"],
|
|
4
|
+
"overrides": [
|
|
5
|
+
{
|
|
6
|
+
"files": ["*.ts"],
|
|
7
|
+
"extends": [
|
|
8
|
+
"plugin:@nx/angular",
|
|
9
|
+
"plugin:@angular-eslint/template/process-inline-templates"
|
|
10
|
+
],
|
|
11
|
+
"rules": {
|
|
12
|
+
"@angular-eslint/directive-selector": [
|
|
13
|
+
"error",
|
|
14
|
+
{
|
|
15
|
+
"type": "attribute",
|
|
16
|
+
"prefix": "v",
|
|
17
|
+
"style": "camelCase"
|
|
18
|
+
}
|
|
19
|
+
],
|
|
20
|
+
"@angular-eslint/component-selector": [
|
|
21
|
+
"error",
|
|
22
|
+
{
|
|
23
|
+
"type": "element",
|
|
24
|
+
"prefix": "v",
|
|
25
|
+
"style": "kebab-case"
|
|
26
|
+
}
|
|
27
|
+
]
|
|
28
|
+
}
|
|
29
|
+
},
|
|
30
|
+
{
|
|
31
|
+
"files": ["*.html"],
|
|
32
|
+
"extends": ["plugin:@nx/angular-template"],
|
|
33
|
+
"rules": {}
|
|
34
|
+
},
|
|
35
|
+
{
|
|
36
|
+
"files": ["*.json"],
|
|
37
|
+
"parser": "jsonc-eslint-parser",
|
|
38
|
+
"rules": {
|
|
39
|
+
"@nx/dependency-checks": "error"
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
]
|
|
43
|
+
}
|
package/README.md
CHANGED
|
@@ -1,7 +1,56 @@
|
|
|
1
1
|
# core
|
|
2
2
|
|
|
3
|
-
|
|
3
|
+
The `core` library provides shared utilities, services, and base functionality for other libraries and applications in the Verisoft Frontend workspace.
|
|
4
4
|
|
|
5
|
-
##
|
|
5
|
+
## Features
|
|
6
6
|
|
|
7
|
-
|
|
7
|
+
- Common Angular services and utilities
|
|
8
|
+
- Shared interfaces and types
|
|
9
|
+
- Reusable logic for other libraries
|
|
10
|
+
|
|
11
|
+
## Installation
|
|
12
|
+
|
|
13
|
+
This package is intended for internal use within the monorepo. To use it in another library or app, add it as a dependency in your `project.json`:
|
|
14
|
+
|
|
15
|
+
```json
|
|
16
|
+
"dependencies": {
|
|
17
|
+
"core": "*"
|
|
18
|
+
}
|
|
19
|
+
```
|
|
20
|
+
|
|
21
|
+
## Usage
|
|
22
|
+
|
|
23
|
+
Import the required modules, services, or utilities from `core`:
|
|
24
|
+
|
|
25
|
+
```typescript
|
|
26
|
+
import { SomeService } from '@verisoft/core';
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
## Development
|
|
30
|
+
|
|
31
|
+
### Building
|
|
32
|
+
|
|
33
|
+
Run the following command to build the library:
|
|
34
|
+
|
|
35
|
+
```sh
|
|
36
|
+
nx build core
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
### Running Unit Tests
|
|
40
|
+
|
|
41
|
+
To execute the unit tests for this library:
|
|
42
|
+
|
|
43
|
+
```sh
|
|
44
|
+
nx test core
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
## Contributing
|
|
48
|
+
|
|
49
|
+
1. Fork the repository and create your branch from `master`.
|
|
50
|
+
2. Make your changes and add tests if needed.
|
|
51
|
+
3. Run `nx test core` to ensure all tests pass.
|
|
52
|
+
4. Submit a pull request.
|
|
53
|
+
|
|
54
|
+
## License
|
|
55
|
+
|
|
56
|
+
This project is licensed under the MIT License.
|
package/jest.config.ts
ADDED
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/* eslint-disable */
|
|
2
|
+
export default {
|
|
3
|
+
displayName: 'core',
|
|
4
|
+
preset: '../../../jest.preset.js',
|
|
5
|
+
setupFilesAfterEnv: ['<rootDir>/src/test-setup.ts'],
|
|
6
|
+
coverageDirectory: '../../../coverage/src/libs/core',
|
|
7
|
+
transform: {
|
|
8
|
+
'^.+\\.(ts|mjs|js|html)$': [
|
|
9
|
+
'jest-preset-angular',
|
|
10
|
+
{
|
|
11
|
+
tsconfig: '<rootDir>/tsconfig.spec.json',
|
|
12
|
+
stringifyContentPathRegex: '\\.(html|svg)$',
|
|
13
|
+
},
|
|
14
|
+
],
|
|
15
|
+
},
|
|
16
|
+
transformIgnorePatterns: ['node_modules/(?!.*\\.mjs$)'],
|
|
17
|
+
snapshotSerializers: [
|
|
18
|
+
'jest-preset-angular/build/serializers/no-ng-attributes',
|
|
19
|
+
'jest-preset-angular/build/serializers/ng-snapshot',
|
|
20
|
+
'jest-preset-angular/build/serializers/html-comment',
|
|
21
|
+
],
|
|
22
|
+
};
|
package/ng-package.json
ADDED
package/package.json
CHANGED
|
@@ -1,29 +1,14 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "@verisoft/core",
|
|
3
|
-
"version": "18.
|
|
4
|
-
"peerDependencies": {
|
|
5
|
-
"@angular/common": "^18.2.8",
|
|
6
|
-
"@angular/core": "^18.2.8",
|
|
7
|
-
"moment": "^2.30.1",
|
|
8
|
-
"rxjs": "~7.8.0",
|
|
9
|
-
"@angular/forms": "18.2.8",
|
|
10
|
-
"@ngx-translate/core": "^15.0.0"
|
|
11
|
-
},
|
|
12
|
-
"sideEffects": false,
|
|
13
|
-
"dependencies": {
|
|
14
|
-
|
|
15
|
-
},
|
|
16
|
-
"module": "fesm2022/verisoft-core.mjs",
|
|
17
|
-
"typings": "index.d.ts",
|
|
18
|
-
"exports": {
|
|
19
|
-
"./package.json": {
|
|
20
|
-
"default": "./package.json"
|
|
21
|
-
},
|
|
22
|
-
".": {
|
|
23
|
-
"types": "./index.d.ts",
|
|
24
|
-
"esm2022": "./esm2022/verisoft-core.mjs",
|
|
25
|
-
"esm": "./esm2022/verisoft-core.mjs",
|
|
26
|
-
"default": "./fesm2022/verisoft-core.mjs"
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
}
|
|
1
|
+
{
|
|
2
|
+
"name": "@verisoft/core",
|
|
3
|
+
"version": "18.7.0",
|
|
4
|
+
"peerDependencies": {
|
|
5
|
+
"@angular/common": "^18.2.8",
|
|
6
|
+
"@angular/core": "^18.2.8",
|
|
7
|
+
"moment": "^2.30.1",
|
|
8
|
+
"rxjs": "~7.8.0",
|
|
9
|
+
"@angular/forms": "18.2.8",
|
|
10
|
+
"@ngx-translate/core": "^15.0.0"
|
|
11
|
+
},
|
|
12
|
+
"sideEffects": false,
|
|
13
|
+
"dependencies": {}
|
|
14
|
+
}
|
package/project.json
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "core",
|
|
3
|
+
"$schema": "../../../node_modules/nx/schemas/project-schema.json",
|
|
4
|
+
"sourceRoot": "src/libs/core/src",
|
|
5
|
+
"prefix": "lib",
|
|
6
|
+
"projectType": "library",
|
|
7
|
+
"tags": [],
|
|
8
|
+
"targets": {
|
|
9
|
+
"build": {
|
|
10
|
+
"executor": "@nx/angular:package",
|
|
11
|
+
"outputs": ["{workspaceRoot}/dist/{projectRoot}"],
|
|
12
|
+
"options": {
|
|
13
|
+
"project": "src/libs/core/ng-package.json"
|
|
14
|
+
},
|
|
15
|
+
"configurations": {
|
|
16
|
+
"production": {
|
|
17
|
+
"tsConfig": "src/libs/core/tsconfig.lib.prod.json"
|
|
18
|
+
},
|
|
19
|
+
"development": {
|
|
20
|
+
"tsConfig": "src/libs/core/tsconfig.lib.json"
|
|
21
|
+
}
|
|
22
|
+
},
|
|
23
|
+
"defaultConfiguration": "production"
|
|
24
|
+
},
|
|
25
|
+
"test": {
|
|
26
|
+
"executor": "@nx/jest:jest",
|
|
27
|
+
"outputs": ["{workspaceRoot}/coverage/{projectRoot}"],
|
|
28
|
+
"options": {
|
|
29
|
+
"jestConfig": "src/libs/core/jest.config.ts"
|
|
30
|
+
}
|
|
31
|
+
},
|
|
32
|
+
"lint": {
|
|
33
|
+
"executor": "@nx/eslint:lint"
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
export * from './lib/models';
|
|
2
|
-
export * from './lib/services';
|
|
3
|
-
export * from './lib/utils';
|
|
1
|
+
export * from './lib/models';
|
|
2
|
+
export * from './lib/services';
|
|
3
|
+
export * from './lib/utils';
|
package/src/lib/index.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './models';
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
import { Observable } from 'rxjs';
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
1
|
+
import { Observable } from 'rxjs';
|
|
2
|
+
|
|
3
|
+
export interface AllItemDatasource<T> {
|
|
4
|
+
getData$: () => Observable<T[]>
|
|
5
|
+
}
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
2
|
+
import { HttpParams, HttpResponse } from '@angular/common/http';
|
|
3
|
+
import { InjectionToken } from '@angular/core';
|
|
4
|
+
|
|
5
|
+
export const BASE_URL_PATH = new InjectionToken<string>('BASE_URL_PATH');
|
|
6
|
+
|
|
7
|
+
export function requestParamsToHttpParams<T>(
|
|
8
|
+
requestParams: Partial<RequestParams<T>>,
|
|
9
|
+
httpParams: HttpParams = new HttpParams()
|
|
10
|
+
): HttpParams {
|
|
11
|
+
if (requestParams.limit != undefined) {
|
|
12
|
+
httpParams = httpParams.append('limit', requestParams.limit.toString());
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
if (requestParams.offset != undefined) {
|
|
16
|
+
httpParams = httpParams.append('offset', requestParams.offset.toString());
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
if (requestParams.id != '' && requestParams.id != undefined) {
|
|
20
|
+
httpParams = httpParams.append('id', requestParams.id as any);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
httpParams = getFilter(requestParams, httpParams);
|
|
24
|
+
httpParams = getSort(requestParams, httpParams);
|
|
25
|
+
|
|
26
|
+
return httpParams;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function getFilter<T>(
|
|
30
|
+
requestParams: Partial<RequestParams<any>>,
|
|
31
|
+
httpParams: HttpParams
|
|
32
|
+
): HttpParams {
|
|
33
|
+
if (!requestParams.filter) {
|
|
34
|
+
return httpParams;
|
|
35
|
+
}
|
|
36
|
+
Object.keys(requestParams?.filter).forEach((key) => {
|
|
37
|
+
const value = requestParams.filter?.[key as keyof Partial<T>];
|
|
38
|
+
if (value != undefined && !(typeof value === 'string' && value.trim() === "")) {
|
|
39
|
+
if (Array.isArray(value)) {
|
|
40
|
+
value.forEach((valueItem: any) => {
|
|
41
|
+
httpParams = httpParams.append(
|
|
42
|
+
'Filter.' + key, valueItem
|
|
43
|
+
);
|
|
44
|
+
});
|
|
45
|
+
} else {
|
|
46
|
+
httpParams = httpParams.append('Filter.' + key, value);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
return httpParams;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
function getSort(
|
|
54
|
+
requestParams: Partial<RequestParams<any>>,
|
|
55
|
+
httpParams: HttpParams
|
|
56
|
+
): HttpParams {
|
|
57
|
+
if (!requestParams.sort) {
|
|
58
|
+
return httpParams;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
requestParams.sort?.forEach((sort) => {
|
|
62
|
+
httpParams = httpParams
|
|
63
|
+
.append('Sort.Field', sort.field)
|
|
64
|
+
.append('Sort.Direction', sort.direction);
|
|
65
|
+
});
|
|
66
|
+
return httpParams;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
export function saveFile(response: HttpResponse<Blob>) {
|
|
70
|
+
const fileName = response.headers
|
|
71
|
+
.get('Content-Disposition')
|
|
72
|
+
?.split(';')[1]
|
|
73
|
+
.split('=')[1];
|
|
74
|
+
const blob: Blob = response.body as Blob;
|
|
75
|
+
const a = document.createElement('a');
|
|
76
|
+
a.download = fileName ?? 'undefined';
|
|
77
|
+
a.href = window.URL.createObjectURL(blob);
|
|
78
|
+
a.click();
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
export enum SortDirection {
|
|
82
|
+
asc = 'asc',
|
|
83
|
+
desc = 'desc',
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
export declare type SortDirectionType = keyof typeof SortDirection;
|
|
87
|
+
|
|
88
|
+
export declare interface Sort {
|
|
89
|
+
field: string;
|
|
90
|
+
direction: SortDirectionType;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
export interface RequestParams<T> extends AllDataRequestParams<T> {
|
|
94
|
+
offset: number;
|
|
95
|
+
limit: number;
|
|
96
|
+
id?: string;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
export interface AllDataRequestParams<T> {
|
|
100
|
+
filter?: Partial<T>;
|
|
101
|
+
sort?: Sort[];
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
export interface Page<T> {
|
|
105
|
+
data: T[];
|
|
106
|
+
total: number;
|
|
107
|
+
limit: number;
|
|
108
|
+
offset: number;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export interface CustomExport<T> {
|
|
112
|
+
sortDefinition?: Sort;
|
|
113
|
+
filter?: Partial<T>;
|
|
114
|
+
columnsToExport: ColumnExportSpecification[];
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
export interface ColumnExportSpecification {
|
|
118
|
+
name: string;
|
|
119
|
+
header: string;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
export const DEFAULT_SEARCH_LIMIT = 50;
|
|
123
|
+
|
|
124
|
+
export const DEFAULT_SEARCH_PARAMS: RequestParams<any> = {
|
|
125
|
+
offset: 0,
|
|
126
|
+
limit: DEFAULT_SEARCH_LIMIT,
|
|
127
|
+
id: '',
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
export function normalizeRequest<T>(
|
|
131
|
+
request: Partial<RequestParams<T>>,
|
|
132
|
+
minLimit: number | undefined = undefined
|
|
133
|
+
): RequestParams<T> {
|
|
134
|
+
return {
|
|
135
|
+
offset: request?.offset ?? 0,
|
|
136
|
+
limit: !request?.limit
|
|
137
|
+
? DEFAULT_SEARCH_LIMIT
|
|
138
|
+
: minLimit && request.limit < minLimit
|
|
139
|
+
? minLimit
|
|
140
|
+
: request.limit,
|
|
141
|
+
filter: request?.filter,
|
|
142
|
+
sort: request?.sort,
|
|
143
|
+
};
|
|
144
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import { HttpClient } from '@angular/common/http';
|
|
2
|
+
import { isObservable, map, Observable, of } from 'rxjs';
|
|
3
|
+
import { BaseHttpService } from '../services/base-http.service';
|
|
4
|
+
import { AllItemDatasource } from './all-item.datasource';
|
|
5
|
+
import { Page, RequestParams } from './base-http.models';
|
|
6
|
+
import { DEFAULT_PAGE_SIZE } from './constants';
|
|
7
|
+
|
|
8
|
+
export type DatasourceType<T> =
|
|
9
|
+
| string
|
|
10
|
+
| Observable<T[]>
|
|
11
|
+
| Observable<Page<T>>
|
|
12
|
+
| BaseHttpService<T>
|
|
13
|
+
| T[]
|
|
14
|
+
| Page<T>
|
|
15
|
+
| AllItemDatasource<T>;
|
|
16
|
+
|
|
17
|
+
export type DataSourceFunctionType<T> = (
|
|
18
|
+
requestParams: RequestParams<T>
|
|
19
|
+
) => Observable<Page<T>>;
|
|
20
|
+
|
|
21
|
+
export function convertDatasource<T>(
|
|
22
|
+
datasource: DatasourceType<T>,
|
|
23
|
+
basePath: string,
|
|
24
|
+
httpClient: HttpClient
|
|
25
|
+
): DataSourceFunctionType<T> {
|
|
26
|
+
if (!datasource) {
|
|
27
|
+
throw new Error('Datasource is not defined');
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
if (typeof datasource === 'string') {
|
|
31
|
+
const service = new BaseHttpService<T>(httpClient, basePath, datasource);
|
|
32
|
+
return (requestParams: RequestParams<T>) =>
|
|
33
|
+
service.fetchList(requestParams);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (datasource instanceof BaseHttpService) {
|
|
37
|
+
return (requestParams: RequestParams<T>) =>
|
|
38
|
+
datasource.fetchList(requestParams);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
if (isObservable(datasource)) {
|
|
42
|
+
return () =>
|
|
43
|
+
(datasource as Observable<Page<T> | T[]>).pipe(map(convertArrayToPage));
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
const allItemDatasource = <AllItemDatasource<T>>datasource;
|
|
47
|
+
if (allItemDatasource.getData$) {
|
|
48
|
+
return () =>
|
|
49
|
+
allItemDatasource
|
|
50
|
+
.getData$()
|
|
51
|
+
.pipe(map((data) => convertArrayToPage(data)));
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const page = <Page<T>>datasource;
|
|
55
|
+
if (page.data) {
|
|
56
|
+
return () => of(page);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (Array.isArray(datasource)) {
|
|
60
|
+
return () => of(convertArrayToPage(datasource));
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
throw new Error('Datasource is not supported');
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
function convertArrayToPage<T>(data: Page<T> | T[]): Page<T> {
|
|
67
|
+
if (Array.isArray(data)) {
|
|
68
|
+
return {
|
|
69
|
+
data: data ?? [],
|
|
70
|
+
total: data?.length ?? 0,
|
|
71
|
+
limit: data?.length ?? DEFAULT_PAGE_SIZE,
|
|
72
|
+
offset: 0,
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
return data as Page<T>;
|
|
77
|
+
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
export interface EnvironmentConfig {
|
|
2
|
-
production: boolean;
|
|
3
|
-
apiUrl: string;
|
|
4
|
-
appVersion: string;
|
|
5
|
-
}
|
|
1
|
+
export interface EnvironmentConfig {
|
|
2
|
+
production: boolean;
|
|
3
|
+
apiUrl: string;
|
|
4
|
+
appVersion: string;
|
|
5
|
+
}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
import { ValidationErrors } from '@angular/forms';
|
|
3
|
+
import { Observable } from 'rxjs';
|
|
4
|
+
|
|
5
|
+
export interface ErrorProvider {
|
|
6
|
+
mapError(errors: ValidationErrors): Observable<string>;
|
|
7
|
+
errors: Record<string, (value?: any) => string>;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export const ERROR_PROVIDER_TOKEN
|
|
11
|
+
= new InjectionToken<ErrorProvider>('ERROR_PROVIDER_TOKEN');
|
|
12
|
+
|
|
13
|
+
export enum CustomValidationCodes {}
|
|
14
|
+
|
|
15
|
+
export const ErrorMap: Record<string, (value?: any) => string> = {
|
|
16
|
+
required: () => `VALIDATIONS.REQUIRED`,
|
|
17
|
+
email: () => `VALIDATIONS.EMAIL`,
|
|
18
|
+
minLength: () => 'VALIDATIONS.MIN_LENGTH',
|
|
19
|
+
maxLength: () => 'VALIDATIONS.MAX_LENGTH',
|
|
20
|
+
};
|
|
@@ -1,5 +1,7 @@
|
|
|
1
|
-
import { RequestParams } from "./base-http.models";
|
|
2
|
-
|
|
3
|
-
export
|
|
4
|
-
|
|
5
|
-
|
|
1
|
+
import { RequestParams } from "./base-http.models";
|
|
2
|
+
|
|
3
|
+
export type LazyLoadEvent = RequestParams<unknown>;
|
|
4
|
+
|
|
5
|
+
export interface FilterEvent {
|
|
6
|
+
filter?: string;
|
|
7
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
export * from './environment.model';
|
|
2
|
-
export * from './event.models';
|
|
3
|
-
export * from './constants';
|
|
4
|
-
export * from './base-http.models';
|
|
5
|
-
export * from './all-item.datasource';
|
|
6
|
-
export * from './datasource.model';
|
|
7
|
-
export * from './error-provider.model';
|
|
1
|
+
export * from './environment.model';
|
|
2
|
+
export * from './event.models';
|
|
3
|
+
export * from './constants';
|
|
4
|
+
export * from './base-http.models';
|
|
5
|
+
export * from './all-item.datasource';
|
|
6
|
+
export * from './datasource.model';
|
|
7
|
+
export * from './error-provider.model';
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
2
|
+
import { HttpClient } from '@angular/common/http';
|
|
3
|
+
import { map, Observable } from 'rxjs';
|
|
4
|
+
import {
|
|
5
|
+
AllDataRequestParams,
|
|
6
|
+
DEFAULT_SEARCH_PARAMS,
|
|
7
|
+
Page,
|
|
8
|
+
RequestParams,
|
|
9
|
+
requestParamsToHttpParams,
|
|
10
|
+
} from '../models/base-http.models';
|
|
11
|
+
import { LazyLoadEvent } from '../models/event.models';
|
|
12
|
+
import { ClearUtils } from '../utils/clear.utils';
|
|
13
|
+
|
|
14
|
+
export class BaseHttpService<T, TKey = number | string> {
|
|
15
|
+
constructor(
|
|
16
|
+
readonly http: HttpClient,
|
|
17
|
+
readonly basePath: string,
|
|
18
|
+
readonly entityName: string
|
|
19
|
+
) {
|
|
20
|
+
this.basePath = basePath;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
protected get apiPath() {
|
|
24
|
+
return this.basePath ? this.basePath + this.entityName : this.entityName;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
fetchList(requestParams: RequestParams<T>): Observable<Page<T>> {
|
|
28
|
+
const params = requestParamsToHttpParams<T>(requestParams);
|
|
29
|
+
return this.http.get<Page<T>>(this.apiPath, { params }).pipe(
|
|
30
|
+
map((response: Page<T>) => {
|
|
31
|
+
response.limit = requestParams.limit;
|
|
32
|
+
return response;
|
|
33
|
+
})
|
|
34
|
+
);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
getData$(
|
|
38
|
+
entityName: keyof T,
|
|
39
|
+
searchTerm?: string | undefined,
|
|
40
|
+
lazyLoad?: LazyLoadEvent
|
|
41
|
+
): Observable<T[]> {
|
|
42
|
+
const filter: Partial<T> = { [entityName]: searchTerm } as Partial<T>;
|
|
43
|
+
const params = lazyLoad
|
|
44
|
+
? requestParamsToHttpParams<T>({
|
|
45
|
+
...DEFAULT_SEARCH_PARAMS,
|
|
46
|
+
offset: lazyLoad.offset,
|
|
47
|
+
limit: lazyLoad.limit,
|
|
48
|
+
filter,
|
|
49
|
+
})
|
|
50
|
+
: requestParamsToHttpParams<T>({
|
|
51
|
+
...DEFAULT_SEARCH_PARAMS,
|
|
52
|
+
filter,
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
return this.http.get<T[]>(this.apiPath, { params });
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
get(id: TKey): Observable<T> {
|
|
59
|
+
const url = `${this.apiPath}/${id}`;
|
|
60
|
+
return this.http.get<T>(url);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
post(entity: Partial<T>): Observable<T> {
|
|
64
|
+
return this.http.post<T>(this.apiPath, entity);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
put(id: TKey, entity: Partial<T>): Observable<T> {
|
|
68
|
+
const url = `${this.apiPath}/${id}`;
|
|
69
|
+
return this.http.put<T>(url, entity);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
delete(id: TKey): Observable<T> {
|
|
73
|
+
const url = `${this.apiPath}/${id}`;
|
|
74
|
+
return this.http.delete<T>(url);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
export(requestParams: AllDataRequestParams<T>): Observable<Blob> {
|
|
78
|
+
const httpParams = requestParamsToHttpParams<T>(requestParams);
|
|
79
|
+
const url = `${this.apiPath}/export`;
|
|
80
|
+
return this.http.get(url, { responseType: 'blob', params: httpParams });
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
removeRange(entity: string[]) {
|
|
84
|
+
return this.http.delete(this.apiPath, {
|
|
85
|
+
body: entity,
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
createParams<T>(requestParams: RequestParams<T>): any {
|
|
90
|
+
const sorter = this.createSorter(requestParams);
|
|
91
|
+
const filter = JSON.parse(JSON.stringify(requestParams.filter));
|
|
92
|
+
const transformedFilter = filter;
|
|
93
|
+
ClearUtils.recursiveObjectAttributesDeletation(transformedFilter);
|
|
94
|
+
return {
|
|
95
|
+
paging: {
|
|
96
|
+
offset: requestParams.offset,
|
|
97
|
+
limit: requestParams.limit,
|
|
98
|
+
},
|
|
99
|
+
...(sorter && { sorting: sorter }),
|
|
100
|
+
...(transformedFilter && { filter: transformedFilter }),
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
private createSorter<T>(requestParams: RequestParams<T>) {
|
|
105
|
+
if (requestParams?.sort?.length === 1) {
|
|
106
|
+
const sorter = requestParams.sort[0];
|
|
107
|
+
return {
|
|
108
|
+
sortField: sorter.field,
|
|
109
|
+
sortOrder: sorter.direction,
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
return null;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { inject, Injectable } from '@angular/core';
|
|
2
|
+
import { ValidationErrors } from '@angular/forms';
|
|
3
|
+
import { TranslateService } from '@ngx-translate/core';
|
|
4
|
+
import { Observable } from 'rxjs';
|
|
5
|
+
import { ErrorProvider, ErrorMap } from '../models/error-provider.model';
|
|
6
|
+
|
|
7
|
+
@Injectable({
|
|
8
|
+
providedIn: 'root'
|
|
9
|
+
})
|
|
10
|
+
export class LocalizableErrorProviderService implements ErrorProvider {
|
|
11
|
+
private readonly translateService = inject(TranslateService);
|
|
12
|
+
readonly errors: Record<string, (value?: any) => string> = ErrorMap;
|
|
13
|
+
|
|
14
|
+
mapError(errors: ValidationErrors): Observable<string> {
|
|
15
|
+
const error = this.getFirstValue(errors);
|
|
16
|
+
try {
|
|
17
|
+
const value = this.errors[error.key](error.value);
|
|
18
|
+
return this.translateService.get(value);
|
|
19
|
+
} catch (_) {
|
|
20
|
+
throw new Error(`Key "${error.key}" is not specified in the current error map!`);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
private getFirstValue(errors: ValidationErrors): { key: string; value: string } {
|
|
25
|
+
const key = Object.keys(errors)[0];
|
|
26
|
+
const value = errors[key];
|
|
27
|
+
return { key, value };
|
|
28
|
+
}
|
|
29
|
+
}
|
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
export * from './base-http.service';
|
|
2
|
-
export * from './local-storage.service';
|
|
3
|
-
export * from './error-provider.service'
|
|
1
|
+
export * from './base-http.service';
|
|
2
|
+
export * from './local-storage.service';
|
|
3
|
+
export * from './error-provider.service'
|