@rolatech/angular-course 17.3.1 → 17.3.2

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.
Files changed (32) hide show
  1. package/esm2022/index.mjs +2 -1
  2. package/esm2022/lib/pages/course-manage/course-manage-content/course-manage-content.component.mjs +12 -0
  3. package/esm2022/lib/pages/course-manage/course-manage-details/course-manage-details.component.mjs +159 -0
  4. package/esm2022/lib/pages/course-manage/course-manage-info/course-manage-info.component.mjs +139 -0
  5. package/esm2022/lib/pages/course-manage/course-manage-layout/course-manage-layout.component.mjs +68 -0
  6. package/esm2022/lib/pages/course-manage/course-manage-media/course-manage-media.component.mjs +137 -0
  7. package/esm2022/lib/pages/course-manage/course-manage-pricing/course-manage-pricing.component.mjs +126 -0
  8. package/esm2022/lib/pages/course-manage/course-manage-schedule/course-manage-schedule.component.mjs +126 -0
  9. package/esm2022/lib/pages/course-manage/course-manage-section/course-manage-section.component.mjs +342 -0
  10. package/esm2022/lib/pages/course-manage/course-manage.routes.mjs +40 -0
  11. package/esm2022/lib/services/booking.service.mjs +24 -0
  12. package/esm2022/lib/services/instructor.service.mjs +24 -0
  13. package/fesm2022/{rolatech-angular-course-course-index.component-Bm6Sg8zH.mjs → rolatech-angular-course-course-index.component-DmBSbnLe.mjs} +5 -4
  14. package/fesm2022/{rolatech-angular-course-course-index.component-Bm6Sg8zH.mjs.map → rolatech-angular-course-course-index.component-DmBSbnLe.mjs.map} +1 -1
  15. package/fesm2022/{rolatech-angular-course-rolatech-angular-course-D-_W2GtF.mjs → rolatech-angular-course-rolatech-angular-course-ChplliNh.mjs} +1095 -13
  16. package/fesm2022/rolatech-angular-course-rolatech-angular-course-ChplliNh.mjs.map +1 -0
  17. package/fesm2022/rolatech-angular-course.mjs +2 -1
  18. package/fesm2022/rolatech-angular-course.mjs.map +1 -1
  19. package/index.d.ts +1 -0
  20. package/lib/pages/course-manage/course-manage-content/course-manage-content.component.d.ts +5 -0
  21. package/lib/pages/course-manage/course-manage-details/course-manage-details.component.d.ts +29 -0
  22. package/lib/pages/course-manage/course-manage-info/course-manage-info.component.d.ts +36 -0
  23. package/lib/pages/course-manage/course-manage-layout/course-manage-layout.component.d.ts +26 -0
  24. package/lib/pages/course-manage/course-manage-media/course-manage-media.component.d.ts +29 -0
  25. package/lib/pages/course-manage/course-manage-pricing/course-manage-pricing.component.d.ts +27 -0
  26. package/lib/pages/course-manage/course-manage-schedule/course-manage-schedule.component.d.ts +27 -0
  27. package/lib/pages/course-manage/course-manage-section/course-manage-section.component.d.ts +37 -0
  28. package/lib/pages/course-manage/course-manage.routes.d.ts +2 -0
  29. package/lib/services/booking.service.d.ts +9 -0
  30. package/lib/services/instructor.service.d.ts +9 -0
  31. package/package.json +1 -1
  32. package/fesm2022/rolatech-angular-course-rolatech-angular-course-D-_W2GtF.mjs.map +0 -1
package/esm2022/index.mjs CHANGED
@@ -2,4 +2,5 @@ export * from './lib/interfaces';
2
2
  export * from './provider';
3
3
  export * from './lib/components';
4
4
  export { courseRoutes } from './lib/pages/course/course.routes';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXItY291cnNlL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGtDQUFrQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9saWIvaW50ZXJmYWNlcyc7XG5leHBvcnQgKiBmcm9tICcuL3Byb3ZpZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMnO1xuZXhwb3J0IHsgY291cnNlUm91dGVzIH0gZnJvbSAnLi9saWIvcGFnZXMvY291cnNlL2NvdXJzZS5yb3V0ZXMnO1xuIl19
