@dssp/supervision 1.0.0-alpha.24 → 1.0.0-alpha.26

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 (70) hide show
  1. package/dist-client/graphql/building-inspection.js +28 -0
  2. package/dist-client/graphql/building-inspection.js.map +1 -1
  3. package/dist-client/pages/building-inspection/building-inspection-detail-checklist.d.ts +9 -0
  4. package/dist-client/pages/building-inspection/building-inspection-detail-checklist.js +114 -4
  5. package/dist-client/pages/building-inspection/building-inspection-detail-checklist.js.map +1 -1
  6. package/dist-client/pages/building-inspection/building-inspection-list.d.ts +6 -0
  7. package/dist-client/pages/building-inspection/building-inspection-list.js +47 -8
  8. package/dist-client/pages/building-inspection/building-inspection-list.js.map +1 -1
  9. package/dist-client/pages/building-inspection/component/image-preview-popup.d.ts +15 -0
  10. package/dist-client/pages/building-inspection/component/image-preview-popup.js +351 -0
  11. package/dist-client/pages/building-inspection/component/image-preview-popup.js.map +1 -0
  12. package/dist-client/pages/building-inspection/component/inspection-document/inspection-request-document.d.ts +20 -0
  13. package/dist-client/pages/building-inspection/component/inspection-document/inspection-request-document.js +436 -0
  14. package/dist-client/pages/building-inspection/component/inspection-document/inspection-request-document.js.map +1 -0
  15. package/dist-client/pages/building-inspection/component/inspection-document/inspection-result-notification.d.ts +27 -0
  16. package/dist-client/pages/building-inspection/component/inspection-document/inspection-result-notification.js +453 -0
  17. package/dist-client/pages/building-inspection/component/inspection-document/inspection-result-notification.js.map +1 -0
  18. package/dist-client/pages/building-inspection/component/inspection-document/name-list-upload-popup.d.ts +21 -0
  19. package/dist-client/pages/building-inspection/component/inspection-document/name-list-upload-popup.js +327 -0
  20. package/dist-client/pages/building-inspection/component/inspection-document/name-list-upload-popup.js.map +1 -0
  21. package/dist-client/pages/building-inspection/component/inspection-document/photo-album-popup.d.ts +27 -0
  22. package/dist-client/pages/building-inspection/component/inspection-document/photo-album-popup.js +401 -0
  23. package/dist-client/pages/building-inspection/component/inspection-document/photo-album-popup.js.map +1 -0
  24. package/dist-client/pages/building-inspection/component/photo-album-popup.d.ts +41 -0
  25. package/dist-client/pages/building-inspection/component/photo-album-popup.js +406 -0
  26. package/dist-client/pages/building-inspection/component/photo-album-popup.js.map +1 -0
  27. package/dist-client/pages/building-inspection/inspection-create-popup.js +17 -5
  28. package/dist-client/pages/building-inspection/inspection-create-popup.js.map +1 -1
  29. package/dist-client/pages/building-inspection-grid/building-inspection-grid-detail.d.ts +1 -0
  30. package/dist-client/pages/building-inspection-grid/building-inspection-grid-detail.js +47 -3
  31. package/dist-client/pages/building-inspection-grid/building-inspection-grid-detail.js.map +1 -1
  32. package/dist-client/pages/building-inspection-grid/component/grid-inspection-create-popup.js +8 -2
  33. package/dist-client/pages/building-inspection-grid/component/grid-inspection-create-popup.js.map +1 -1
  34. package/dist-client/pages/checklist/attachment-list-popup.d.ts +1 -0
  35. package/dist-client/pages/checklist/attachment-list-popup.js +244 -68
  36. package/dist-client/pages/checklist/attachment-list-popup.js.map +1 -1
  37. package/dist-client/pages/checklist/checklist-view.d.ts +5 -0
  38. package/dist-client/pages/checklist/checklist-view.js +64 -14
  39. package/dist-client/pages/checklist/checklist-view.js.map +1 -1
  40. package/dist-client/pages/checklist/file-preview-popup.js +0 -1
  41. package/dist-client/pages/checklist/file-preview-popup.js.map +1 -1
  42. package/dist-client/pages/checklist/inspection-tab-popup.js +3 -1
  43. package/dist-client/pages/checklist/inspection-tab-popup.js.map +1 -1
  44. package/dist-client/pages/checklist-template/checklist-template-list.js +1 -1
  45. package/dist-client/pages/checklist-template/checklist-template-list.js.map +1 -1
  46. package/dist-client/route.d.ts +1 -1
  47. package/dist-client/tsconfig.tsbuildinfo +1 -1
  48. package/dist-server/service/building-inspection/building-inspection-history.d.ts +1 -0
  49. package/dist-server/service/building-inspection/building-inspection-history.js +5 -0
  50. package/dist-server/service/building-inspection/building-inspection-history.js.map +1 -1
  51. package/dist-server/service/building-inspection/building-inspection-mutation.js +48 -21
  52. package/dist-server/service/building-inspection/building-inspection-mutation.js.map +1 -1
  53. package/dist-server/service/building-inspection/building-inspection-type.d.ts +6 -0
  54. package/dist-server/service/building-inspection/building-inspection-type.js +21 -0
  55. package/dist-server/service/building-inspection/building-inspection-type.js.map +1 -1
  56. package/dist-server/service/building-inspection/building-inspection.js +1 -1
  57. package/dist-server/service/building-inspection/building-inspection.js.map +1 -1
  58. package/dist-server/service/building-inspection/index.d.ts +1 -1
  59. package/dist-server/service/checklist/checklist-query.d.ts +6 -0
  60. package/dist-server/service/checklist/checklist-query.js +63 -0
  61. package/dist-server/service/checklist/checklist-query.js.map +1 -1
  62. package/dist-server/service/checklist/checklist.d.ts +20 -0
  63. package/dist-server/service/checklist/checklist.js +62 -1
  64. package/dist-server/service/checklist/checklist.js.map +1 -1
  65. package/dist-server/service/checklist-item/checklist-item-query.d.ts +1 -1
  66. package/dist-server/service/checklist-item/checklist-item-query.js +5 -3
  67. package/dist-server/service/checklist-item/checklist-item-query.js.map +1 -1
  68. package/dist-server/service/index.d.ts +1 -1
  69. package/dist-server/tsconfig.tsbuildinfo +1 -1
  70. package/package.json +2 -2
