@rosoftlab/ionic 1.0.0-alpha-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/README.md +27 -0
- package/esm2022/lib/components/index.mjs +4 -0
- package/esm2022/lib/components/rsl-ionic-crud.component/rsl-ionic-crud.component.mjs +266 -0
- package/esm2022/lib/components/rsl-ionic-data-table/rsl-ionic-data-table.component.mjs +292 -0
- package/esm2022/lib/components/rsl-ionic-grid/rsl-ionic-grid.component.mjs +269 -0
- package/esm2022/lib/decorators/index.mjs +3 -0
- package/esm2022/lib/decorators/ionic-datatable-layout.mjs +10 -0
- package/esm2022/lib/decorators/ionic-list.decorator.mjs +9 -0
- package/esm2022/lib/interfaces/index.mjs +2 -0
- package/esm2022/lib/interfaces/ionic-list-layout-config.mjs +2 -0
- package/esm2022/lib/ionic-dialog.service.mjs +51 -0
- package/esm2022/lib/rsl-ionic-module.module.mjs +86 -0
- package/esm2022/lib/translate.extension.mjs +48 -0
- package/esm2022/lib/types/index.mjs +2 -0
- package/esm2022/lib/types/repeat/repeat-section.type.mjs +135 -0
- package/esm2022/lib/validators/index.mjs +2 -0
- package/esm2022/lib/validators/must-match.mjs +12 -0
- package/esm2022/lib/wrappers/accordion-wrapper.component.mjs +50 -0
- package/esm2022/lib/wrappers/index.mjs +3 -0
- package/esm2022/lib/wrappers/panel-wrapper.component.mjs +37 -0
- package/esm2022/lib/wrappers/wrappers.module.mjs +56 -0
- package/esm2022/public-api.mjs +12 -0
- package/esm2022/rosoftlab-ionic.mjs +5 -0
- package/fesm2022/rosoftlab-ionic.mjs +1251 -0
- package/fesm2022/rosoftlab-ionic.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/index.d.ts +3 -0
- package/lib/components/rsl-ionic-crud.component/rsl-ionic-crud.component.d.ts +69 -0
- package/lib/components/rsl-ionic-data-table/rsl-ionic-data-table.component.d.ts +66 -0
- package/lib/components/rsl-ionic-grid/rsl-ionic-grid.component.d.ts +58 -0
- package/lib/decorators/index.d.ts +2 -0
- package/lib/decorators/ionic-datatable-layout.d.ts +7 -0
- package/lib/decorators/ionic-list.decorator.d.ts +2 -0
- package/lib/interfaces/index.d.ts +1 -0
- package/lib/interfaces/ionic-list-layout-config.d.ts +8 -0
- package/lib/ionic-dialog.service.d.ts +10 -0
- package/lib/rsl-ionic-module.module.d.ts +13 -0
- package/lib/translate.extension.d.ts +17 -0
- package/lib/types/index.d.ts +1 -0
- package/lib/types/repeat/repeat-section.type.d.ts +51 -0
- package/lib/validators/index.d.ts +1 -0
- package/lib/validators/must-match.d.ts +6 -0
- package/lib/wrappers/accordion-wrapper.component.d.ts +6 -0
- package/lib/wrappers/index.d.ts +2 -0
- package/lib/wrappers/panel-wrapper.component.d.ts +6 -0
- package/lib/wrappers/wrappers.module.d.ts +13 -0
- package/package.json +31 -0
- package/public-api.d.ts +8 -0
package/README.md
ADDED
@@ -0,0 +1,27 @@
|
|
1
|
+
# RslWorkspace
|
2
|
+
|
3
|
+
This project was generated with [Angular CLI](https://github.com/angular/angular-cli) version 12.1.3.
|
4
|
+
|
5
|
+
## Development server
|
6
|
+
|
7
|
+
Run `ng serve` for a dev server. Navigate to `http://localhost:4200/`. The app will automatically reload if you change any of the source files.
|
8
|
+
|
9
|
+
## Code scaffolding
|
10
|
+
|
11
|
+
Run `ng generate component component-name` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module`.
|
12
|
+
|
13
|
+
## Build
|
14
|
+
|
15
|
+
Run `ng build` to build the project. The build artifacts will be stored in the `dist/` directory.
|
16
|
+
|
17
|
+
## Running unit tests
|
18
|
+
|
19
|
+
Run `ng test` to execute the unit tests via [Karma](https://karma-runner.github.io).
|
20
|
+
|
21
|
+
## Running end-to-end tests
|
22
|
+
|
23
|
+
Run `ng e2e` to execute the end-to-end tests via a platform of your choice. To use this command, you need to first add a package that implements end-to-end testing capabilities.
|
24
|
+
|
25
|
+
## Further help
|
26
|
+
|
27
|
+
To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.io/cli) page.
|
@@ -0,0 +1,4 @@
|
|
1
|
+
export { GenericIonicCrudComponent } from './rsl-ionic-crud.component/rsl-ionic-crud.component';
|
2
|
+
export { RslIonicDataTableComponent } from "./rsl-ionic-data-table/rsl-ionic-data-table.component";
|
3
|
+
export { RslIonicGridComponent } from './rsl-ionic-grid/rsl-ionic-grid.component';
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yb3NvZnRsYWIvaW9uaWMvc3JjL2xpYi9jb21wb25lbnRzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx5QkFBeUIsRUFBRSxNQUFNLHFEQUFxRCxDQUFBO0FBQy9GLE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLHVEQUF1RCxDQUFBO0FBQ2xHLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDJDQUEyQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgR2VuZXJpY0lvbmljQ3J1ZENvbXBvbmVudCB9IGZyb20gJy4vcnNsLWlvbmljLWNydWQuY29tcG9uZW50L3JzbC1pb25pYy1jcnVkLmNvbXBvbmVudCdcclxuZXhwb3J0IHsgUnNsSW9uaWNEYXRhVGFibGVDb21wb25lbnQgfSBmcm9tIFwiLi9yc2wtaW9uaWMtZGF0YS10YWJsZS9yc2wtaW9uaWMtZGF0YS10YWJsZS5jb21wb25lbnRcIlxyXG5leHBvcnQgeyBSc2xJb25pY0dyaWRDb21wb25lbnQgfSBmcm9tICcuL3JzbC1pb25pYy1ncmlkL3JzbC1pb25pYy1ncmlkLmNvbXBvbmVudCdcclxuIl19
|
@@ -0,0 +1,266 @@
|
|
1
|
+
import { Location } from '@angular/common';
|
2
|
+
import { Component, Injector } from '@angular/core';
|
3
|
+
import { UntypedFormBuilder, UntypedFormControl, UntypedFormGroup } from '@angular/forms';
|
4
|
+
import { ActivatedRoute, Router } from '@angular/router';
|
5
|
+
import { FormlyJsonschema } from '@ngx-formly/core/json-schema';
|
6
|
+
import { TranslateService } from '@ngx-translate/core';
|
7
|
+
import * as _ from 'lodash';
|
8
|
+
import { Observable, from } from 'rxjs';
|
9
|
+
import { IonicDialogService } from '../../ionic-dialog.service';
|
10
|
+
import { RslIonicModuleModule } from '../../rsl-ionic-module.module';
|
11
|
+
import * as i0 from "@angular/core";
|
12
|
+
import * as i1 from "@angular/forms";
|
13
|
+
import * as i2 from "@angular/router";
|
14
|
+
import * as i3 from "../../ionic-dialog.service";
|
15
|
+
import * as i4 from "@ngx-translate/core";
|
16
|
+
import * as i5 from "@angular/common";
|
17
|
+
import * as i6 from "@ngx-formly/core/json-schema";
|
18
|
+
import * as i7 from "@ionic/angular";
|
19
|
+
import * as i8 from "@ngx-formly/core";
|
20
|
+
export class GenericIonicCrudComponent {
|
21
|
+
//#endregion
|
22
|
+
constructor(fb, router, route, dialogService, translate, location, injector, formlyJsonschema) {
|
23
|
+
this.fb = fb;
|
24
|
+
this.router = router;
|
25
|
+
this.route = route;
|
26
|
+
this.dialogService = dialogService;
|
27
|
+
this.translate = translate;
|
28
|
+
this.location = location;
|
29
|
+
this.injector = injector;
|
30
|
+
this.formlyJsonschema = formlyJsonschema;
|
31
|
+
this.isLoading = true;
|
32
|
+
this.changeUrlRoute = true;
|
33
|
+
this.options = {};
|
34
|
+
const SERVICE_TOKEN = route.snapshot.data['requiredService'];
|
35
|
+
this.modelService = this.injector.get(SERVICE_TOKEN);
|
36
|
+
this.modelService = this.injector.get(SERVICE_TOKEN);
|
37
|
+
this.confirmQuestion = this.translate.instant('General.Discard.Question');
|
38
|
+
this.confirmButton = this.translate.instant('General.Discard.ConfirmButton');
|
39
|
+
this.saveMessage = this.translate.instant('General.Save.Message');
|
40
|
+
}
|
41
|
+
ionViewWillLeave() {
|
42
|
+
this.options.resetModel();
|
43
|
+
this.baseForm.reset();
|
44
|
+
}
|
45
|
+
afterSave(model) {
|
46
|
+
return new Observable((observer) => {
|
47
|
+
observer.next(model);
|
48
|
+
observer.complete();
|
49
|
+
});
|
50
|
+
}
|
51
|
+
beforeSave(model) {
|
52
|
+
return new Observable((observer) => {
|
53
|
+
observer.next(model);
|
54
|
+
observer.complete();
|
55
|
+
});
|
56
|
+
}
|
57
|
+
ngOnInit() {
|
58
|
+
this.initForm();
|
59
|
+
}
|
60
|
+
initForm(customInclude = '', newModelId = null, model = null) {
|
61
|
+
if (model === null) {
|
62
|
+
this.modelId = this.route.snapshot.paramMap.get('id') ?? newModelId;
|
63
|
+
this.isEdit = false;
|
64
|
+
if (this.modelId) {
|
65
|
+
this.modelService.get(this.modelId, customInclude).subscribe((value) => {
|
66
|
+
this.isEdit = true;
|
67
|
+
this.generateForm(value);
|
68
|
+
});
|
69
|
+
}
|
70
|
+
else {
|
71
|
+
if (this.changeUrlRoute) {
|
72
|
+
const addUrl = this.router.createUrlTree([]).toString();
|
73
|
+
this.editRoute = this.router.createUrlTree([addUrl.replace('add', 'edit')]).toString();
|
74
|
+
}
|
75
|
+
// }
|
76
|
+
this.generateForm(this.modelService.newModel());
|
77
|
+
}
|
78
|
+
}
|
79
|
+
else {
|
80
|
+
this.modelId = model.id;
|
81
|
+
this.isEdit = true;
|
82
|
+
this.generateForm(model);
|
83
|
+
}
|
84
|
+
}
|
85
|
+
generateForm(model) {
|
86
|
+
this.isLoading = false;
|
87
|
+
this.modelId = model.id;
|
88
|
+
this.model = model;
|
89
|
+
this.origModel = _.cloneDeep(model);
|
90
|
+
;
|
91
|
+
this.title = this.model.modelConfig.formTitle;
|
92
|
+
this.baseForm = new UntypedFormGroup({}); //= this.modelService.toFormGroup(this.fb, model);
|
93
|
+
this.afterFormGenerated();
|
94
|
+
this.fields = this.modelService.getFormlyFields(this.model);
|
95
|
+
}
|
96
|
+
afterFormGenerated() {
|
97
|
+
}
|
98
|
+
getFromGroup(formGroup = null) {
|
99
|
+
if (!formGroup)
|
100
|
+
return this.baseForm;
|
101
|
+
if (formGroup instanceof UntypedFormGroup)
|
102
|
+
return formGroup;
|
103
|
+
return this.baseForm.controls[formGroup];
|
104
|
+
}
|
105
|
+
validateAllFormFields(formGroup = null) {
|
106
|
+
const fg = this.getFromGroup(formGroup);
|
107
|
+
Object.keys(fg.controls).forEach(field => {
|
108
|
+
// console.log(field);
|
109
|
+
const control = fg.get(field);
|
110
|
+
if (control instanceof UntypedFormControl) {
|
111
|
+
control.markAsTouched({ onlySelf: true });
|
112
|
+
}
|
113
|
+
else if (control instanceof UntypedFormGroup) {
|
114
|
+
this.validateAllFormFields(control);
|
115
|
+
}
|
116
|
+
});
|
117
|
+
}
|
118
|
+
isFieldValid(field, formGroup = null) {
|
119
|
+
const fg = this.getFromGroup(formGroup);
|
120
|
+
const filedControl = fg.get(field);
|
121
|
+
return !filedControl.valid && filedControl.touched;
|
122
|
+
}
|
123
|
+
isFieldValidFromArray(arrayIndex, field, arrayName = 'formArray') {
|
124
|
+
const fieldControl = this.baseForm.get(arrayName).get([arrayIndex]).get(field);
|
125
|
+
return !fieldControl.valid && fieldControl.touched;
|
126
|
+
}
|
127
|
+
displayFieldCss(field) {
|
128
|
+
return {
|
129
|
+
'has-error': this.isFieldValid(field),
|
130
|
+
'has-feedback': this.isFieldValid(field)
|
131
|
+
};
|
132
|
+
}
|
133
|
+
onCancel() {
|
134
|
+
this.router.navigate([this.cancelRoute]);
|
135
|
+
}
|
136
|
+
onSave() {
|
137
|
+
this.saveModel(this.baseForm);
|
138
|
+
}
|
139
|
+
onSubmit(model) {
|
140
|
+
this.saveModel(this.baseForm);
|
141
|
+
}
|
142
|
+
saveModel(formGroup = null) {
|
143
|
+
const fg = this.getFromGroup(formGroup);
|
144
|
+
const that = this;
|
145
|
+
if (fg) {
|
146
|
+
if (fg.valid) {
|
147
|
+
this.beforeSave(this.model).subscribe(_ => {
|
148
|
+
this.modelService.save(this.model, this.modelId, this.origModel).subscribe((newModel) => {
|
149
|
+
this.model = newModel;
|
150
|
+
this.modelId = newModel.id;
|
151
|
+
if (this.editRoute) {
|
152
|
+
this.isEdit = true;
|
153
|
+
if (this.changeUrlRoute) {
|
154
|
+
const url = this.router.createUrlTree([this.editRoute, this.modelId]).toString();
|
155
|
+
this.location.replaceState(url);
|
156
|
+
}
|
157
|
+
}
|
158
|
+
this.afterSave(newModel).subscribe((val) => {
|
159
|
+
from(this.dialogService.showSaveMessage(this.saveMessage)).subscribe(d => {
|
160
|
+
this.options.updateInitialValue();
|
161
|
+
fg.markAsPristine();
|
162
|
+
});
|
163
|
+
});
|
164
|
+
}, err => {
|
165
|
+
this.serverErrors(err);
|
166
|
+
});
|
167
|
+
});
|
168
|
+
}
|
169
|
+
else {
|
170
|
+
this.validateAllFormFields(formGroup);
|
171
|
+
}
|
172
|
+
}
|
173
|
+
}
|
174
|
+
serverErrors(err) {
|
175
|
+
if (err.error) {
|
176
|
+
if (err.error.errors) {
|
177
|
+
const validationErrors = err.error.errors;
|
178
|
+
if (Array.isArray(validationErrors)) {
|
179
|
+
validationErrors.forEach(prop => {
|
180
|
+
const formControl = this.baseForm.get(this.firstCharToLowerCase(prop));
|
181
|
+
if (formControl) {
|
182
|
+
// activate the error message
|
183
|
+
formControl.setErrors({
|
184
|
+
serverError: { message: validationErrors[prop].join('\n') }
|
185
|
+
});
|
186
|
+
}
|
187
|
+
});
|
188
|
+
}
|
189
|
+
else {
|
190
|
+
const keys = Object.keys(validationErrors);
|
191
|
+
keys.forEach(prop => {
|
192
|
+
const formControl = this.baseForm.get(this.firstCharToLowerCase(prop));
|
193
|
+
if (formControl) {
|
194
|
+
// activate the error message
|
195
|
+
formControl.setErrors({
|
196
|
+
serverError: { message: validationErrors[prop].join('\n') }
|
197
|
+
});
|
198
|
+
}
|
199
|
+
});
|
200
|
+
}
|
201
|
+
}
|
202
|
+
}
|
203
|
+
}
|
204
|
+
firstCharToLowerCase(str) {
|
205
|
+
if (str.length === 0) {
|
206
|
+
return str; // Return an empty string if the input is empty
|
207
|
+
}
|
208
|
+
const firstChar = str.charAt(0).toLowerCase();
|
209
|
+
const restOfString = str.slice(1);
|
210
|
+
return firstChar + restOfString;
|
211
|
+
}
|
212
|
+
async canDeactivate() {
|
213
|
+
if (!this.baseForm.dirty) {
|
214
|
+
return true;
|
215
|
+
}
|
216
|
+
var result = await this.dialogService.confirm(this.confirmQuestion, null, this.confirmButton);
|
217
|
+
return result;
|
218
|
+
}
|
219
|
+
getFiledName(filedTranslationKey) {
|
220
|
+
return { field: this.translate.instant(filedTranslationKey) };
|
221
|
+
}
|
222
|
+
getCustomErrorMessage(error, fieldLabel) {
|
223
|
+
return '';
|
224
|
+
}
|
225
|
+
getErrorMessageFromArray(arrayIndex, field, filedTranslationKey, arrayName = 'formArray') {
|
226
|
+
const fieldControl = this.baseForm.get(arrayName).get([arrayIndex]).get(field);
|
227
|
+
return this.getErrorMessageForField(fieldControl, filedTranslationKey);
|
228
|
+
}
|
229
|
+
getErrorMessage(field, filedTranslationKey, formGroup = null) {
|
230
|
+
const fg = this.getFromGroup(formGroup);
|
231
|
+
return this.getErrorMessageForField(fg.get(field), filedTranslationKey);
|
232
|
+
}
|
233
|
+
getErrorMessageForField(fieldControl, filedTranslationKey) {
|
234
|
+
const error = fieldControl.errors;
|
235
|
+
const fieldLabel = this.translate.instant(filedTranslationKey);
|
236
|
+
let rvalue = '';
|
237
|
+
if (error !== null) {
|
238
|
+
if (error['required'] === true) {
|
239
|
+
rvalue = this.translate.instant('General.Field.Required', { field: fieldLabel });
|
240
|
+
}
|
241
|
+
if (error['minlength']) {
|
242
|
+
rvalue = this.translate.instant('General.Field.MinLength', { field: fieldLabel, requiredLength: error.minlength.requiredLength });
|
243
|
+
}
|
244
|
+
if (error['email'] === true) {
|
245
|
+
rvalue = this.translate.instant('General.Field.InvalidEmail');
|
246
|
+
}
|
247
|
+
if (error['url'] === true) {
|
248
|
+
rvalue = this.translate.instant('General.Field.InvalidUrl');
|
249
|
+
}
|
250
|
+
if (error['serverError']) {
|
251
|
+
rvalue = error['serverError'];
|
252
|
+
}
|
253
|
+
if (rvalue === '') {
|
254
|
+
rvalue = this.getCustomErrorMessage(error, fieldLabel);
|
255
|
+
}
|
256
|
+
}
|
257
|
+
return rvalue;
|
258
|
+
}
|
259
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.1", ngImport: i0, type: GenericIonicCrudComponent, deps: [{ token: i1.UntypedFormBuilder }, { token: i2.Router }, { token: i2.ActivatedRoute }, { token: i3.IonicDialogService }, { token: i4.TranslateService }, { token: i5.Location }, { token: i0.Injector }, { token: i6.FormlyJsonschema }], target: i0.ɵɵFactoryTarget.Component }); }
|
260
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.1", type: GenericIonicCrudComponent, isStandalone: true, selector: "rslc-ionic-crud", ngImport: i0, template: "<ion-header [translucent]=\"true\">\n <ion-toolbar>\n <ion-buttons slot=\"start\">\n <ion-menu-button></ion-menu-button>\n <ion-back-button></ion-back-button>\n </ion-buttons>\n <ng-container [ngTemplateOutlet]=\"header\"></ng-container>\n </ion-toolbar>\n</ion-header>\n\n<ion-content [fullscreen]=\"true\">\n <ng-container *ngIf=\"!isLoading\">\n <form [formGroup]=\"baseForm\" (ngSubmit)=\"onSubmit(model)\">\n <formly-form [form]=\"baseForm\" [fields]=\"fields\" [model]=\"model\" [options]=\"options\"></formly-form>\n </form>\n </ng-container>\n</ion-content>\n\n<ng-template #header>\n <ion-title class=\"ion-text-center\">{{title |translate}}</ion-title>\n <ion-button *ngIf=\"!isLoading\" slot=\"end\" fill=\"clear\" (click)=\"onSave()\"\n [disabled]=\"baseForm.invalid || !baseForm.dirty\">\n <ion-icon size=\"large\" name=\"save\"></ion-icon>\n </ion-button>\n</ng-template>", styles: [""], dependencies: [{ kind: "ngmodule", type: RslIonicModuleModule }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i7.IonBackButton, selector: "ion-back-button", inputs: ["color", "defaultHref", "disabled", "icon", "mode", "routerAnimation", "text", "type"] }, { kind: "component", type: i7.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i7.IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: i7.IonContent, selector: "ion-content", inputs: ["color", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: i7.IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: i7.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i7.IonMenuButton, selector: "ion-menu-button", inputs: ["autoHide", "color", "disabled", "menu", "mode", "type"] }, { kind: "component", type: i7.IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: i7.IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "directive", type: i7.IonBackButtonDelegate, selector: "ion-back-button", inputs: ["defaultHref", "routerAnimation"] }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i8.FormlyForm, selector: "formly-form", inputs: ["form", "model", "fields", "options"], outputs: ["modelChange"] }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }] }); }
|
261
|
+
}
|
262
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.1", ngImport: i0, type: GenericIonicCrudComponent, decorators: [{
|
263
|
+
type: Component,
|
264
|
+
args: [{ standalone: true, selector: 'rslc-ionic-crud', imports: [RslIonicModuleModule], template: "<ion-header [translucent]=\"true\">\n <ion-toolbar>\n <ion-buttons slot=\"start\">\n <ion-menu-button></ion-menu-button>\n <ion-back-button></ion-back-button>\n </ion-buttons>\n <ng-container [ngTemplateOutlet]=\"header\"></ng-container>\n </ion-toolbar>\n</ion-header>\n\n<ion-content [fullscreen]=\"true\">\n <ng-container *ngIf=\"!isLoading\">\n <form [formGroup]=\"baseForm\" (ngSubmit)=\"onSubmit(model)\">\n <formly-form [form]=\"baseForm\" [fields]=\"fields\" [model]=\"model\" [options]=\"options\"></formly-form>\n </form>\n </ng-container>\n</ion-content>\n\n<ng-template #header>\n <ion-title class=\"ion-text-center\">{{title |translate}}</ion-title>\n <ion-button *ngIf=\"!isLoading\" slot=\"end\" fill=\"clear\" (click)=\"onSave()\"\n [disabled]=\"baseForm.invalid || !baseForm.dirty\">\n <ion-icon size=\"large\" name=\"save\"></ion-icon>\n </ion-button>\n</ng-template>" }]
|
265
|
+
}], ctorParameters: function () { return [{ type: i1.UntypedFormBuilder }, { type: i2.Router }, { type: i2.ActivatedRoute }, { type: i3.IonicDialogService }, { type: i4.TranslateService }, { type: i5.Location }, { type: i0.Injector }, { type: i6.FormlyJsonschema }]; } });
|
266
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rsl-ionic-crud.component.js","sourceRoot":"","sources":["../../../../../../../projects/rosoftlab/ionic/src/lib/components/rsl-ionic-crud.component/rsl-ionic-crud.component.ts","../../../../../../../projects/rosoftlab/ionic/src/lib/components/rsl-ionic-crud.component/rsl-ionic-crud.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Location } from '@angular/common';\nimport { Component, Injector, OnInit } from '@angular/core';\nimport { UntypedFormBuilder, UntypedFormControl, UntypedFormGroup } from '@angular/forms';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { FormlyFieldConfig, FormlyFormOptions } from '@ngx-formly/core';\nimport { FormlyJsonschema } from '@ngx-formly/core/json-schema';\nimport { TranslateService } from '@ngx-translate/core';\nimport { BaseModelFormly, BaseServiceFormly } from '@rosoftlab/formly';\nimport * as _ from 'lodash';\nimport { Observable, from } from 'rxjs';\nimport { IonicDialogService } from '../../ionic-dialog.service';\nimport { RslIonicModuleModule } from '../../rsl-ionic-module.module';\n@Component({\n  standalone: true,\n  selector: 'rslc-ionic-crud',\n  templateUrl: './rsl-ionic-crud.component.html',\n  styleUrls: ['./rsl-ionic-crud.component.css'],\n  imports: [RslIonicModuleModule]\n\n})\nexport class GenericIonicCrudComponent<T extends BaseModelFormly, U extends BaseServiceFormly<T>> implements OnInit {\n  baseForm: UntypedFormGroup;\n  modelId: string;\n  model: T;\n  origModel: T;\n  isEdit: boolean;\n  isLoading = true;\n  cancelRoute: string;\n  editRoute: string;\n  modelService: U\n  public changeUrlRoute: boolean = true;\n  public title: string;\n  fields: FormlyFieldConfig[]\n  options: FormlyFormOptions = {};\n  //#region Dialog messages\n  confirmQuestion: string\n  confirmButton: string\n  saveMessage: string\n  //#endregion\n  constructor(\n    public fb: UntypedFormBuilder,\n    public router: Router,\n    public route: ActivatedRoute,\n    public dialogService: IonicDialogService,\n    public translate: TranslateService,\n    public location: Location,\n    private injector: Injector,\n    private formlyJsonschema: FormlyJsonschema,\n  ) {\n\n    const SERVICE_TOKEN = route.snapshot.data['requiredService'];\n    this.modelService = this.injector.get<U>(<any>SERVICE_TOKEN);\n\n    this.modelService = this.injector.get<U>(<any>SERVICE_TOKEN);\n    this.confirmQuestion = this.translate.instant('General.Discard.Question');\n    this.confirmButton = this.translate.instant('General.Discard.ConfirmButton');\n    this.saveMessage = this.translate.instant('General.Save.Message');\n  }\n\n  ionViewWillLeave() {\n    this.options.resetModel()\n    this.baseForm.reset();\n  }\n\n  public afterSave(model: T): Observable<T> {\n    return new Observable((observer) => {\n      observer.next(model);\n      observer.complete();\n    });\n  }\n\n  public beforeSave(model: T): Observable<T> {\n    return new Observable((observer) => {\n      observer.next(model);\n      observer.complete();\n    });\n  }\n\n  ngOnInit() {\n    this.initForm();\n  }\n  initForm(customInclude: string = '', newModelId: string = null, model: T = null) {\n    if (model === null) {\n      this.modelId = this.route.snapshot.paramMap.get('id') ?? newModelId;\n      this.isEdit = false;\n      if (this.modelId) {\n        this.modelService.get(this.modelId, customInclude).subscribe((value: T) => {\n          this.isEdit = true;\n          this.generateForm(value)\n        }\n        );\n      } else {\n        if (this.changeUrlRoute) {\n          const addUrl = this.router.createUrlTree([]).toString();\n          this.editRoute = this.router.createUrlTree([addUrl.replace('add', 'edit')]).toString();\n        }\n        // }\n        this.generateForm(this.modelService.newModel());\n      }\n    } else {\n      this.modelId = model.id;\n      this.isEdit = true;\n      this.generateForm(model)\n    }\n  }\n  generateForm(model?: T) {\n    this.isLoading = false;\n    this.modelId = model.id;\n    this.model = model;\n    this.origModel = _.cloneDeep(model);;\n    this.title = this.model.modelConfig.formTitle\n    this.baseForm = new UntypedFormGroup({});//= this.modelService.toFormGroup(this.fb, model);\n    this.afterFormGenerated();\n    this.fields = this.modelService.getFormlyFields(this.model)\n  }\n  public afterFormGenerated() {\n\n  }\n  private getFromGroup(formGroup: UntypedFormGroup | string = null): UntypedFormGroup {\n    if (!formGroup)\n      return this.baseForm;\n    if (formGroup instanceof UntypedFormGroup)\n      return formGroup;\n    return this.baseForm.controls[formGroup] as UntypedFormGroup;\n  }\n  validateAllFormFields(formGroup: UntypedFormGroup | string = null) {\n    const fg = this.getFromGroup(formGroup)\n    Object.keys(fg.controls).forEach(field => {\n      // console.log(field);\n      const control = fg.get(field);\n      if (control instanceof UntypedFormControl) {\n        control.markAsTouched({ onlySelf: true });\n      } else if (control instanceof UntypedFormGroup) {\n        this.validateAllFormFields(control);\n      }\n    });\n  }\n  isFieldValid(field: string, formGroup: UntypedFormGroup | string = null) {\n    const fg = this.getFromGroup(formGroup)\n    const filedControl = fg.get(field)\n    return !filedControl.valid && filedControl.touched;\n  }\n\n  isFieldValidFromArray(arrayIndex: number, field: string, arrayName: string = 'formArray') {\n    const fieldControl = this.baseForm.get(arrayName).get([arrayIndex]).get(field)\n    return !fieldControl.valid && fieldControl.touched;\n  }\n\n  displayFieldCss(field: string) {\n    return {\n      'has-error': this.isFieldValid(field),\n      'has-feedback': this.isFieldValid(field)\n    };\n  }\n  onCancel() {\n    this.router.navigate([this.cancelRoute]);\n  }\n  onSave() {\n    this.saveModel(this.baseForm)\n  }\n  onSubmit(model) {\n    this.saveModel(this.baseForm)\n  }\n\n  saveModel(formGroup: UntypedFormGroup | string = null) {\n    const fg = this.getFromGroup(formGroup)\n    const that = this;\n    if (fg) {\n      if (fg.valid) {\n        this.beforeSave(this.model).subscribe(_ => {\n          this.modelService.save(this.model, this.modelId, this.origModel).subscribe(\n            (newModel: T) => {\n              this.model = newModel;\n              this.modelId = newModel.id;\n              if (this.editRoute) {\n                this.isEdit = true;\n                if (this.changeUrlRoute) {\n                  const url = this.router.createUrlTree([this.editRoute, this.modelId]).toString();\n                  this.location.replaceState(url);\n                }\n              }\n              this.afterSave(newModel).subscribe((val: T) => {\n                from(this.dialogService.showSaveMessage(this.saveMessage)).subscribe(d => {\n                  this.options.updateInitialValue()\n                  fg.markAsPristine();\n                });\n              });\n            },\n            err => {\n              this.serverErrors(err)\n            });\n        });\n      } else {\n        this.validateAllFormFields(formGroup);\n      }\n    }\n  }\n  serverErrors(err: any) {\n    if (err.error) {\n      if (err.error.errors) {\n        const validationErrors = err.error.errors;\n        if (Array.isArray(validationErrors)) {\n          validationErrors.forEach(prop => {\n            const formControl = this.baseForm.get(this.firstCharToLowerCase(prop));\n            if (formControl) {\n              // activate the error message\n              formControl.setErrors({\n                serverError: { message: validationErrors[prop].join('\\n') }\n              });\n            }\n          });\n        } else {\n          const keys = Object.keys(validationErrors);\n          keys.forEach(prop => {\n            const formControl = this.baseForm.get(this.firstCharToLowerCase(prop));\n            if (formControl) {\n              // activate the error message\n              formControl.setErrors({\n                serverError: { message: validationErrors[prop].join('\\n') }\n              });\n            }\n          });\n        }\n      }\n    }\n  }\n  private firstCharToLowerCase(str: string): string {\n    if (str.length === 0) {\n      return str; // Return an empty string if the input is empty\n    }\n\n    const firstChar = str.charAt(0).toLowerCase();\n    const restOfString = str.slice(1);\n\n    return firstChar + restOfString;\n  }\n  async canDeactivate(): Promise<boolean> {\n    if (!this.baseForm.dirty) {\n      return true;\n    }\n    var result = await this.dialogService.confirm(this.confirmQuestion, null, this.confirmButton)\n    return result;\n  }\n  getFiledName(filedTranslationKey: string) {\n    return { field: this.translate.instant(filedTranslationKey) };\n  }\n  getCustomErrorMessage(error: any, fieldLabel: string): string {\n    return '';\n  }\n  getErrorMessageFromArray(arrayIndex: number, field: string, filedTranslationKey: string, arrayName: string = 'formArray'): string {\n    const fieldControl = this.baseForm.get(arrayName).get([arrayIndex]).get(field)\n    return this.getErrorMessageForField(fieldControl, filedTranslationKey)\n  }\n  getErrorMessage(field: string, filedTranslationKey: string, formGroup: UntypedFormGroup | string = null): string {\n    const fg = this.getFromGroup(formGroup)\n    return this.getErrorMessageForField(fg.get(field), filedTranslationKey)\n  }\n  getErrorMessageForField(fieldControl: any, filedTranslationKey: string): string {\n    const error = fieldControl.errors;\n    const fieldLabel = this.translate.instant(filedTranslationKey);\n    let rvalue = '';\n    if (error !== null) {\n      if (error['required'] === true) {\n        rvalue = this.translate.instant('General.Field.Required', { field: fieldLabel });\n      }\n      if (error['minlength']) {\n        rvalue = this.translate.instant('General.Field.MinLength', { field: fieldLabel, requiredLength: error.minlength.requiredLength });\n      }\n      if (error['email'] === true) {\n        rvalue = this.translate.instant('General.Field.InvalidEmail');\n      }\n      if (error['url'] === true) {\n        rvalue = this.translate.instant('General.Field.InvalidUrl');\n      }\n      if (error['serverError']) {\n        rvalue = error['serverError'];\n      }\n      if (rvalue === '') {\n        rvalue = this.getCustomErrorMessage(error, fieldLabel);\n      }\n    }\n    return rvalue;\n  }\n}\n","<ion-header [translucent]=\"true\">\n  <ion-toolbar>\n    <ion-buttons slot=\"start\">\n      <ion-menu-button></ion-menu-button>\n      <ion-back-button></ion-back-button>\n    </ion-buttons>\n    <ng-container [ngTemplateOutlet]=\"header\"></ng-container>\n  </ion-toolbar>\n</ion-header>\n\n<ion-content [fullscreen]=\"true\">\n  <ng-container *ngIf=\"!isLoading\">\n    <form [formGroup]=\"baseForm\" (ngSubmit)=\"onSubmit(model)\">\n      <formly-form [form]=\"baseForm\" [fields]=\"fields\" [model]=\"model\" [options]=\"options\"></formly-form>\n    </form>\n  </ng-container>\n</ion-content>\n\n<ng-template #header>\n  <ion-title  class=\"ion-text-center\">{{title |translate}}</ion-title>\n  <ion-button *ngIf=\"!isLoading\" slot=\"end\" fill=\"clear\" (click)=\"onSave()\"\n    [disabled]=\"baseForm.invalid || !baseForm.dirty\">\n    <ion-icon size=\"large\" name=\"save\"></ion-icon>\n  </ion-button>\n</ng-template>"]}
|