5
+ export { courseManageRoutes } from './lib/pages/course-manage/course-manage.routes';
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXItY291cnNlL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGtDQUFrQyxDQUFDO0FBQ2hFLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGdEQUFnRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9saWIvaW50ZXJmYWNlcyc7XG5leHBvcnQgKiBmcm9tICcuL3Byb3ZpZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMnO1xuZXhwb3J0IHsgY291cnNlUm91dGVzIH0gZnJvbSAnLi9saWIvcGFnZXMvY291cnNlL2NvdXJzZS5yb3V0ZXMnO1xuZXhwb3J0IHsgY291cnNlTWFuYWdlUm91dGVzIH0gZnJvbSAnLi9saWIvcGFnZXMvY291cnNlLW1hbmFnZS9jb3Vyc2UtbWFuYWdlLnJvdXRlcyc7XG4iXX0=
@@ -0,0 +1,12 @@
1
+ import { Component } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from "@angular/core";
4
+ export class CourseManageContentComponent {
5
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageContentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
6
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.1", type: CourseManageContentComponent, isStandalone: true, selector: "rolatech-course-manage-content", ngImport: i0, template: "<ng-content select=\"rolatech-toolbar\"></ng-content>\n<div class=\"p-3\">\n <ng-content></ng-content>\n</div>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
7
+ }
8
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageContentComponent, decorators: [{
9
+ type: Component,
10
+ args: [{ selector: 'rolatech-course-manage-content', standalone: true, imports: [CommonModule], template: "<ng-content select=\"rolatech-toolbar\"></ng-content>\n<div class=\"p-3\">\n <ng-content></ng-content>\n</div>\n" }]
11
+ }] });
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY291cnNlLW1hbmFnZS1jb250ZW50LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci1jb3Vyc2Uvc3JjL2xpYi9wYWdlcy9jb3Vyc2UtbWFuYWdlL2NvdXJzZS1tYW5hZ2UtY29udGVudC9jb3Vyc2UtbWFuYWdlLWNvbnRlbnQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyLWNvdXJzZS9zcmMvbGliL3BhZ2VzL2NvdXJzZS1tYW5hZ2UvY291cnNlLW1hbmFnZS1jb250ZW50L2NvdXJzZS1tYW5hZ2UtY29udGVudC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzFDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7QUFTL0MsTUFBTSxPQUFPLDRCQUE0Qjs4R0FBNUIsNEJBQTRCO2tHQUE1Qiw0QkFBNEIsMEZDVnpDLG1IQUlBLHlEREVZLFlBQVk7OzJGQUlYLDRCQUE0QjtrQkFQeEMsU0FBUzsrQkFDRSxnQ0FBZ0MsY0FDOUIsSUFBSSxXQUNQLENBQUMsWUFBWSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdyb2xhdGVjaC1jb3Vyc2UtbWFuYWdlLWNvbnRlbnQnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcbiAgdGVtcGxhdGVVcmw6ICcuL2NvdXJzZS1tYW5hZ2UtY29udGVudC5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsOiAnLi9jb3Vyc2UtbWFuYWdlLWNvbnRlbnQuY29tcG9uZW50LnNjc3MnLFxufSlcbmV4cG9ydCBjbGFzcyBDb3Vyc2VNYW5hZ2VDb250ZW50Q29tcG9uZW50IHt9XG4iLCI8bmctY29udGVudCBzZWxlY3Q9XCJyb2xhdGVjaC10b29sYmFyXCI+PC9uZy1jb250ZW50PlxuPGRpdiBjbGFzcz1cInAtM1wiPlxuICA8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG48L2Rpdj5cbiJdfQ==
@@ -0,0 +1,159 @@
1
+ import { Component, inject } from '@angular/core';
2
+ import { MatDialog } from '@angular/material/dialog';
3
+ import { MatSnackBar } from '@angular/material/snack-bar';
4
+ import { ActivatedRoute } from '@angular/router';
5
+ import { findIndex, remove } from 'lodash';
6
+ import { MatIconModule } from '@angular/material/icon';
7
+ import { MatButtonModule } from '@angular/material/button';
8
+ import { ConfirmationDialogComponent, ToolbarComponent } from '@rolatech/angular-components';
9
+ import { CourseManageContentComponent } from '../course-manage-content/course-manage-content.component';
10
+ import { DetailItemComponent } from '../../../components';
11
+ import { CourseStatus, CourseType } from '../../../interfaces';
12
+ import { CourseService } from '../../../services';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "@angular/material/button";
15
+ import * as i2 from "@angular/material/icon";
16
+ export class CourseManageDetailsComponent {
17
+ constructor() {
18
+ this.route = inject(ActivatedRoute);
19
+ this.courseService = inject(CourseService);
20
+ this.dialog = inject(MatDialog);
21
+ this.snackBar = inject(MatSnackBar);
22
+ this.isLoading = false;
23
+ this.isUploading = false;
24
+ this.details = [];
25
+ this.status = CourseStatus;
26
+ this.courseType = CourseType;
27
+ this.id = this.route.parent?.snapshot.paramMap.get('id');
28
+ }
29
+ ngOnInit() {
30
+ this.find();
31
+ }
32
+ find() {
33
+ this.courseService.findDetails(this.id).subscribe({
34
+ next: (res) => {
35
+ if (res.data) {
36
+ this.details = res.data;
37
+ }
38
+ },
39
+ });
40
+ }
41
+ addDetail() {
42
+ if (!this.details) {
43
+ this.details = [];
44
+ }
45
+ this.courseService
46
+ .addDetail(this.id, {
47
+ title: '',
48
+ description: '',
49
+ content: '',
50
+ media: [],
51
+ })
52
+ .subscribe({
53
+ next: (res) => {
54
+ const detail = res.data;
55
+ this.details.push({
56
+ id: detail.id,
57
+ title: '',
58
+ description: '',
59
+ content: '',
60
+ media: [],
61
+ });
62
+ },
63
+ error: (e) => {
64
+ this.snackBar.open(e.message);
65
+ },
66
+ });
67
+ }
68
+ onDetailMediaUpload(event) {
69
+ const detailId = event.id;
70
+ const index = findIndex(this.details, (item) => item.id === detailId);
71
+ const file = event.data.target.files[0];
72
+ if (file) {
73
+ const reader = new FileReader();
74
+ reader.readAsDataURL(file);
75
+ reader.onload = () => {
76
+ if (!this.details[index].media) {
77
+ this.details[index].media = [];
78
+ }
79
+ this.details[index].media.push({
80
+ url: reader.result,
81
+ alt: 'upload image',
82
+ });
83
+ this.details[index].isUploading = true;
84
+ const formData = new FormData();
85
+ formData.append('file', file);
86
+ this.courseService.uploadDetailMedia(this.id, detailId, formData).subscribe({
87
+ next: (res) => {
88
+ this.isUploading = false;
89
+ this.details[index].isUploading = false;
90
+ const tmp = this.details[index].media;
91
+ delete res.data.productDetail;
92
+ this.details[index].media[this.details[index].media.length - 1] = res.data;
93
+ },
94
+ error: (e) => {
95
+ this.isUploading = false;
96
+ this.snackBar.open('上传失败: ' + e.message);
97
+ },
98
+ });
99
+ };
100
+ reader.onerror = (error) => {
101
+ this.isUploading = false;
102
+ };
103
+ }
104
+ }
105
+ onDetailMediaDelete(event) {
106
+ const detailId = event.id;
107
+ const mediaId = event.media.id;
108
+ this.courseService.deleteDetailMedia(this.id, detailId, mediaId).subscribe({
109
+ next: (res) => {
110
+ const detail = this.details.find((item) => item.id === detailId);
111
+ remove(detail.media, {
112
+ id: mediaId,
113
+ });
114
+ },
115
+ });
116
+ }
117
+ onDetailSave(detail) {
118
+ delete detail.isUploading;
119
+ this.courseService.updateDetail(this.id, detail.id, detail).subscribe({
120
+ next: (res) => {
121
+ this.snackBar.open('保存成功');
122
+ },
123
+ error: (e) => {
124
+ this.snackBar.open(e.message);
125
+ },
126
+ });
127
+ }
128
+ onDetailDelete(detail) {
129
+ const dialogRef = this.dialog.open(ConfirmationDialogComponent, {
130
+ width: '400px',
131
+ data: {
132
+ title: '删除详情',
133
+ message: '确定删除此详情吗?',
134
+ },
135
+ });
136
+ dialogRef.afterClosed().subscribe((result) => {
137
+ if (result) {
138
+ this.courseService.deleteDetail(this.id, detail.id).subscribe({
139
+ next: (res) => {
140
+ this.snackBar.open(res.data);
141
+ remove(this.details, {
142
+ id: detail.id,
143
+ });
144
+ },
145
+ error: (e) => {
146
+ this.snackBar.open(e.message);
147
+ },
148
+ });
149
+ }
150
+ });
151
+ }
152
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageDetailsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
153
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.1", type: CourseManageDetailsComponent, isStandalone: true, selector: "rolatech-course-manage-details", ngImport: i0, template: "<rolatech-course-manage-content>\n <rolatech-toolbar title=\"\u8BE6\u60C5\" class=\"hidden sm:block\" divider> </rolatech-toolbar>\n <div>\n <div>\n @for (detail of details; track detail) {\n <rolatech-detail-item\n [detail]=\"detail\"\n (upload)=\"onDetailMediaUpload($event)\"\n (deleteMedia)=\"onDetailMediaDelete($event)\"\n (save)=\"onDetailSave($event)\"\n (delete)=\"onDetailDelete($event)\"\n [actions]=\"true\"\n ></rolatech-detail-item>\n }\n </div>\n <button mat-stroked-button (click)=\"addDetail()\" class=\"mt-3\">\n <mat-icon>add</mat-icon>\n <span>\u6DFB\u52A0\u8BE6\u60C5</span>\n </button>\n </div>\n</rolatech-course-manage-content>\n", styles: [""], dependencies: [{ kind: "component", type: DetailItemComponent, selector: "rolatech-detail-item", inputs: ["isUploading", "detail", "actions", "selectMedia"], outputs: ["upload", "delete", "save", "deleteMedia"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i1.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: ToolbarComponent, selector: "rolatech-toolbar", inputs: ["title", "subtitle", "back", "link", "large", "divider"] }, { kind: "component", type: CourseManageContentComponent, selector: "rolatech-course-manage-content" }] }); }
154
+ }
155
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageDetailsComponent, decorators: [{
156
+ type: Component,
157
+ args: [{ selector: 'rolatech-course-manage-details', standalone: true, imports: [DetailItemComponent, MatButtonModule, MatIconModule, ToolbarComponent, CourseManageContentComponent], template: "<rolatech-course-manage-content>\n <rolatech-toolbar title=\"\u8BE6\u60C5\" class=\"hidden sm:block\" divider> </rolatech-toolbar>\n <div>\n <div>\n @for (detail of details; track detail) {\n <rolatech-detail-item\n [detail]=\"detail\"\n (upload)=\"onDetailMediaUpload($event)\"\n (deleteMedia)=\"onDetailMediaDelete($event)\"\n (save)=\"onDetailSave($event)\"\n (delete)=\"onDetailDelete($event)\"\n [actions]=\"true\"\n ></rolatech-detail-item>\n }\n </div>\n <button mat-stroked-button (click)=\"addDetail()\" class=\"mt-3\">\n <mat-icon>add</mat-icon>\n <span>\u6DFB\u52A0\u8BE6\u60C5</span>\n </button>\n </div>\n</rolatech-course-manage-content>\n" }]
158
+ }], ctorParameters: () => [] });
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"course-manage-details.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular-course/src/lib/pages/course-manage/course-manage-details/course-manage-details.component.ts","../../../../../../../../libs/angular-course/src/lib/pages/course-manage/course-manage-details/course-manage-details.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, inject } from '@angular/core';\nimport { MatDialog } from '@angular/material/dialog';\nimport { MatSnackBar } from '@angular/material/snack-bar';\nimport { ActivatedRoute } from '@angular/router';\nimport { findIndex, last, remove } from 'lodash';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatButtonModule } from '@angular/material/button';\nimport { ConfirmationDialogComponent, ToolbarComponent } from '@rolatech/angular-components';\nimport { CourseManageContentComponent } from '../course-manage-content/course-manage-content.component';\nimport { DetailItemComponent } from '../../../components';\nimport { CourseDetail, CourseStatus, CourseType } from '../../../interfaces';\nimport { CourseService } from '../../../services';\n\n@Component({\n  selector: 'rolatech-course-manage-details',\n  templateUrl: './course-manage-details.component.html',\n  styleUrls: ['./course-manage-details.component.scss'],\n  standalone: true,\n  imports: [DetailItemComponent, MatButtonModule, MatIconModule, ToolbarComponent, CourseManageContentComponent],\n})\nexport class CourseManageDetailsComponent implements OnInit {\n  route = inject(ActivatedRoute);\n  courseService = inject(CourseService);\n  dialog = inject(MatDialog);\n  snackBar = inject(MatSnackBar);\n\n  id: string;\n  isLoading = false;\n  isUploading = false;\n  details: CourseDetail[] = [];\n  status = CourseStatus;\n  courseType: any = CourseType;\n  constructor() {\n    this.id = this.route.parent?.snapshot.paramMap.get('id') as string;\n  }\n  ngOnInit(): void {\n    this.find();\n  }\n  find() {\n    this.courseService.findDetails(this.id).subscribe({\n      next: (res) => {\n        if (res.data) {\n          this.details = res.data;\n        }\n      },\n    });\n  }\n  addDetail() {\n    if (!this.details) {\n      this.details = [];\n    }\n    this.courseService\n      .addDetail(this.id, {\n        title: '',\n        description: '',\n        content: '',\n        media: [],\n      })\n      .subscribe({\n        next: (res: any) => {\n          const detail = res.data;\n          this.details.push({\n            id: detail.id,\n            title: '',\n            description: '',\n            content: '',\n            media: [],\n          });\n        },\n        error: (e) => {\n          this.snackBar.open(e.message);\n        },\n      });\n  }\n  onDetailMediaUpload(event: any) {\n    const detailId = event.id;\n    const index = findIndex(this.details, (item) => item.id === detailId);\n    const file = event.data.target.files[0];\n    if (file) {\n      const reader = new FileReader();\n      reader.readAsDataURL(file);\n      reader.onload = () => {\n        if (!this.details[index].media) {\n          this.details[index].media = [];\n        }\n        (this.details[index].media as any).push({\n          url: reader.result,\n          alt: 'upload image',\n        });\n        this.details[index].isUploading = true;\n        const formData = new FormData();\n        formData.append('file', file);\n        this.courseService.uploadDetailMedia(this.id, detailId, formData).subscribe({\n          next: (res: any) => {\n            this.isUploading = false;\n            this.details[index].isUploading = false;\n            const tmp = this.details[index].media as any;\n            delete res.data.productDetail;\n            this.details[index].media[this.details[index].media.length - 1] = res.data;\n          },\n          error: (e) => {\n            this.isUploading = false;\n            this.snackBar.open('上传失败: ' + e.message);\n          },\n        });\n      };\n      reader.onerror = (error) => {\n        this.isUploading = false;\n      };\n    }\n  }\n  onDetailMediaDelete(event: any) {\n    const detailId = event.id;\n    const mediaId = event.media.id;\n    this.courseService.deleteDetailMedia(this.id, detailId, mediaId).subscribe({\n      next: (res: any) => {\n        const detail: any = this.details.find((item) => item.id === detailId);\n        remove(detail.media, {\n          id: mediaId,\n        });\n      },\n    });\n  }\n  onDetailSave(detail: CourseDetail) {\n    delete detail.isUploading;\n    this.courseService.updateDetail(this.id, detail.id, detail).subscribe({\n      next: (res: any) => {\n        this.snackBar.open('保存成功');\n      },\n      error: (e) => {\n        this.snackBar.open(e.message);\n      },\n    });\n  }\n  onDetailDelete(detail: CourseDetail) {\n    const dialogRef = this.dialog.open(ConfirmationDialogComponent, {\n      width: '400px',\n      data: {\n        title: '删除详情',\n        message: '确定删除此详情吗?',\n      },\n    });\n    dialogRef.afterClosed().subscribe((result) => {\n      if (result) {\n        this.courseService.deleteDetail(this.id, detail.id).subscribe({\n          next: (res: any) => {\n            this.snackBar.open(res.data);\n            remove(this.details, {\n              id: detail.id,\n            });\n          },\n          error: (e) => {\n            this.snackBar.open(e.message);\n          },\n        });\n      }\n    });\n  }\n}\n","<rolatech-course-manage-content>\n  <rolatech-toolbar title=\"详情\" class=\"hidden sm:block\" divider> </rolatech-toolbar>\n  <div>\n    <div>\n      @for (detail of details; track detail) {\n        <rolatech-detail-item\n          [detail]=\"detail\"\n          (upload)=\"onDetailMediaUpload($event)\"\n          (deleteMedia)=\"onDetailMediaDelete($event)\"\n          (save)=\"onDetailSave($event)\"\n          (delete)=\"onDetailDelete($event)\"\n          [actions]=\"true\"\n        ></rolatech-detail-item>\n      }\n    </div>\n    <button mat-stroked-button (click)=\"addDetail()\" class=\"mt-3\">\n      <mat-icon>add</mat-icon>\n      <span>添加详情</span>\n    </button>\n  </div>\n</rolatech-course-manage-content>\n"]}
@@ -0,0 +1,139 @@
1
+ import { Component, inject } from '@angular/core';
2
+ import { BaseComponent, ToolbarComponent } from '@rolatech/angular-components';
3
+ import { clone } from 'lodash';
4
+ import { MatButtonModule } from '@angular/material/button';
5
+ import { MatOptionModule } from '@angular/material/core';
6
+ import { MatSelectModule } from '@angular/material/select';
7
+ import { TextFieldModule } from '@angular/cdk/text-field';
8
+ import { MatInputModule } from '@angular/material/input';
9
+ import { MatFormFieldModule } from '@angular/material/form-field';
10
+ import { FormsModule } from '@angular/forms';
11
+ import { CourseManageContentComponent } from '../course-manage-content/course-manage-content.component';
12
+ import { CourseStatus } from '../../../interfaces';
13
+ import { CategoryService, CourseService } from '../../../services';
14
+ import { BookingService } from '../../../services/booking.service';
15
+ import { InstructorService } from '../../../services/instructor.service';
16
+ import * as i0 from "@angular/core";
17
+ import * as i1 from "@angular/forms";
18
+ import * as i2 from "@angular/material/form-field";
19
+ import * as i3 from "@angular/material/input";
20
+ import * as i4 from "@angular/cdk/text-field";
21
+ import * as i5 from "@angular/material/select";
22
+ import * as i6 from "@angular/material/core";
23
+ import * as i7 from "@angular/material/button";
24
+ export class CourseManageInfoComponent extends BaseComponent {
25
+ constructor() {
26
+ super(...arguments);
27
+ this.courseService = inject(CourseService);
28
+ this.instructorService = inject(InstructorService);
29
+ this.categoryService = inject(CategoryService);
30
+ this.bookingService = inject(BookingService);
31
+ this.isLoading = false;
32
+ this.status = CourseStatus;
33
+ this.courseType = [
34
+ {
35
+ key: 'ONLINE',
36
+ value: '线上',
37
+ },
38
+ {
39
+ key: 'OFFLINE',
40
+ value: '线下',
41
+ },
42
+ {
43
+ key: 'MIXED',
44
+ value: '混合',
45
+ },
46
+ ];
47
+ }
48
+ ngOnInit() {
49
+ this.id = this.route.parent?.snapshot.paramMap.get('id');
50
+ this.find();
51
+ this.categoryService.find({}).subscribe({
52
+ next: (res) => {
53
+ this.categories = res.data;
54
+ },
55
+ });
56
+ this.findClassrooms();
57
+ this.findBookedClassrooms();
58
+ }
59
+ findBookedClassrooms() {
60
+ this.bookingService.items({ status: 'paid' }).subscribe({
61
+ next: (res) => {
62
+ this.bookedClassrooms = res.data;
63
+ },
64
+ });
65
+ }
66
+ findClassrooms() {
67
+ this.instructorService.findMyClassrooms({}).subscribe({
68
+ next: (res) => {
69
+ this.classrooms = res.data;
70
+ },
71
+ });
72
+ }
73
+ find() {
74
+ this.courseService.get(this.id).subscribe({
75
+ next: (res) => {
76
+ this.course = res.data;
77
+ },
78
+ });
79
+ }
80
+ compareFn(o1, o2) {
81
+ return o1.id === o2?.id;
82
+ }
83
+ compareClassroom(o1, o2) {
84
+ if (o1.room) {
85
+ return (o1.latitude === o2?.latitude &&
86
+ o1.longitude === o2?.longitude &&
87
+ o1.room === o2.room &&
88
+ o1.startAt === o2.startAt &&
89
+ o1.endAt === o2.endAt);
90
+ }
91
+ else {
92
+ return o1.latitude === o2?.latitude && o1.longitude === o2?.longitude;
93
+ }
94
+ }
95
+ onSelectionChange(event) {
96
+ this.selectedCategoyIds = event.value;
97
+ }
98
+ onClassroomSelected(event) {
99
+ this.selectedClassroom = event.value;
100
+ }
101
+ update() {
102
+ const { name, description, categories, type, classroom } = this.course;
103
+ const finalClassroom = clone(classroom);
104
+ finalClassroom.address = classroom.room ? classroom.address + ' ' + classroom.room + '室' : classroom.address;
105
+ const data = {
106
+ name,
107
+ description,
108
+ categories,
109
+ type,
110
+ classroom: finalClassroom,
111
+ };
112
+ this.courseService.update(this.id, data).subscribe({
113
+ next: (res) => {
114
+ this.snackBarService.open('保存成功');
115
+ // this.router.navigate([`../${res.data.id}`], { relativeTo: this.route });
116
+ },
117
+ error: (e) => {
118
+ this.snackBarService.open(e.message);
119
+ },
120
+ });
121
+ }
122
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageInfoComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
123
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.1", type: CourseManageInfoComponent, isStandalone: true, selector: "rolatech-course-manage-info", usesInheritance: true, ngImport: i0, template: "<rolatech-course-manage-content>\n <rolatech-toolbar title=\"\u57FA\u672C\u4FE1\u606F\" class=\"hidden sm:block\" divider></rolatech-toolbar>\n @if (course) {\n <div class=\"flex flex-col\">\n <form #courseForm=\"ngForm\" (ngSubmit)=\"update()\">\n <mat-form-field appearance=\"fill\">\n <mat-label> \u8BFE\u7A0B\u540D\u79F0 </mat-label>\n <input matInput type=\"text\" [(ngModel)]=\"course.name\" name=\"name\" required />\n </mat-form-field>\n <mat-form-field appearance=\"fill\">\n <mat-label> \u8BFE\u7A0B\u63CF\u8FF0 </mat-label>\n <textarea\n matInput\n type=\"text\"\n [(ngModel)]=\"course.description\"\n name=\"description\"\n required\n cdkTextareaAutosize\n cdkAutosizeMinRows=\"3\"\n ></textarea>\n </mat-form-field>\n <mat-form-field appearance=\"fill\" ngModelGroup=\"categories\" required>\n <mat-label>\u5206\u7C7B</mat-label>\n <mat-select\n multiple\n name=\"id\"\n [compareWith]=\"compareFn\"\n [(ngModel)]=\"course.categories\"\n #select=\"matSelect\"\n (selectionChange)=\"onSelectionChange($event)\"\n >\n @for (item of categories; track item) {\n <mat-option [value]=\"item\">\n {{ item.name }}\n </mat-option>\n }\n </mat-select>\n </mat-form-field>\n <mat-form-field appearance=\"fill\">\n <mat-label>\u7C7B\u578B</mat-label>\n <mat-select name=\"type\" [(ngModel)]=\"course.type\" required>\n @for (item of courseType; track item) {\n <mat-option [value]=\"item.key\">\n {{ item.value }}\n </mat-option>\n }\n </mat-select>\n </mat-form-field>\n @if (course.type && course.type.toString() !== 'ONLINE') {\n <mat-form-field appearance=\"fill\" ngModelGroup=\"classroom\">\n <mat-label>\u6559\u5BA4</mat-label>\n <mat-select\n name=\"id\"\n [(ngModel)]=\"course.classroom\"\n [compareWith]=\"compareClassroom\"\n (selectionChange)=\"onClassroomSelected($event)\"\n required\n >\n <mat-optgroup label=\"\u81EA\u6709\u6559\u5BA4\">\n @for (classroom of classrooms; track classroom) {\n <mat-option [value]=\"classroom\">\n <div class=\"flex gap-3\">\n <span>{{ classroom.name }}</span>\n <span> {{ classroom.address }}</span>\n </div>\n </mat-option>\n }\n </mat-optgroup>\n <mat-optgroup label=\"\u5DF2\u8D2D\u6559\u5BA4\">\n @for (item of bookedClassrooms; track item) {\n <mat-option [value]=\"item\">\n <div class=\"flex flex-col py-3\">\n <span class=\"font-bold\"> {{ item.address }} {{ item.room }}\u5BA4</span>\n <span class=\"text-md font-thin\"> {{ item.startAt }} - {{ item.endAt }}</span>\n </div>\n </mat-option>\n }\n </mat-optgroup>\n </mat-select>\n </mat-form-field>\n }\n </form>\n </div>\n <div>\n <button mat-flat-button type=\"primary\" (click)=\"update()\">\u4FDD\u5B58</button>\n </div>\n }\n</rolatech-course-manage-content>\n", styles: ["mat-form-field{width:100%}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i1.NgModelGroup, selector: "[ngModelGroup]", inputs: ["ngModelGroup"], exportAs: ["ngModelGroup"] }, { kind: "directive", type: i1.NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: ["ngFormOptions"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i2.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i2.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i3.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "directive", type: i4.CdkTextareaAutosize, selector: "textarea[cdkTextareaAutosize]", inputs: ["cdkAutosizeMinRows", "cdkAutosizeMaxRows", "cdkTextareaAutosize", "placeholder"], exportAs: ["cdkTextareaAutosize"] }, { kind: "ngmodule", type: TextFieldModule }, { kind: "ngmodule", type: MatSelectModule }, { kind: "component", type: i5.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i6.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "component", type: i6.MatOptgroup, selector: "mat-optgroup", inputs: ["label", "disabled"], exportAs: ["matOptgroup"] }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i7.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: ToolbarComponent, selector: "rolatech-toolbar", inputs: ["title", "subtitle", "back", "link", "large", "divider"] }, { kind: "component", type: CourseManageContentComponent, selector: "rolatech-course-manage-content" }] }); }
124
+ }
125
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageInfoComponent, decorators: [{
126
+ type: Component,
127
+ args: [{ selector: 'rolatech-course-manage-info', standalone: true, imports: [
128
+ FormsModule,
129
+ MatFormFieldModule,
130
+ MatInputModule,
131
+ TextFieldModule,
132
+ MatSelectModule,
133
+ MatOptionModule,
134
+ MatButtonModule,
135
+ ToolbarComponent,
136
+ CourseManageContentComponent,
137
+ ], template: "<rolatech-course-manage-content>\n <rolatech-toolbar title=\"\u57FA\u672C\u4FE1\u606F\" class=\"hidden sm:block\" divider></rolatech-toolbar>\n @if (course) {\n <div class=\"flex flex-col\">\n <form #courseForm=\"ngForm\" (ngSubmit)=\"update()\">\n <mat-form-field appearance=\"fill\">\n <mat-label> \u8BFE\u7A0B\u540D\u79F0 </mat-label>\n <input matInput type=\"text\" [(ngModel)]=\"course.name\" name=\"name\" required />\n </mat-form-field>\n <mat-form-field appearance=\"fill\">\n <mat-label> \u8BFE\u7A0B\u63CF\u8FF0 </mat-label>\n <textarea\n matInput\n type=\"text\"\n [(ngModel)]=\"course.description\"\n name=\"description\"\n required\n cdkTextareaAutosize\n cdkAutosizeMinRows=\"3\"\n ></textarea>\n </mat-form-field>\n <mat-form-field appearance=\"fill\" ngModelGroup=\"categories\" required>\n <mat-label>\u5206\u7C7B</mat-label>\n <mat-select\n multiple\n name=\"id\"\n [compareWith]=\"compareFn\"\n [(ngModel)]=\"course.categories\"\n #select=\"matSelect\"\n (selectionChange)=\"onSelectionChange($event)\"\n >\n @for (item of categories; track item) {\n <mat-option [value]=\"item\">\n {{ item.name }}\n </mat-option>\n }\n </mat-select>\n </mat-form-field>\n <mat-form-field appearance=\"fill\">\n <mat-label>\u7C7B\u578B</mat-label>\n <mat-select name=\"type\" [(ngModel)]=\"course.type\" required>\n @for (item of courseType; track item) {\n <mat-option [value]=\"item.key\">\n {{ item.value }}\n </mat-option>\n }\n </mat-select>\n </mat-form-field>\n @if (course.type && course.type.toString() !== 'ONLINE') {\n <mat-form-field appearance=\"fill\" ngModelGroup=\"classroom\">\n <mat-label>\u6559\u5BA4</mat-label>\n <mat-select\n name=\"id\"\n [(ngModel)]=\"course.classroom\"\n [compareWith]=\"compareClassroom\"\n (selectionChange)=\"onClassroomSelected($event)\"\n required\n >\n <mat-optgroup label=\"\u81EA\u6709\u6559\u5BA4\">\n @for (classroom of classrooms; track classroom) {\n <mat-option [value]=\"classroom\">\n <div class=\"flex gap-3\">\n <span>{{ classroom.name }}</span>\n <span> {{ classroom.address }}</span>\n </div>\n </mat-option>\n }\n </mat-optgroup>\n <mat-optgroup label=\"\u5DF2\u8D2D\u6559\u5BA4\">\n @for (item of bookedClassrooms; track item) {\n <mat-option [value]=\"item\">\n <div class=\"flex flex-col py-3\">\n <span class=\"font-bold\"> {{ item.address }} {{ item.room }}\u5BA4</span>\n <span class=\"text-md font-thin\"> {{ item.startAt }} - {{ item.endAt }}</span>\n </div>\n </mat-option>\n }\n </mat-optgroup>\n </mat-select>\n </mat-form-field>\n }\n </form>\n </div>\n <div>\n <button mat-flat-button type=\"primary\" (click)=\"update()\">\u4FDD\u5B58</button>\n </div>\n }\n</rolatech-course-manage-content>\n", styles: ["mat-form-field{width:100%}\n"] }]
138
+ }] });
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"course-manage-info.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular-course/src/lib/pages/course-manage/course-manage-info/course-manage-info.component.ts","../../../../../../../../libs/angular-course/src/lib/pages/course-manage/course-manage-info/course-manage-info.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, inject } from '@angular/core';\nimport { BaseComponent, ToolbarComponent } from '@rolatech/angular-components';\nimport { clone } from 'lodash';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatOptionModule } from '@angular/material/core';\nimport { MatSelectModule } from '@angular/material/select';\nimport { TextFieldModule } from '@angular/cdk/text-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { FormsModule } from '@angular/forms';\nimport { CourseManageContentComponent } from '../course-manage-content/course-manage-content.component';\nimport { Course, CourseCategory, CourseStatus } from '../../../interfaces';\nimport { CategoryService, CourseService } from '../../../services';\nimport { BookingService } from '../../../services/booking.service';\nimport { InstructorService } from '../../../services/instructor.service';\n\n@Component({\n  selector: 'rolatech-course-manage-info',\n  templateUrl: './course-manage-info.component.html',\n  styleUrls: ['./course-manage-info.component.scss'],\n  standalone: true,\n  imports: [\n    FormsModule,\n    MatFormFieldModule,\n    MatInputModule,\n    TextFieldModule,\n    MatSelectModule,\n    MatOptionModule,\n    MatButtonModule,\n    ToolbarComponent,\n    CourseManageContentComponent,\n  ],\n})\nexport class CourseManageInfoComponent extends BaseComponent implements OnInit {\n  courseService = inject(CourseService);\n  instructorService = inject(InstructorService);\n  categoryService = inject(CategoryService);\n  bookingService = inject(BookingService);\n\n  isLoading = false;\n  course!: Course;\n  status = CourseStatus;\n  // courseType: any = CourseType;\n  selectedCategoyIds!: string[];\n  selectedClassroom: any;\n  categories!: CourseCategory[];\n  classrooms: any;\n  bookedClassrooms: any;\n  courseType = [\n    {\n      key: 'ONLINE',\n      value: '线上',\n    },\n    {\n      key: 'OFFLINE',\n      value: '线下',\n    },\n    {\n      key: 'MIXED',\n      value: '混合',\n    },\n  ];\n\n  ngOnInit(): void {\n    this.id = this.route.parent?.snapshot.paramMap.get('id') as string;\n    this.find();\n    this.categoryService.find({}).subscribe({\n      next: (res: any) => {\n        this.categories = res.data;\n      },\n    });\n    this.findClassrooms();\n    this.findBookedClassrooms();\n  }\n  findBookedClassrooms() {\n    this.bookingService.items({ status: 'paid' }).subscribe({\n      next: (res: any) => {\n        this.bookedClassrooms = res.data;\n      },\n    });\n  }\n\n  findClassrooms() {\n    this.instructorService.findMyClassrooms({}).subscribe({\n      next: (res) => {\n        this.classrooms = res.data;\n      },\n    });\n  }\n  find() {\n    this.courseService.get(this.id).subscribe({\n      next: (res: any) => {\n        this.course = res.data;\n      },\n    });\n  }\n  compareFn(o1: CourseCategory, o2: CourseCategory) {\n    return o1.id === o2?.id;\n  }\n  compareClassroom(o1: any, o2: any) {\n    if (o1.room) {\n      return (\n        o1.latitude === o2?.latitude &&\n        o1.longitude === o2?.longitude &&\n        o1.room === o2.room &&\n        o1.startAt === o2.startAt &&\n        o1.endAt === o2.endAt\n      );\n    } else {\n      return o1.latitude === o2?.latitude && o1.longitude === o2?.longitude;\n    }\n  }\n  onSelectionChange(event: any) {\n    this.selectedCategoyIds = event.value;\n  }\n  onClassroomSelected(event: any) {\n    this.selectedClassroom = event.value;\n  }\n  update() {\n    const { name, description, categories, type, classroom } = this.course;\n    const finalClassroom = clone(classroom);\n    finalClassroom.address = classroom.room ? classroom.address + ' ' + classroom.room + '室' : classroom.address;\n    const data = {\n      name,\n      description,\n      categories,\n      type,\n      classroom: finalClassroom,\n    };\n    this.courseService.update(this.id, data).subscribe({\n      next: (res) => {\n        this.snackBarService.open('保存成功');\n        // this.router.navigate([`../${res.data.id}`], { relativeTo: this.route });\n      },\n      error: (e) => {\n        this.snackBarService.open(e.message);\n      },\n    });\n  }\n}\n","<rolatech-course-manage-content>\n  <rolatech-toolbar title=\"基本信息\" class=\"hidden sm:block\" divider></rolatech-toolbar>\n  @if (course) {\n    <div class=\"flex flex-col\">\n      <form #courseForm=\"ngForm\" (ngSubmit)=\"update()\">\n        <mat-form-field appearance=\"fill\">\n          <mat-label> 课程名称 </mat-label>\n          <input matInput type=\"text\" [(ngModel)]=\"course.name\" name=\"name\" required />\n        </mat-form-field>\n        <mat-form-field appearance=\"fill\">\n          <mat-label> 课程描述 </mat-label>\n          <textarea\n            matInput\n            type=\"text\"\n            [(ngModel)]=\"course.description\"\n            name=\"description\"\n            required\n            cdkTextareaAutosize\n            cdkAutosizeMinRows=\"3\"\n          ></textarea>\n        </mat-form-field>\n        <mat-form-field appearance=\"fill\" ngModelGroup=\"categories\" required>\n          <mat-label>分类</mat-label>\n          <mat-select\n            multiple\n            name=\"id\"\n            [compareWith]=\"compareFn\"\n            [(ngModel)]=\"course.categories\"\n            #select=\"matSelect\"\n            (selectionChange)=\"onSelectionChange($event)\"\n          >\n            @for (item of categories; track item) {\n              <mat-option [value]=\"item\">\n                {{ item.name }}\n              </mat-option>\n            }\n          </mat-select>\n        </mat-form-field>\n        <mat-form-field appearance=\"fill\">\n          <mat-label>类型</mat-label>\n          <mat-select name=\"type\" [(ngModel)]=\"course.type\" required>\n            @for (item of courseType; track item) {\n              <mat-option [value]=\"item.key\">\n                {{ item.value }}\n              </mat-option>\n            }\n          </mat-select>\n        </mat-form-field>\n        @if (course.type && course.type.toString() !== 'ONLINE') {\n          <mat-form-field appearance=\"fill\" ngModelGroup=\"classroom\">\n            <mat-label>教室</mat-label>\n            <mat-select\n              name=\"id\"\n              [(ngModel)]=\"course.classroom\"\n              [compareWith]=\"compareClassroom\"\n              (selectionChange)=\"onClassroomSelected($event)\"\n              required\n            >\n              <mat-optgroup label=\"自有教室\">\n                @for (classroom of classrooms; track classroom) {\n                  <mat-option [value]=\"classroom\">\n                    <div class=\"flex gap-3\">\n                      <span>{{ classroom.name }}</span>\n                      <span> {{ classroom.address }}</span>\n                    </div>\n                  </mat-option>\n                }\n              </mat-optgroup>\n              <mat-optgroup label=\"已购教室\">\n                @for (item of bookedClassrooms; track item) {\n                  <mat-option [value]=\"item\">\n                    <div class=\"flex flex-col py-3\">\n                      <span class=\"font-bold\"> {{ item.address }} {{ item.room }}室</span>\n                      <span class=\"text-md font-thin\"> {{ item.startAt }} - {{ item.endAt }}</span>\n                    </div>\n                  </mat-option>\n                }\n              </mat-optgroup>\n            </mat-select>\n          </mat-form-field>\n        }\n      </form>\n    </div>\n    <div>\n      <button mat-flat-button type=\"primary\" (click)=\"update()\">保存</button>\n    </div>\n  }\n</rolatech-course-manage-content>\n"]}
@@ -0,0 +1,68 @@
1
+ import { Component, inject } from '@angular/core';
2
+ import { MatButtonModule } from '@angular/material/button';
3
+ import { MatDialog } from '@angular/material/dialog';
4
+ import { MatSnackBar } from '@angular/material/snack-bar';
5
+ import { RouterLink, RouterLinkActive, RouterOutlet, ActivatedRoute } from '@angular/router';
6
+ import { CourseManageContentComponent } from '../course-manage-content/course-manage-content.component';
7
+ import { CourseService } from '../../../services';
8
+ import { CourseStatus } from '../../../interfaces';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "@angular/material/button";
11
+ export class CourseManageLayoutComponent {
12
+ constructor() {
13
+ this.submitted = false;
14
+ this.accepted = false;
15
+ this.isDraft = false;
16
+ this.route = inject(ActivatedRoute);
17
+ this.courseService = inject(CourseService);
18
+ this.dialog = inject(MatDialog);
19
+ this.snackBar = inject(MatSnackBar);
20
+ this.id = this.route.parent?.snapshot.paramMap.get('id');
21
+ }
22
+ ngOnInit() {
23
+ this.get();
24
+ }
25
+ get() {
26
+ this.courseService.get(this.id).subscribe({
27
+ next: (res) => {
28
+ this.course = res.data;
29
+ this.updateStatus();
30
+ },
31
+ });
32
+ }
33
+ updateStatus() {
34
+ this.isDraft = this.course.status.toString() === 'DRAFT' || this.course.status.toString() === 'PENDING';
35
+ this.accepted = this.course.status.toString() === 'ACCEPTED';
36
+ }
37
+ submitForReview() {
38
+ this.courseService.review(this.id).subscribe({
39
+ next: (res) => {
40
+ this.snackBar.open('提交成功, 等待审核');
41
+ this.course.status = CourseStatus['审核中'];
42
+ this.updateStatus();
43
+ },
44
+ error: (error) => {
45
+ this.snackBar.open(error.message);
46
+ },
47
+ });
48
+ }
49
+ publish() {
50
+ this.courseService.publish(this.id).subscribe({
51
+ next: (res) => {
52
+ this.snackBar.open('发布成功');
53
+ this.course.status = CourseStatus['已发布'];
54
+ this.updateStatus();
55
+ },
56
+ error: (error) => {
57
+ this.snackBar.open(error.message);
58
+ },
59
+ });
60
+ }
61
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageLayoutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
62
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.1", type: CourseManageLayoutComponent, isStandalone: true, selector: "rolatech-course-manage-layout", ngImport: i0, template: "<div class=\"flex flex-col md:flex-row m-auto\">\n <div\n class=\"min-w-[256px] px-3 flex flex-row md:flex-col md:h-full items-center md:items-start h-16 overflow-x-scroll overflow-y-hidden scrollbar-hide whitespace-pre\"\n >\n <div class=\"flex flex-row md:flex-col md:w-full\">\n <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">\u8BFE\u7A0B\u4FE1\u606F</div>\n <a routerLink=\"./info\" routerLinkActive=\"manage-active\" class=\"p-3\">\u57FA\u672C\u4FE1\u606F</a>\n <a routerLink=\"./media\" routerLinkActive=\"manage-active\" class=\"p-3\">\u56FE\u7247</a>\n <a routerLink=\"./details\" routerLinkActive=\"manage-active\" class=\"p-3\">\u8BE6\u60C5</a>\n </div>\n <div class=\"flex flex-row md:flex-col md:w-full\">\n <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">\u8BFE\u7A0B\u5185\u5BB9</div>\n <a routerLink=\"./section\" routerLinkActive=\"manage-active\" class=\"p-3\">\u7AE0\u8282</a>\n <a routerLink=\"./schedule\" routerLinkActive=\"manage-active\" class=\"p-3\">\u8BFE\u8868</a>\n </div>\n <div class=\"flex flex-row md:flex-col md:w-full\">\n <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">\u5176\u4ED6\u4FE1\u606F</div>\n <a routerLink=\"./pricing\" routerLinkActive=\"manage-active\" class=\"p-3\">\u4EF7\u683C</a>\n </div>\n\n @if (isDraft) {\n <div class=\"md:mt-6 ml-3 md:ml-2\">\n <!-- <button mat-stroked-button (click)=\"publish()\">\u9884\u89C8</button> -->\n <button mat-flat-button (click)=\"submitForReview()\">\u63D0\u4EA4\u5BA1\u6838</button>\n </div>\n }\n\n @if (accepted) {\n <div class=\"md:mt-6 md:ml-2\">\n <button mat-flat-button color=\"primary\" (click)=\"publish()\">\u53D1\u5E03\u8BFE\u7A0B</button>\n </div>\n }\n </div>\n <div class=\"w-full\">\n <router-outlet></router-outlet>\n </div>\n</div>\n", styles: [".manage-active{background-color:#0000000d;box-shadow:5px 0 #000 inset;font-weight:600}@media (max-width: 768px){.manage-active{box-shadow:inset 0 -4px #000}}\n"], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i1.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "directive", type: RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }] }); }
63
+ }
64
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: CourseManageLayoutComponent, decorators: [{
65
+ type: Component,
66
+ args: [{ selector: 'rolatech-course-manage-layout', standalone: true, imports: [RouterLink, RouterLinkActive, MatButtonModule, RouterOutlet, CourseManageContentComponent], template: "<div class=\"flex flex-col md:flex-row m-auto\">\n <div\n class=\"min-w-[256px] px-3 flex flex-row md:flex-col md:h-full items-center md:items-start h-16 overflow-x-scroll overflow-y-hidden scrollbar-hide whitespace-pre\"\n >\n <div class=\"flex flex-row md:flex-col md:w-full\">\n <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">\u8BFE\u7A0B\u4FE1\u606F</div>\n <a routerLink=\"./info\" routerLinkActive=\"manage-active\" class=\"p-3\">\u57FA\u672C\u4FE1\u606F</a>\n <a routerLink=\"./media\" routerLinkActive=\"manage-active\" class=\"p-3\">\u56FE\u7247</a>\n <a routerLink=\"./details\" routerLinkActive=\"manage-active\" class=\"p-3\">\u8BE6\u60C5</a>\n </div>\n <div class=\"flex flex-row md:flex-col md:w-full\">\n <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">\u8BFE\u7A0B\u5185\u5BB9</div>\n <a routerLink=\"./section\" routerLinkActive=\"manage-active\" class=\"p-3\">\u7AE0\u8282</a>\n <a routerLink=\"./schedule\" routerLinkActive=\"manage-active\" class=\"p-3\">\u8BFE\u8868</a>\n </div>\n <div class=\"flex flex-row md:flex-col md:w-full\">\n <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">\u5176\u4ED6\u4FE1\u606F</div>\n <a routerLink=\"./pricing\" routerLinkActive=\"manage-active\" class=\"p-3\">\u4EF7\u683C</a>\n </div>\n\n @if (isDraft) {\n <div class=\"md:mt-6 ml-3 md:ml-2\">\n <!-- <button mat-stroked-button (click)=\"publish()\">\u9884\u89C8</button> -->\n <button mat-flat-button (click)=\"submitForReview()\">\u63D0\u4EA4\u5BA1\u6838</button>\n </div>\n }\n\n @if (accepted) {\n <div class=\"md:mt-6 md:ml-2\">\n <button mat-flat-button color=\"primary\" (click)=\"publish()\">\u53D1\u5E03\u8BFE\u7A0B</button>\n </div>\n }\n </div>\n <div class=\"w-full\">\n <router-outlet></router-outlet>\n </div>\n</div>\n", styles: [".manage-active{background-color:#0000000d;box-shadow:5px 0 #000 inset;font-weight:600}@media (max-width: 768px){.manage-active{box-shadow:inset 0 -4px #000}}\n"] }]
67
+ }], ctorParameters: () => [] });
68
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"course-manage-layout.component.js","sourceRoot":"","sources":["../../../../../../../../libs/angular-course/src/lib/pages/course-manage/course-manage-layout/course-manage-layout.component.ts","../../../../../../../../libs/angular-course/src/lib/pages/course-manage/course-manage-layout/course-manage-layout.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, inject } from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog } from '@angular/material/dialog';\nimport { MatSnackBar } from '@angular/material/snack-bar';\nimport { RouterLink, RouterLinkActive, RouterOutlet, ActivatedRoute } from '@angular/router';\nimport { CourseManageContentComponent } from '../course-manage-content/course-manage-content.component';\nimport { CourseService } from '../../../services';\nimport { Course, CourseStatus } from '../../../interfaces';\n\n@Component({\n  selector: 'rolatech-course-manage-layout',\n  standalone: true,\n  templateUrl: './course-manage-layout.component.html',\n  styleUrl: './course-manage-layout.component.scss',\n  imports: [RouterLink, RouterLinkActive, MatButtonModule, RouterOutlet, CourseManageContentComponent],\n})\nexport class CourseManageLayoutComponent implements OnInit {\n  id: string;\n  course!: Course;\n  submitted = false;\n  accepted = false;\n  isDraft = false;\n  route = inject(ActivatedRoute);\n  courseService = inject(CourseService);\n  dialog = inject(MatDialog);\n  snackBar = inject(MatSnackBar);\n\n  constructor() {\n    this.id = this.route.parent?.snapshot.paramMap.get('id') as string;\n  }\n  ngOnInit(): void {\n    this.get();\n  }\n  get() {\n    this.courseService.get(this.id).subscribe({\n      next: (res: any) => {\n        this.course = res.data;\n        this.updateStatus();\n      },\n    });\n  }\n  updateStatus() {\n    this.isDraft = this.course.status.toString() === 'DRAFT' || this.course.status.toString() === 'PENDING';\n    this.accepted = this.course.status.toString() === 'ACCEPTED';\n  }\n  submitForReview() {\n    this.courseService.review(this.id).subscribe({\n      next: (res) => {\n        this.snackBar.open('提交成功, 等待审核');\n\n        this.course.status = CourseStatus['审核中' as keyof typeof CourseStatus];\n        this.updateStatus();\n      },\n      error: (error) => {\n        this.snackBar.open(error.message);\n      },\n    });\n  }\n  publish() {\n    this.courseService.publish(this.id).subscribe({\n      next: (res) => {\n        this.snackBar.open('发布成功');\n        this.course.status = CourseStatus['已发布' as keyof typeof CourseStatus];\n        this.updateStatus();\n      },\n      error: (error) => {\n        this.snackBar.open(error.message);\n      },\n    });\n  }\n}\n","<div class=\"flex flex-col md:flex-row m-auto\">\n  <div\n    class=\"min-w-[256px] px-3 flex flex-row md:flex-col md:h-full items-center md:items-start h-16 overflow-x-scroll overflow-y-hidden scrollbar-hide whitespace-pre\"\n  >\n    <div class=\"flex flex-row md:flex-col md:w-full\">\n      <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">课程信息</div>\n      <a routerLink=\"./info\" routerLinkActive=\"manage-active\" class=\"p-3\">基本信息</a>\n      <a routerLink=\"./media\" routerLinkActive=\"manage-active\" class=\"p-3\">图片</a>\n      <a routerLink=\"./details\" routerLinkActive=\"manage-active\" class=\"p-3\">详情</a>\n    </div>\n    <div class=\"flex flex-row md:flex-col md:w-full\">\n      <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">课程内容</div>\n      <a routerLink=\"./section\" routerLinkActive=\"manage-active\" class=\"p-3\">章节</a>\n      <a routerLink=\"./schedule\" routerLinkActive=\"manage-active\" class=\"p-3\">课表</a>\n    </div>\n    <div class=\"flex flex-row md:flex-col md:w-full\">\n      <div class=\"hidden md:flex text-xl font-bold h-14 items-center px-2\">其他信息</div>\n      <a routerLink=\"./pricing\" routerLinkActive=\"manage-active\" class=\"p-3\">价格</a>\n    </div>\n\n    @if (isDraft) {\n      <div class=\"md:mt-6 ml-3 md:ml-2\">\n        <!-- <button mat-stroked-button (click)=\"publish()\">预览</button> -->\n        <button mat-flat-button (click)=\"submitForReview()\">提交审核</button>\n      </div>\n    }\n\n    @if (accepted) {\n      <div class=\"md:mt-6 md:ml-2\">\n        <button mat-flat-button color=\"primary\" (click)=\"publish()\">发布课程</button>\n      </div>\n    }\n  </div>\n  <div class=\"w-full\">\n    <router-outlet></router-outlet>\n  </div>\n</div>\n"]}