@@ -0,0 +1,401 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import '@material/web/button/elevated-button.js';
3
+ import '@material/web/icon/icon.js';
4
+ import '@material/web/tabs/tabs.js';
5
+ import '@material/web/tabs/primary-tab.js';
6
+ import gql from 'graphql-tag';
7
+ import { client } from '@operato/graphql';
8
+ import { css, html, LitElement } from 'lit';
9
+ import { customElement, property, state } from 'lit/decorators.js';
10
+ import { ButtonContainerStyles, ScrollbarStyles } from '@operato/styles';
11
+ import { notify, openPopup } from '@operato/layout';
12
+ import '../image-preview-popup';
13
+ import { BuildingInspectionStatus } from '../../building-inspection-list';
14
+ export var PhotoTab;
15
+ (function (PhotoTab) {
16
+ PhotoTab["CONSTRUCTOR"] = "constructor";
17
+ PhotoTab["SUPERVISOR"] = "supervisor";
18
+ })(PhotoTab || (PhotoTab = {}));
19
+ let PhotoAlbumPopup = class PhotoAlbumPopup extends LitElement {
20
+ constructor() {
21
+ super(...arguments);
22
+ this.checklistId = '';
23
+ this.activeTab = PhotoTab.CONSTRUCTOR;
24
+ this.constructorFiles = [];
25
+ this.supervisorFiles = [];
26
+ }
27
+ render() {
28
+ const currentFiles = this.activeTab === PhotoTab.CONSTRUCTOR ? this.constructorFiles : this.supervisorFiles;
29
+ const constructorCount = this.constructorFiles.length;
30
+ const supervisorCount = this.supervisorFiles.length;
31
+ return html `
32
+ <div body>
33
+ <div class="tabs-container">
34
+ <md-tabs>
35
+ <md-primary-tab
36
+ ?active=${this.activeTab === PhotoTab.CONSTRUCTOR}
37
+ @click=${() => this._setActiveTab(PhotoTab.CONSTRUCTOR)}
38
+ >
39
+ 시공자 (${constructorCount})
40
+ </md-primary-tab>
41
+ <md-primary-tab
42
+ ?active=${this.activeTab === PhotoTab.SUPERVISOR}
43
+ @click=${() => this._setActiveTab(PhotoTab.SUPERVISOR)}
44
+ >
45
+ 감리자 (${supervisorCount})
46
+ </md-primary-tab>
47
+ </md-tabs>
48
+ </div>
49
+
50
+ <div class="tab-content">
51
+ <div class="photo-count">
52
+ ${this.activeTab === PhotoTab.CONSTRUCTOR ? '시공자' : '감리자'} 첨부파일: ${currentFiles.length}건
53
+ </div>
54
+
55
+ ${currentFiles.length > 0
56
+ ? html `
57
+ <div class="photo-grid">
58
+ ${currentFiles.map((file, index) => {
59
+ var _a;
60
+ return html `
61
+ <div class="photo-item" @click=${() => this._openFilePreview(currentFiles, index)}>
62
+ ${this._isImageFile(file.fullpath)
63
+ ? html `<img
64
+ src="${file.fullpath}"
65
+ alt="${file.name}"
66
+ @error=${(e) => this._onImageError(e, file)}
67
+ />`
68
+ : html `
69
+ <div class="file-icon-container">
70
+ <md-icon class="file-icon">description</md-icon>
71
+ <div class="file-extension">${this._getFileExtension(file.fullpath)}</div>
72
+ </div>
73
+ `}
74
+ <div class="photo-overlay">
75
+ <div class="photo-info">
76
+ <div class="photo-name">${file.name}</div>
77
+ <div class="photo-creator">
78
+ <md-icon style="--md-icon-size: 10px;">account_circle</md-icon>
79
+ ${((_a = file.creator) === null || _a === void 0 ? void 0 : _a.name) || ''}
80
+ </div>
81
+ <div class="photo-date">${this._formatDate(file.createdAt)}</div>
82
+ </div>
83
+ </div>
84
+ </div>
85
+ `;
86
+ })}
87
+ </div>
88
+ `
89
+ : html `
90
+ <div class="empty-state">
91
+ <md-icon>attach_file</md-icon>
92
+ <div>${this.activeTab === PhotoTab.CONSTRUCTOR ? '시공자' : '감리자'}가 업로드한 첨부파일이 없습니다.</div>
93
+ </div>
94
+ `}
95
+ </div>
96
+ </div>
97
+
98
+ <div class="button-container">
99
+ <md-elevated-button @click=${this._close}> <md-icon slot="icon">close</md-icon>닫기 </md-elevated-button>
100
+ </div>
101
+ `;
102
+ }
103
+ async firstUpdated() {
104
+ await this._loadPhotos();
105
+ }
106
+ async _loadPhotos() {
107
+ var _a;
108
+ try {
109
+ const response = await client.query({
110
+ query: gql `
111
+ query ChecklistPhotos($checklistId: String!) {
112
+ checklist(id: $checklistId) {
113
+ id
114
+ checklistItems {
115
+ id
116
+ name
117
+ checklistItemAttachments {
118
+ id
119
+ name
120
+ fullpath
121
+ description
122
+ creator {
123
+ id
124
+ name
125
+ email
126
+ }
127
+ createdAt
128
+ }
129
+ }
130
+ }
131
+ }
132
+ `,
133
+ variables: {
134
+ checklistId: this.checklistId
135
+ }
136
+ });
137
+ if (response.errors) {
138
+ notify({ message: '사진 목록을 불러오는데 실패했습니다.', level: 'error' });
139
+ return;
140
+ }
141
+ const checklist = response.data.checklist;
142
+ // 모든 체크리스트 아이템의 첨부파일을 수집 (이미지와 일반 파일 모두)
143
+ const allAttachments = ((_a = checklist.checklistItems) === null || _a === void 0 ? void 0 : _a.flatMap((item) => {
144
+ var _a;
145
+ return ((_a = item.checklistItemAttachments) === null || _a === void 0 ? void 0 : _a.map((attachment) => (Object.assign(Object.assign({}, attachment), { previewTitle: `[검사 항목] ${item.name}` })))) || [];
146
+ })) || [];
147
+ // 시공자와 감리자로 분류 (이메일 도메인이나 역할로 구분)
148
+ this.constructorFiles = allAttachments.filter(attachment => [BuildingInspectionStatus.WAIT, BuildingInspectionStatus.OVERALL_WAIT, BuildingInspectionStatus.FAIL].includes(attachment.description));
149
+ this.supervisorFiles = allAttachments.filter(attachment => [BuildingInspectionStatus.REQUEST, BuildingInspectionStatus.OVERALL_REQUEST].includes(attachment.description));
150
+ }
151
+ catch (error) {
152
+ console.error('사진 목록 로딩 중 오류 발생:', error);
153
+ notify({ message: '사진 목록을 불러오는데 실패했습니다.', level: 'error' });
154
+ }
155
+ }
156
+ _isImageFile(filepath) {
157
+ if (!filepath)
158
+ return false;
159
+ const imageExtensions = ['jpg', 'jpeg', 'png', 'gif', 'bmp', 'webp', 'svg'];
160
+ const extension = filepath.toLowerCase().split('.').pop();
161
+ return imageExtensions.includes(extension || '');
162
+ }
163
+ _getFileExtension(filepath) {
164
+ if (!filepath)
165
+ return '';
166
+ const extension = filepath.toLowerCase().split('.').pop();
167
+ return extension || 'file';
168
+ }
169
+ _setActiveTab(tab) {
170
+ this.activeTab = tab;
171
+ }
172
+ _openFilePreview(files, startIndex) {
173
+ const file = files[startIndex];
174
+ if (this._isImageFile(file.fullpath)) {
175
+ // 이미지들만 필터링해서 이미지 미리보기 팝업 열기
176
+ const imageFiles = files.filter(f => this._isImageFile(f.fullpath));
177
+ const imageIndex = imageFiles.findIndex(f => f.id === file.id);
178
+ openPopup(html ` <image-preview-popup .images=${imageFiles} .currentIndex=${Math.max(0, imageIndex)}></image-preview-popup> `, { backdrop: true, size: 'large', title: '사진 미리보기' });
179
+ }
180
+ else {
181
+ // 일반 파일은 새 탭에서 열기 또는 다운로드
182
+ window.open(file.fullpath, '_blank');
183
+ }
184
+ }
185
+ _onImageError(e, file) {
186
+ const target = e.target;
187
+ target.style.display = 'none';
188
+ // 오류 발생 시 기본 아이콘 표시
189
+ const container = target.parentElement;
190
+ container.innerHTML += `
191
+ <div style="display: flex; align-items: center; justify-content: center; width: 100%; height: 100%; background-color: #f5f5f5;">
192
+ <md-icon style="--md-icon-size: 48px; color: #ccc;">broken_image</md-icon>
193
+ </div>
194
+ `;
195
+ }
196
+ _formatDate(date) {
197
+ const _date = new Date(date);
198
+ if (isNaN(_date.getTime())) {
199
+ return '날짜 없음';
200
+ }
201
+ const year = _date.getFullYear();
202
+ const month = String(_date.getMonth() + 1).padStart(2, '0');
203
+ const day = String(_date.getDate()).padStart(2, '0');
204
+ const hours = String(_date.getHours()).padStart(2, '0');
205
+ const minutes = String(_date.getMinutes()).padStart(2, '0');
206
+ return `${year}.${month}.${day} ${hours}:${minutes}`;
207
+ }
208
+ _close() {
209
+ history.back();
210
+ }
211
+ };
212
+ PhotoAlbumPopup.styles = [
213
+ ButtonContainerStyles,
214
+ ScrollbarStyles,
215
+ css `
216
+ :host {
217
+ display: flex;
218
+ flex-direction: column;
219
+ background-color: #fff;
220
+ width: 100%;
221
+ max-height: 80vh;
222
+ }
223
+
224
+ div[body] {
225
+ flex: 1;
226
+ overflow-y: auto;
227
+ padding: 20px;
228
+ }
229
+
230
+ .tabs-container {
231
+ margin-bottom: 20px;
232
+ border-bottom: 2px solid #ddd;
233
+ }
234
+
235
+ md-tabs {
236
+ --md-primary-tab-container-color: transparent;
237
+ --md-primary-tab-active-indicator-color: #2196f3;
238
+ --md-primary-tab-label-text-color: #666;
239
+ --md-primary-tab-active-label-text-color: #2196f3;
240
+ }
241
+
242
+ .photo-grid {
243
+ display: grid;
244
+ grid-template-columns: repeat(auto-fill, minmax(150px, 1fr));
245
+ gap: 15px;
246
+ margin-bottom: 20px;
247
+ }
248
+
249
+ .photo-item {
250
+ position: relative;
251
+ aspect-ratio: 1;
252
+ border-radius: 8px;
253
+ overflow: hidden;
254
+ cursor: pointer;
255
+ transition: all 0.2s ease;
256
+ border: 2px solid #ddd;
257
+ }
258
+
259
+ .photo-item:hover {
260
+ transform: translateY(-2px);
261
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
262
+ border-color: #2196f3;
263
+ }
264
+
265
+ .photo-item img {
266
+ width: 100%;
267
+ height: 100%;
268
+ object-fit: cover;
269
+ }
270
+
271
+ .photo-overlay {
272
+ position: absolute;
273
+ bottom: 0;
274
+ left: 0;
275
+ right: 0;
276
+ background: linear-gradient(transparent, rgba(0, 0, 0, 0.8));
277
+ color: white;
278
+ padding: 10px 8px 8px;
279
+ font-size: 12px;
280
+ line-height: 1.2;
281
+ }
282
+
283
+ .photo-info {
284
+ display: flex;
285
+ flex-direction: column;
286
+ gap: 2px;
287
+ }
288
+
289
+ .photo-name {
290
+ font-weight: bold;
291
+ overflow: hidden;
292
+ text-overflow: ellipsis;
293
+ white-space: nowrap;
294
+ }
295
+
296
+ .photo-creator {
297
+ font-size: 10px;
298
+ opacity: 0.9;
299
+ display: flex;
300
+ align-items: center;
301
+ gap: 3px;
302
+ }
303
+
304
+ .photo-date {
305
+ font-size: 10px;
306
+ opacity: 0.8;
307
+ }
308
+
309
+ .file-icon-container {
310
+ width: 100%;
311
+ height: 100%;
312
+ display: flex;
313
+ flex-direction: column;
314
+ align-items: center;
315
+ justify-content: center;
316
+ background-color: #f5f5f5;
317
+ position: relative;
318
+ }
319
+
320
+ .file-icon {
321
+ --md-icon-size: 48px;
322
+ color: #666;
323
+ margin-bottom: 8px;
324
+ }
325
+
326
+ .file-extension {
327
+ font-size: 12px;
328
+ font-weight: bold;
329
+ color: #444;
330
+ background-color: #e0e0e0;
331
+ padding: 2px 6px;
332
+ border-radius: 3px;
333
+ text-transform: uppercase;
334
+ }
335
+
336
+ .empty-state {
337
+ text-align: center;
338
+ color: #666;
339
+ padding: 60px 20px;
340
+ }
341
+
342
+ .empty-state md-icon {
343
+ --md-icon-size: 64px;
344
+ color: #ccc;
345
+ margin-bottom: 16px;
346
+ }
347
+
348
+ .tab-content {
349
+ min-height: 300px;
350
+ }
351
+
352
+ .photo-count {
353
+ color: #666;
354
+ font-size: 14px;
355
+ margin-bottom: 15px;
356
+ padding: 10px;
357
+ background-color: #f5f5f5;
358
+ border-radius: 5px;
359
+ text-align: center;
360
+ }
361
+
362
+ h3 {
363
+ position: relative;
364
+ color: rgb(5, 149, 229);
365
+ font-size: 17px;
366
+ font-weight: 700;
367
+ margin: 0px 0px 20px 0px;
368
+ padding-top: 0px;
369
+ }
370
+
371
+ .button-container {
372
+ display: flex;
373
+ justify-content: center;
374
+ padding: 15px;
375
+ border-top: 1px solid #ddd;
376
+ background-color: #f9f9f9;
377
+ margin-left: unset;
378
+ }
379
+ `
380
+ ];
381
+ __decorate([
382
+ property({ type: String }),
383
+ __metadata("design:type", String)
384
+ ], PhotoAlbumPopup.prototype, "checklistId", void 0);
385
+ __decorate([
386
+ state(),
387
+ __metadata("design:type", String)
388
+ ], PhotoAlbumPopup.prototype, "activeTab", void 0);
389
+ __decorate([
390
+ state(),
391
+ __metadata("design:type", Array)
392
+ ], PhotoAlbumPopup.prototype, "constructorFiles", void 0);
393
+ __decorate([
394
+ state(),
395
+ __metadata("design:type", Array)
396
+ ], PhotoAlbumPopup.prototype, "supervisorFiles", void 0);
397
+ PhotoAlbumPopup = __decorate([
398
+ customElement('photo-album-popup')
399
+ ], PhotoAlbumPopup);
400
+ export { PhotoAlbumPopup };
401
+ //# sourceMappingURL=photo-album-popup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"photo-album-popup.js","sourceRoot":"","sources":["../../../../../client/pages/building-inspection/component/inspection-document/photo-album-popup.ts"],"names":[],"mappings":"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","sourcesContent":["import '@material/web/button/elevated-button.js'\nimport '@material/web/icon/icon.js'\nimport '@material/web/tabs/tabs.js'\nimport '@material/web/tabs/primary-tab.js'\n\nimport gql from 'graphql-tag'\nimport { client } from '@operato/graphql'\nimport { css, html, LitElement } from 'lit'\nimport { customElement, property, state } from 'lit/decorators.js'\nimport { ButtonContainerStyles, ScrollbarStyles } from '@operato/styles'\nimport { notify, openPopup } from '@operato/layout'\nimport '../image-preview-popup'\nimport { BuildingInspectionStatus } from '../../building-inspection-list'\n\nexport enum PhotoTab {\n CONSTRUCTOR = 'constructor',\n SUPERVISOR = 'supervisor'\n}\n\n@customElement('photo-album-popup')\nexport class PhotoAlbumPopup extends LitElement {\n static styles = [\n ButtonContainerStyles,\n ScrollbarStyles,\n css`\n :host {\n display: flex;\n flex-direction: column;\n background-color: #fff;\n width: 100%;\n max-height: 80vh;\n }\n\n div[body] {\n flex: 1;\n overflow-y: auto;\n padding: 20px;\n }\n\n .tabs-container {\n margin-bottom: 20px;\n border-bottom: 2px solid #ddd;\n }\n\n md-tabs {\n --md-primary-tab-container-color: transparent;\n --md-primary-tab-active-indicator-color: #2196f3;\n --md-primary-tab-label-text-color: #666;\n --md-primary-tab-active-label-text-color: #2196f3;\n }\n\n .photo-grid {\n display: grid;\n grid-template-columns: repeat(auto-fill, minmax(150px, 1fr));\n gap: 15px;\n margin-bottom: 20px;\n }\n\n .photo-item {\n position: relative;\n aspect-ratio: 1;\n border-radius: 8px;\n overflow: hidden;\n cursor: pointer;\n transition: all 0.2s ease;\n border: 2px solid #ddd;\n }\n\n .photo-item:hover {\n transform: translateY(-2px);\n box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\n border-color: #2196f3;\n }\n\n .photo-item img {\n width: 100%;\n height: 100%;\n object-fit: cover;\n }\n\n .photo-overlay {\n position: absolute;\n bottom: 0;\n left: 0;\n right: 0;\n background: linear-gradient(transparent, rgba(0, 0, 0, 0.8));\n color: white;\n padding: 10px 8px 8px;\n font-size: 12px;\n line-height: 1.2;\n }\n\n .photo-info {\n display: flex;\n flex-direction: column;\n gap: 2px;\n }\n\n .photo-name {\n font-weight: bold;\n overflow: hidden;\n text-overflow: ellipsis;\n white-space: nowrap;\n }\n\n .photo-creator {\n font-size: 10px;\n opacity: 0.9;\n display: flex;\n align-items: center;\n gap: 3px;\n }\n\n .photo-date {\n font-size: 10px;\n opacity: 0.8;\n }\n\n .file-icon-container {\n width: 100%;\n height: 100%;\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n background-color: #f5f5f5;\n position: relative;\n }\n\n .file-icon {\n --md-icon-size: 48px;\n color: #666;\n margin-bottom: 8px;\n }\n\n .file-extension {\n font-size: 12px;\n font-weight: bold;\n color: #444;\n background-color: #e0e0e0;\n padding: 2px 6px;\n border-radius: 3px;\n text-transform: uppercase;\n }\n\n .empty-state {\n text-align: center;\n color: #666;\n padding: 60px 20px;\n }\n\n .empty-state md-icon {\n --md-icon-size: 64px;\n color: #ccc;\n margin-bottom: 16px;\n }\n\n .tab-content {\n min-height: 300px;\n }\n\n .photo-count {\n color: #666;\n font-size: 14px;\n margin-bottom: 15px;\n padding: 10px;\n background-color: #f5f5f5;\n border-radius: 5px;\n text-align: center;\n }\n\n h3 {\n position: relative;\n color: rgb(5, 149, 229);\n font-size: 17px;\n font-weight: 700;\n margin: 0px 0px 20px 0px;\n padding-top: 0px;\n }\n\n .button-container {\n display: flex;\n justify-content: center;\n padding: 15px;\n border-top: 1px solid #ddd;\n background-color: #f9f9f9;\n margin-left: unset;\n }\n `\n ]\n\n @property({ type: String }) checklistId: string = ''\n\n @state() activeTab: PhotoTab = PhotoTab.CONSTRUCTOR\n @state() constructorFiles: any[] = []\n @state() supervisorFiles: any[] = []\n\n render() {\n const currentFiles = this.activeTab === PhotoTab.CONSTRUCTOR ? this.constructorFiles : this.supervisorFiles\n const constructorCount = this.constructorFiles.length\n const supervisorCount = this.supervisorFiles.length\n\n return html`\n <div body>\n <div class=\"tabs-container\">\n <md-tabs>\n <md-primary-tab\n ?active=${this.activeTab === PhotoTab.CONSTRUCTOR}\n @click=${() => this._setActiveTab(PhotoTab.CONSTRUCTOR)}\n >\n 시공자 (${constructorCount})\n </md-primary-tab>\n <md-primary-tab\n ?active=${this.activeTab === PhotoTab.SUPERVISOR}\n @click=${() => this._setActiveTab(PhotoTab.SUPERVISOR)}\n >\n 감리자 (${supervisorCount})\n </md-primary-tab>\n </md-tabs>\n </div>\n\n <div class=\"tab-content\">\n <div class=\"photo-count\">\n ${this.activeTab === PhotoTab.CONSTRUCTOR ? '시공자' : '감리자'} 첨부파일: ${currentFiles.length}건\n </div>\n\n ${currentFiles.length > 0\n ? html`\n <div class=\"photo-grid\">\n ${currentFiles.map(\n (file, index) => html`\n <div class=\"photo-item\" @click=${() => this._openFilePreview(currentFiles, index)}>\n ${this._isImageFile(file.fullpath)\n ? html`<img\n src=\"${file.fullpath}\"\n alt=\"${file.name}\"\n @error=${(e: Event) => this._onImageError(e, file)}\n />`\n : html`\n <div class=\"file-icon-container\">\n <md-icon class=\"file-icon\">description</md-icon>\n <div class=\"file-extension\">${this._getFileExtension(file.fullpath)}</div>\n </div>\n `}\n <div class=\"photo-overlay\">\n <div class=\"photo-info\">\n <div class=\"photo-name\">${file.name}</div>\n <div class=\"photo-creator\">\n <md-icon style=\"--md-icon-size: 10px;\">account_circle</md-icon>\n ${file.creator?.name || ''}\n </div>\n <div class=\"photo-date\">${this._formatDate(file.createdAt)}</div>\n </div>\n </div>\n </div>\n `\n )}\n </div>\n `\n : html`\n <div class=\"empty-state\">\n <md-icon>attach_file</md-icon>\n <div>${this.activeTab === PhotoTab.CONSTRUCTOR ? '시공자' : '감리자'}가 업로드한 첨부파일이 없습니다.</div>\n </div>\n `}\n </div>\n </div>\n\n <div class=\"button-container\">\n <md-elevated-button @click=${this._close}> <md-icon slot=\"icon\">close</md-icon>닫기 </md-elevated-button>\n </div>\n `\n }\n\n async firstUpdated() {\n await this._loadPhotos()\n }\n\n private async _loadPhotos() {\n try {\n const response = await client.query({\n query: gql`\n query ChecklistPhotos($checklistId: String!) {\n checklist(id: $checklistId) {\n id\n checklistItems {\n id\n name\n checklistItemAttachments {\n id\n name\n fullpath\n description\n creator {\n id\n name\n email\n }\n createdAt\n }\n }\n }\n }\n `,\n variables: {\n checklistId: this.checklistId\n }\n })\n\n if (response.errors) {\n notify({ message: '사진 목록을 불러오는데 실패했습니다.', level: 'error' })\n return\n }\n\n const checklist = response.data.checklist\n // 모든 체크리스트 아이템의 첨부파일을 수집 (이미지와 일반 파일 모두)\n const allAttachments =\n checklist.checklistItems?.flatMap(\n (item: any) =>\n item.checklistItemAttachments?.map((attachment: any) => ({\n ...attachment,\n previewTitle: `[검사 항목] ${item.name}`\n })) || []\n ) || []\n\n // 시공자와 감리자로 분류 (이메일 도메인이나 역할로 구분)\n this.constructorFiles = allAttachments.filter(attachment =>\n [BuildingInspectionStatus.WAIT, BuildingInspectionStatus.OVERALL_WAIT, BuildingInspectionStatus.FAIL].includes(\n attachment.description\n )\n )\n\n this.supervisorFiles = allAttachments.filter(attachment =>\n [BuildingInspectionStatus.REQUEST, BuildingInspectionStatus.OVERALL_REQUEST].includes(attachment.description)\n )\n } catch (error) {\n console.error('사진 목록 로딩 중 오류 발생:', error)\n notify({ message: '사진 목록을 불러오는데 실패했습니다.', level: 'error' })\n }\n }\n\n private _isImageFile(filepath: string): boolean {\n if (!filepath) return false\n const imageExtensions = ['jpg', 'jpeg', 'png', 'gif', 'bmp', 'webp', 'svg']\n const extension = filepath.toLowerCase().split('.').pop()\n return imageExtensions.includes(extension || '')\n }\n\n private _getFileExtension(filepath: string): string {\n if (!filepath) return ''\n const extension = filepath.toLowerCase().split('.').pop()\n return extension || 'file'\n }\n\n private _setActiveTab(tab: PhotoTab) {\n this.activeTab = tab\n }\n\n private _openFilePreview(files: any[], startIndex: number) {\n const file = files[startIndex]\n\n if (this._isImageFile(file.fullpath)) {\n // 이미지들만 필터링해서 이미지 미리보기 팝업 열기\n const imageFiles = files.filter(f => this._isImageFile(f.fullpath))\n const imageIndex = imageFiles.findIndex(f => f.id === file.id)\n\n openPopup(\n html` <image-preview-popup .images=${imageFiles} .currentIndex=${Math.max(0, imageIndex)}></image-preview-popup> `,\n { backdrop: true, size: 'large', title: '사진 미리보기' }\n )\n } else {\n // 일반 파일은 새 탭에서 열기 또는 다운로드\n window.open(file.fullpath, '_blank')\n }\n }\n\n private _onImageError(e: Event, file: any) {\n const target = e.target as HTMLImageElement\n target.style.display = 'none'\n\n // 오류 발생 시 기본 아이콘 표시\n const container = target.parentElement!\n container.innerHTML += `\n <div style=\"display: flex; align-items: center; justify-content: center; width: 100%; height: 100%; background-color: #f5f5f5;\">\n <md-icon style=\"--md-icon-size: 48px; color: #ccc;\">broken_image</md-icon>\n </div>\n `\n }\n\n private _formatDate(date: string | Date): string {\n const _date = new Date(date)\n\n if (isNaN(_date.getTime())) {\n return '날짜 없음'\n }\n\n const year = _date.getFullYear()\n const month = String(_date.getMonth() + 1).padStart(2, '0')\n const day = String(_date.getDate()).padStart(2, '0')\n const hours = String(_date.getHours()).padStart(2, '0')\n const minutes = String(_date.getMinutes()).padStart(2, '0')\n\n return `${year}.${month}.${day} ${hours}:${minutes}`\n }\n\n private _close() {\n history.back()\n }\n}\n"]}
@@ -0,0 +1,41 @@
1
+ import '@material/web/button/elevated-button.js';
2
+ import '@material/web/icon/icon.js';
3
+ import '@material/web/tabs/tabs.js';
4
+ import '@material/web/tabs/primary-tab.js';
5
+ import { LitElement } from 'lit';
6
+ import { User } from '@operato/shell';
7
+ import './image-preview-popup';
8
+ export declare enum PhotoTab {
9
+ CONSTRUCTOR = "constructor",
10
+ SUPERVISOR = "supervisor"
11
+ }
12
+ declare const PhotoAlbumPopup_base: (new (...args: any[]) => {
13
+ _storeUnsubscribe: import("redux").Unsubscribe;
14
+ connectedCallback(): void;
15
+ disconnectedCallback(): void;
16
+ stateChanged(_state: unknown): void;
17
+ readonly isConnected: boolean;
18
+ }) & typeof LitElement;
19
+ export declare class PhotoAlbumPopup extends PhotoAlbumPopup_base {
20
+ static styles: import("lit").CSSResult[];
21
+ checklistId: string;
22
+ title: string;
23
+ activeTab: PhotoTab;
24
+ constructorPhotos: any[];
25
+ supervisorPhotos: any[];
26
+ user: User;
27
+ render(): import("lit-html").TemplateResult<1>;
28
+ firstUpdated(): Promise<void>;
29
+ private _loadPhotos;
30
+ private _isImageFile;
31
+ private _isConstructor;
32
+ private _isSupervisor;
33
+ private _checkUserRole;
34
+ private _setActiveTab;
35
+ private _onTabChange;
36
+ private _openPhotoPreview;
37
+ private _onImageError;
38
+ private _formatDate;
39
+ private _close;
40
+ }
41
+ export {};