@dssp/dkpi 1.0.0-alpha.5 → 1.0.0-alpha.50

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 (128) hide show
  1. package/assets/favicon.ico +0 -0
  2. package/assets/manifest/apple-1024.png +0 -0
  3. package/assets/manifest/apple-120.png +0 -0
  4. package/assets/manifest/apple-152.png +0 -0
  5. package/assets/manifest/apple-167.png +0 -0
  6. package/assets/manifest/apple-180.png +0 -0
  7. package/assets/manifest/apple-touch-icon.png +0 -0
  8. package/assets/manifest/badge-128x128.png +0 -0
  9. package/assets/manifest/chrome-splashscreen-icon-384x384.png +0 -0
  10. package/assets/manifest/chrome-touch-icon-192x192.png +0 -0
  11. package/assets/manifest/icon-128x128.png +0 -0
  12. package/assets/manifest/icon-192x192.png +0 -0
  13. package/assets/manifest/icon-512x512.png +0 -0
  14. package/assets/manifest/icon-72x72.png +0 -0
  15. package/assets/manifest/icon-96x96.png +0 -0
  16. package/assets/manifest/image-metaog.png +0 -0
  17. package/assets/manifest/maskable_icon.png +0 -0
  18. package/assets/manifest/ms-icon-144x144.png +0 -0
  19. package/assets/manifest/ms-touch-icon-144x144-precomposed.png +0 -0
  20. package/assets/videos/intro.mp4 +0 -0
  21. package/dist-client/bootstrap.js +64 -4
  22. package/dist-client/bootstrap.js.map +1 -1
  23. package/dist-client/components/kpi-single-boxplot-chart.d.ts +24 -0
  24. package/dist-client/components/kpi-single-boxplot-chart.js +317 -0
  25. package/dist-client/components/kpi-single-boxplot-chart.js.map +1 -0
  26. package/dist-client/components/sv-pagenation-control.d.ts +18 -0
  27. package/dist-client/components/sv-pagenation-control.js +142 -0
  28. package/dist-client/components/sv-pagenation-control.js.map +1 -0
  29. package/dist-client/icons/menu-icons.d.ts +6 -0
  30. package/dist-client/icons/menu-icons.js +42 -0
  31. package/dist-client/icons/menu-icons.js.map +1 -1
  32. package/dist-client/menu.d.ts +23 -1
  33. package/dist-client/menu.js +57 -2
  34. package/dist-client/menu.js.map +1 -1
  35. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.d.ts +58 -0
  36. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js +731 -0
  37. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js.map +1 -0
  38. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.d.ts +23 -0
  39. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js +76 -0
  40. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js.map +1 -0
  41. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.d.ts +69 -0
  42. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js +385 -0
  43. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js.map +1 -0
  44. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.d.ts +12 -0
  45. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js +174 -0
  46. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js.map +1 -0
  47. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.d.ts +41 -0
  48. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js +191 -0
  49. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js.map +1 -0
  50. package/dist-client/pages/kpi-value/kpi-value-importer.d.ts +23 -0
  51. package/dist-client/pages/kpi-value/kpi-value-importer.js +93 -0
  52. package/dist-client/pages/kpi-value/kpi-value-importer.js.map +1 -0
  53. package/dist-client/pages/kpi-value/kpi-value-list-page.d.ts +72 -0
  54. package/dist-client/pages/kpi-value/kpi-value-list-page.js +465 -0
  55. package/dist-client/pages/kpi-value/kpi-value-list-page.js.map +1 -0
  56. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.d.ts +13 -0
  57. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.js +248 -0
  58. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.js.map +1 -0
  59. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.d.ts +10 -0
  60. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.js +229 -0
  61. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.js.map +1 -0
  62. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.d.ts +12 -0
  63. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.js +213 -0
  64. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.js.map +1 -0
  65. package/dist-client/pages/sv-project-complete.d.ts +22 -0
  66. package/dist-client/pages/sv-project-complete.js +213 -0
  67. package/dist-client/pages/sv-project-complete.js.map +1 -0
  68. package/dist-client/pages/sv-project-completed-list.d.ts +24 -0
  69. package/dist-client/pages/sv-project-completed-list.js +357 -0
  70. package/dist-client/pages/sv-project-completed-list.js.map +1 -0
  71. package/dist-client/pages/sv-project-detail.d.ts +13 -0
  72. package/dist-client/pages/sv-project-detail.js +494 -0
  73. package/dist-client/pages/sv-project-detail.js.map +1 -0
  74. package/dist-client/pages/sv-project-list.d.ts +157 -0
  75. package/dist-client/pages/sv-project-list.js +431 -0
  76. package/dist-client/pages/sv-project-list.js.map +1 -0
  77. package/dist-client/route.d.ts +1 -1
  78. package/dist-client/route.js +22 -1
  79. package/dist-client/route.js.map +1 -1
  80. package/dist-client/shared/complete-api.d.ts +5 -0
  81. package/dist-client/shared/complete-api.js +125 -0
  82. package/dist-client/shared/complete-api.js.map +1 -0
  83. package/dist-client/shared/func.d.ts +2 -0
  84. package/dist-client/shared/func.js +22 -0
  85. package/dist-client/shared/func.js.map +1 -0
  86. package/dist-client/themes/dark.css +24 -24
  87. package/dist-client/themes/light.css +23 -23
  88. package/dist-client/tsconfig.tsbuildinfo +1 -1
  89. package/dist-client/viewparts/menu-tools.d.ts +37 -1
  90. package/dist-client/viewparts/menu-tools.js +348 -15
  91. package/dist-client/viewparts/menu-tools.js.map +1 -1
  92. package/dist-server/index.d.ts +2 -0
  93. package/dist-server/index.js +5 -0
  94. package/dist-server/index.js.map +1 -1
  95. package/dist-server/migrations/index.d.ts +1 -0
  96. package/dist-server/migrations/index.js +12 -0
  97. package/dist-server/migrations/index.js.map +1 -0
  98. package/dist-server/scripts/calculate-kpi-scores.d.ts +10 -0
  99. package/dist-server/scripts/calculate-kpi-scores.js +271 -0
  100. package/dist-server/scripts/calculate-kpi-scores.js.map +1 -0
  101. package/dist-server/scripts/load-grade-data-migration.d.ts +10 -0
  102. package/dist-server/scripts/load-grade-data-migration.js +194 -0
  103. package/dist-server/scripts/load-grade-data-migration.js.map +1 -0
  104. package/dist-server/scripts/propagate-parent-kpi-values.d.ts +10 -0
  105. package/dist-server/scripts/propagate-parent-kpi-values.js +440 -0
  106. package/dist-server/scripts/propagate-parent-kpi-values.js.map +1 -0
  107. package/dist-server/service/index.d.ts +6 -0
  108. package/dist-server/service/index.js +21 -0
  109. package/dist-server/service/index.js.map +1 -0
  110. package/dist-server/service/kpi-metric-value/index.d.ts +4 -0
  111. package/dist-server/service/kpi-metric-value/index.js +8 -0
  112. package/dist-server/service/kpi-metric-value/index.js.map +1 -0
  113. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.d.ts +7 -0
  114. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.js +112 -0
  115. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.js.map +1 -0
  116. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.d.ts +7 -0
  117. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js +47 -0
  118. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js.map +1 -0
  119. package/dist-server/service/kpi-value/index.d.ts +3 -0
  120. package/dist-server/service/kpi-value/index.js +7 -0
  121. package/dist-server/service/kpi-value/index.js.map +1 -0
  122. package/dist-server/service/kpi-value/kpi-value-query.d.ts +7 -0
  123. package/dist-server/service/kpi-value/kpi-value-query.js +47 -0
  124. package/dist-server/service/kpi-value/kpi-value-query.js.map +1 -0
  125. package/dist-server/tsconfig.tsbuildinfo +1 -1
  126. package/package.json +62 -51
  127. package/schema.graphql +11497 -3215
  128. package/things-factory.config.js +7 -1
@@ -0,0 +1,357 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import '@material/web/icon/icon.js';
3
+ import '../components/kpi-single-boxplot-chart';
4
+ import { PageView, navigate } from '@operato/shell';
5
+ import { css, html } from 'lit';
6
+ import { customElement, state } from 'lit/decorators.js';
7
+ import { ScopedElementsMixin } from '@open-wc/scoped-elements';
8
+ import { client } from '@operato/graphql';
9
+ import gql from 'graphql-tag';
10
+ import { PROJECT_STATE, ProjectState } from './sv-project-list';
11
+ import '../components/sv-pagenation-control';
12
+ let SvProjectCompletedListPage = class SvProjectCompletedListPage extends ScopedElementsMixin(PageView) {
13
+ constructor() {
14
+ super(...arguments);
15
+ this.projectName = '';
16
+ this.projectList = [];
17
+ this.projectCount = 0;
18
+ this.currentPage = 1;
19
+ this.pageLimit = 20;
20
+ }
21
+ get context() {
22
+ return {
23
+ title: '완료 프로젝트'
24
+ };
25
+ }
26
+ get totalPages() {
27
+ return Math.max(1, Math.ceil((this.projectCount || 0) / this.pageLimit));
28
+ }
29
+ render() {
30
+ var _a;
31
+ return html `
32
+ <div header>
33
+ <div search>
34
+ <md-icon>manage_search</md-icon>
35
+ <input
36
+ type="search"
37
+ name="projectName"
38
+ placeholder="프로젝트명"
39
+ .value=${this.projectName}
40
+ @input=${this._onInputChange}
41
+ @keypress=${this._onKeypress}
42
+ />
43
+ </div>
44
+
45
+ <span class="spacer"></span>
46
+ <div count>총 ${this.projectCount}건</div>
47
+ </div>
48
+
49
+ <div table>
50
+ <div table-header>
51
+ <div col></div>
52
+ <div col>현장상태</div>
53
+ <div col>현장정보</div>
54
+ <div col>기타정보</div>
55
+ <div col>입력요청</div>
56
+ <div col>종합KPI</div>
57
+ </div>
58
+ <div table-body>
59
+ ${(_a = this.projectList) === null || _a === void 0 ? void 0 : _a.map((project) => {
60
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k;
61
+ const areaText = `연면적 : ${((_b = (_a = project === null || project === void 0 ? void 0 : project.buildingComplex) === null || _a === void 0 ? void 0 : _a.area) === null || _b === void 0 ? void 0 : _b.toLocaleString()) || '-'} ㎡`;
62
+ const costText = `공사비 : ${((_d = (_c = project === null || project === void 0 ? void 0 : project.buildingComplex) === null || _c === void 0 ? void 0 : _c.constructionCost) === null || _d === void 0 ? void 0 : _d.toLocaleString()) || '-'} 원`;
63
+ const householdCountText = `세대수 : ${((_f = (_e = project === null || project === void 0 ? void 0 : project.buildingComplex) === null || _e === void 0 ? void 0 : _e.householdCount) === null || _f === void 0 ? void 0 : _f.toLocaleString()) || '-'} 세대`;
64
+ return html `
65
+ <div tr @click=${() => navigate(`project-detail/${project.id}`)}>
66
+ <div td-pics>
67
+ <img pic src=${((_g = project.mainPhoto) === null || _g === void 0 ? void 0 : _g.fullpath) || '/assets/images/no-image.png'} alt="${project.name}" />
68
+ </div>
69
+ <div td-status>${PROJECT_STATE[project.state]}</div>
70
+ <div td-info>
71
+ <div class="name">${project.name}</div>
72
+ <div class="sub">주소 : ${((_h = project.buildingComplex) === null || _h === void 0 ? void 0 : _h.address) || ''}</div>
73
+ <div class="sub">착공~준공 : ${project.startDate} ~ ${project.endDate}</div>
74
+ </div>
75
+ <div td-etc>${areaText + '\n' + costText + '\n' + householdCountText}</div>
76
+ <div td-request>KPI입력 요청 - 건</div>
77
+ <div td-kpi>
78
+ <kpi-single-boxplot-chart
79
+ .data=${{
80
+ min: 0,
81
+ max: 100,
82
+ mean: 50,
83
+ median: 60,
84
+ q1: 25,
85
+ q3: 75,
86
+ value: project.kpi
87
+ }}
88
+ ></kpi-single-boxplot-chart>
89
+ <span class="kpi-value">${(_k = (_j = project.kpi) === null || _j === void 0 ? void 0 : _j.toFixed(0)) !== null && _k !== void 0 ? _k : '-'}</span>
90
+ </div>
91
+ </div>
92
+ `;
93
+ })}
94
+ </div>
95
+ </div>
96
+ <sv-pagenation-control
97
+ .currentPage=${this.currentPage}
98
+ .totalItems=${this.projectCount}
99
+ .pageLimit=${this.pageLimit}
100
+ @page-change=${(e) => this._changePage(e.detail.page)}
101
+ ></sv-pagenation-control>
102
+ `;
103
+ }
104
+ async pageUpdated(changes, lifecycle) {
105
+ if (this.active) {
106
+ this.getProjectList();
107
+ }
108
+ }
109
+ async getProjectList() {
110
+ var _a, _b;
111
+ const response = await client.query({
112
+ query: gql `
113
+ query Projects($filters: [Filter!], $sortings: [Sorting!], $pagination: Pagination) {
114
+ projects(filters: $filters, sortings: $sortings, pagination: $pagination) {
115
+ items {
116
+ id
117
+ name
118
+ state
119
+ startDate
120
+ endDate
121
+ mainPhoto {
122
+ fullpath
123
+ }
124
+ totalProgress
125
+ weeklyProgress
126
+ kpi
127
+ inspPassRate
128
+ robotProgressRate
129
+ structuralSafetyRate
130
+ buildingComplex {
131
+ address
132
+ area
133
+ clientCompany
134
+ constructionCost
135
+ householdCount
136
+ }
137
+ }
138
+ total
139
+ }
140
+ }
141
+ `,
142
+ variables: {
143
+ filters: [
144
+ { name: 'name', operator: 'search', value: `%${this.projectName}%` },
145
+ { name: 'state', operator: 'eq', value: ProjectState.COMPLETED }
146
+ ],
147
+ sortings: [{ name: 'createdAt', desc: true }],
148
+ pagination: { page: this.currentPage, limit: this.pageLimit }
149
+ }
150
+ });
151
+ this.projectList = ((_a = response.data.projects) === null || _a === void 0 ? void 0 : _a.items) || [];
152
+ this.projectCount = ((_b = response.data.projects) === null || _b === void 0 ? void 0 : _b.total) || 0;
153
+ }
154
+ // Input 요소의 값이 변경될 때 호출되는 콜백 함수
155
+ _onInputChange(event) {
156
+ const target = event.target;
157
+ this[target.name] = target.value;
158
+ if (target.name === 'projectName') {
159
+ this.currentPage = 1;
160
+ }
161
+ }
162
+ // 검색창에서 엔터입력시 검색
163
+ _onKeypress(event) {
164
+ if (event.code === 'Enter') {
165
+ this.currentPage = 1;
166
+ this.getProjectList();
167
+ }
168
+ }
169
+ _changePage(page) {
170
+ const nextPage = Math.min(Math.max(1, page), this.totalPages);
171
+ if (nextPage !== this.currentPage) {
172
+ this.currentPage = nextPage;
173
+ this.getProjectList();
174
+ }
175
+ }
176
+ };
177
+ SvProjectCompletedListPage.styles = [
178
+ css `
179
+ :host {
180
+ display: flex;
181
+ flex-direction: column;
182
+ overflow-y: auto;
183
+
184
+ width: 100%;
185
+ height: 100%;
186
+ background-color: var(--md-sys-color-background, #f6f6f6);
187
+
188
+ --grid-record-emphasized-background-color: red;
189
+ --grid-record-emphasized-color: yellow;
190
+ }
191
+
192
+ /* Search bar (Figma: content > search bar) */
193
+ div[header] {
194
+ display: flex;
195
+ align-items: center;
196
+ gap: 10px;
197
+ margin: var(--spacing-large, 12px);
198
+ margin-bottom: 0;
199
+ padding: 12px 11px;
200
+ border-radius: 7px;
201
+ background-color: rgba(46, 164, 223, 0.1);
202
+ border: 1px solid rgba(46, 164, 223, 0.3);
203
+ }
204
+ div[header] div[search] {
205
+ display: flex;
206
+ align-items: center;
207
+ gap: 6px;
208
+ min-width: 186px;
209
+ }
210
+ div[header] div[search] md-icon {
211
+ color: #212529;
212
+ }
213
+ div[header] div[search] input[type='search'] {
214
+ border: none;
215
+ outline: none;
216
+ background: transparent;
217
+ font-size: 16px;
218
+ color: #212529;
219
+ padding: 2px 0;
220
+ border-bottom: 1px solid rgba(0, 0, 0, 0.2);
221
+ }
222
+ div[header] .spacer {
223
+ flex: 1;
224
+ }
225
+ div[header] div[count] {
226
+ font-size: 16px;
227
+ color: #000000;
228
+ margin-right: 6px;
229
+ }
230
+ div[header] md-icon[view] {
231
+ color: #35618e;
232
+ cursor: pointer;
233
+ }
234
+ div[header] md-icon[add] {
235
+ color: #4cbb49;
236
+ cursor: pointer;
237
+ }
238
+
239
+ /* Table (Figma: content > table) */
240
+ div[table] {
241
+ margin: var(--spacing-large, 12px);
242
+ border: 1px solid rgba(0, 0, 0, 0.15);
243
+ border-top: 2px solid #0c4da2;
244
+ border-radius: var(--md-sys-shape-corner-small, 5px);
245
+ overflow: hidden;
246
+ background: #ffffff;
247
+ overflow-y: auto;
248
+ min-height: fit-content;
249
+ }
250
+ div[table-header] {
251
+ display: grid;
252
+ grid-template-columns: 150px 80px 1fr 220px 150px 100px;
253
+ align-items: center;
254
+ justify-items: center;
255
+ height: 35px;
256
+ background: #f3f3fa;
257
+ padding: 0 25px;
258
+ column-gap: 30px;
259
+ }
260
+ div[table-header] div[col] {
261
+ font-size: 16px;
262
+ color: #212529;
263
+ line-height: 1.875em;
264
+ }
265
+ div[table-body] {
266
+ display: block;
267
+ }
268
+ div[tr] {
269
+ display: grid;
270
+ grid-template-columns: 150px 80px 1fr 220px 150px 100px;
271
+ align-items: center;
272
+ justify-items: center;
273
+ column-gap: 30px;
274
+ padding: 12px 25px;
275
+ background: #ffffff;
276
+ border-top: 1px solid rgba(0, 0, 0, 0.15);
277
+ cursor: pointer;
278
+ }
279
+ /* 좌측 정렬 요구사항 */
280
+ div[td-info],
281
+ div[td-etc] {
282
+ justify-self: start;
283
+ text-align: left;
284
+ }
285
+ div[td-pics] img[pic] {
286
+ width: 150px;
287
+ height: 90px;
288
+ object-fit: fill;
289
+ background: #e5e7eb;
290
+ border-radius: 2px;
291
+ }
292
+ div[td-status] {
293
+ font-weight: 700;
294
+ font-size: 16px;
295
+ color: #4cbb49;
296
+ line-height: 1;
297
+ }
298
+ div[td-info] {
299
+ display: flex;
300
+ flex-direction: column;
301
+ gap: 4px;
302
+ }
303
+ div[td-info] .name {
304
+ font-weight: 700;
305
+ font-size: 14px;
306
+ color: #000000;
307
+ }
308
+ div[td-info] .sub {
309
+ font-size: 14px;
310
+ color: #000000;
311
+ line-height: 1.7;
312
+ }
313
+ div[td-etc] {
314
+ font-size: 16px;
315
+ color: #212529;
316
+ white-space: pre-line;
317
+ }
318
+ div[td-request] {
319
+ font-weight: 700;
320
+ font-size: 16px;
321
+ color: #35618e;
322
+ white-space: pre-line;
323
+ }
324
+ div[td-kpi] {
325
+ display: flex;
326
+ align-items: center;
327
+ height: 100%;
328
+ }
329
+ div[td-kpi] .kpi-value {
330
+ font-weight: 700;
331
+ font-size: 22px;
332
+ color: #35618e;
333
+ width: 30px;
334
+ }
335
+ `
336
+ ];
337
+ __decorate([
338
+ state(),
339
+ __metadata("design:type", String)
340
+ ], SvProjectCompletedListPage.prototype, "projectName", void 0);
341
+ __decorate([
342
+ state(),
343
+ __metadata("design:type", Array)
344
+ ], SvProjectCompletedListPage.prototype, "projectList", void 0);
345
+ __decorate([
346
+ state(),
347
+ __metadata("design:type", Number)
348
+ ], SvProjectCompletedListPage.prototype, "projectCount", void 0);
349
+ __decorate([
350
+ state(),
351
+ __metadata("design:type", Number)
352
+ ], SvProjectCompletedListPage.prototype, "currentPage", void 0);
353
+ SvProjectCompletedListPage = __decorate([
354
+ customElement('sv-project-completed-list')
355
+ ], SvProjectCompletedListPage);
356
+ export { SvProjectCompletedListPage };
357
+ //# sourceMappingURL=sv-project-completed-list.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sv-project-completed-list.js","sourceRoot":"","sources":["../../client/pages/sv-project-completed-list.ts"],"names":[],"mappings":"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","sourcesContent":["import '@material/web/icon/icon.js'\nimport '../components/kpi-single-boxplot-chart'\n\nimport { PageView, navigate } from '@operato/shell'\nimport { css, html } from 'lit'\nimport { customElement, state } from 'lit/decorators.js'\nimport { ScopedElementsMixin } from '@open-wc/scoped-elements'\nimport { client } from '@operato/graphql'\nimport gql from 'graphql-tag'\nimport { Project, PROJECT_STATE, ProjectState } from './sv-project-list'\nimport '../components/sv-pagenation-control'\n\n@customElement('sv-project-completed-list')\nexport class SvProjectCompletedListPage extends ScopedElementsMixin(PageView) {\n static styles = [\n css`\n :host {\n display: flex;\n flex-direction: column;\n overflow-y: auto;\n\n width: 100%;\n height: 100%;\n background-color: var(--md-sys-color-background, #f6f6f6);\n\n --grid-record-emphasized-background-color: red;\n --grid-record-emphasized-color: yellow;\n }\n\n /* Search bar (Figma: content > search bar) */\n div[header] {\n display: flex;\n align-items: center;\n gap: 10px;\n margin: var(--spacing-large, 12px);\n margin-bottom: 0;\n padding: 12px 11px;\n border-radius: 7px;\n background-color: rgba(46, 164, 223, 0.1);\n border: 1px solid rgba(46, 164, 223, 0.3);\n }\n div[header] div[search] {\n display: flex;\n align-items: center;\n gap: 6px;\n min-width: 186px;\n }\n div[header] div[search] md-icon {\n color: #212529;\n }\n div[header] div[search] input[type='search'] {\n border: none;\n outline: none;\n background: transparent;\n font-size: 16px;\n color: #212529;\n padding: 2px 0;\n border-bottom: 1px solid rgba(0, 0, 0, 0.2);\n }\n div[header] .spacer {\n flex: 1;\n }\n div[header] div[count] {\n font-size: 16px;\n color: #000000;\n margin-right: 6px;\n }\n div[header] md-icon[view] {\n color: #35618e;\n cursor: pointer;\n }\n div[header] md-icon[add] {\n color: #4cbb49;\n cursor: pointer;\n }\n\n /* Table (Figma: content > table) */\n div[table] {\n margin: var(--spacing-large, 12px);\n border: 1px solid rgba(0, 0, 0, 0.15);\n border-top: 2px solid #0c4da2;\n border-radius: var(--md-sys-shape-corner-small, 5px);\n overflow: hidden;\n background: #ffffff;\n overflow-y: auto;\n min-height: fit-content;\n }\n div[table-header] {\n display: grid;\n grid-template-columns: 150px 80px 1fr 220px 150px 100px;\n align-items: center;\n justify-items: center;\n height: 35px;\n background: #f3f3fa;\n padding: 0 25px;\n column-gap: 30px;\n }\n div[table-header] div[col] {\n font-size: 16px;\n color: #212529;\n line-height: 1.875em;\n }\n div[table-body] {\n display: block;\n }\n div[tr] {\n display: grid;\n grid-template-columns: 150px 80px 1fr 220px 150px 100px;\n align-items: center;\n justify-items: center;\n column-gap: 30px;\n padding: 12px 25px;\n background: #ffffff;\n border-top: 1px solid rgba(0, 0, 0, 0.15);\n cursor: pointer;\n }\n /* 좌측 정렬 요구사항 */\n div[td-info],\n div[td-etc] {\n justify-self: start;\n text-align: left;\n }\n div[td-pics] img[pic] {\n width: 150px;\n height: 90px;\n object-fit: fill;\n background: #e5e7eb;\n border-radius: 2px;\n }\n div[td-status] {\n font-weight: 700;\n font-size: 16px;\n color: #4cbb49;\n line-height: 1;\n }\n div[td-info] {\n display: flex;\n flex-direction: column;\n gap: 4px;\n }\n div[td-info] .name {\n font-weight: 700;\n font-size: 14px;\n color: #000000;\n }\n div[td-info] .sub {\n font-size: 14px;\n color: #000000;\n line-height: 1.7;\n }\n div[td-etc] {\n font-size: 16px;\n color: #212529;\n white-space: pre-line;\n }\n div[td-request] {\n font-weight: 700;\n font-size: 16px;\n color: #35618e;\n white-space: pre-line;\n }\n div[td-kpi] {\n display: flex;\n align-items: center;\n height: 100%;\n }\n div[td-kpi] .kpi-value {\n font-weight: 700;\n font-size: 22px;\n color: #35618e;\n width: 30px;\n }\n `\n ]\n\n get context() {\n return {\n title: '완료 프로젝트'\n }\n }\n\n @state() private projectName: string = ''\n @state() private projectList: Project[] = []\n @state() private projectCount: number = 0\n @state() private currentPage: number = 1\n\n private readonly pageLimit: number = 20\n\n get totalPages(): number {\n return Math.max(1, Math.ceil((this.projectCount || 0) / this.pageLimit))\n }\n\n render() {\n return html`\n <div header>\n <div search>\n <md-icon>manage_search</md-icon>\n <input\n type=\"search\"\n name=\"projectName\"\n placeholder=\"프로젝트명\"\n .value=${this.projectName}\n @input=${this._onInputChange}\n @keypress=${this._onKeypress}\n />\n </div>\n\n <span class=\"spacer\"></span>\n <div count>총 ${this.projectCount}건</div>\n </div>\n\n <div table>\n <div table-header>\n <div col></div>\n <div col>현장상태</div>\n <div col>현장정보</div>\n <div col>기타정보</div>\n <div col>입력요청</div>\n <div col>종합KPI</div>\n </div>\n <div table-body>\n ${this.projectList?.map((project: Project) => {\n const areaText = `연면적 : ${project?.buildingComplex?.area?.toLocaleString() || '-'} ㎡`\n const costText = `공사비 : ${project?.buildingComplex?.constructionCost?.toLocaleString() || '-'} 원`\n const householdCountText = `세대수 : ${project?.buildingComplex?.householdCount?.toLocaleString() || '-'} 세대`\n\n return html`\n <div tr @click=${() => navigate(`project-detail/${project.id}`)}>\n <div td-pics>\n <img pic src=${project.mainPhoto?.fullpath || '/assets/images/no-image.png'} alt=\"${project.name}\" />\n </div>\n <div td-status>${PROJECT_STATE[project.state]}</div>\n <div td-info>\n <div class=\"name\">${project.name}</div>\n <div class=\"sub\">주소 : ${project.buildingComplex?.address || ''}</div>\n <div class=\"sub\">착공~준공 : ${project.startDate} ~ ${project.endDate}</div>\n </div>\n <div td-etc>${areaText + '\\n' + costText + '\\n' + householdCountText}</div>\n <div td-request>KPI입력 요청 - 건</div>\n <div td-kpi>\n <kpi-single-boxplot-chart\n .data=${{\n min: 0,\n max: 100,\n mean: 50,\n median: 60,\n q1: 25,\n q3: 75,\n value: project.kpi\n }}\n ></kpi-single-boxplot-chart>\n <span class=\"kpi-value\">${project.kpi?.toFixed(0) ?? '-'}</span>\n </div>\n </div>\n `\n })}\n </div>\n </div>\n <sv-pagenation-control\n .currentPage=${this.currentPage}\n .totalItems=${this.projectCount}\n .pageLimit=${this.pageLimit}\n @page-change=${(e: CustomEvent) => this._changePage(e.detail.page)}\n ></sv-pagenation-control>\n `\n }\n\n async pageUpdated(changes: any, lifecycle: any) {\n if (this.active) {\n this.getProjectList()\n }\n }\n\n async getProjectList() {\n const response = await client.query({\n query: gql`\n query Projects($filters: [Filter!], $sortings: [Sorting!], $pagination: Pagination) {\n projects(filters: $filters, sortings: $sortings, pagination: $pagination) {\n items {\n id\n name\n state\n startDate\n endDate\n mainPhoto {\n fullpath\n }\n totalProgress\n weeklyProgress\n kpi\n inspPassRate\n robotProgressRate\n structuralSafetyRate\n buildingComplex {\n address\n area\n clientCompany\n constructionCost\n householdCount\n }\n }\n total\n }\n }\n `,\n variables: {\n filters: [\n { name: 'name', operator: 'search', value: `%${this.projectName}%` },\n { name: 'state', operator: 'eq', value: ProjectState.COMPLETED }\n ],\n sortings: [{ name: 'createdAt', desc: true }],\n pagination: { page: this.currentPage, limit: this.pageLimit }\n }\n })\n\n this.projectList = response.data.projects?.items || []\n this.projectCount = response.data.projects?.total || 0\n }\n\n // Input 요소의 값이 변경될 때 호출되는 콜백 함수\n private _onInputChange(event: InputEvent) {\n const target = event.target as HTMLInputElement\n this[target.name] = target.value\n if (target.name === 'projectName') {\n this.currentPage = 1\n }\n }\n\n // 검색창에서 엔터입력시 검색\n private _onKeypress(event: KeyboardEvent) {\n if (event.code === 'Enter') {\n this.currentPage = 1\n this.getProjectList()\n }\n }\n\n private _changePage(page: number) {\n const nextPage = Math.min(Math.max(1, page), this.totalPages)\n if (nextPage !== this.currentPage) {\n this.currentPage = nextPage\n this.getProjectList()\n }\n }\n}\n"]}
@@ -0,0 +1,13 @@
1
+ import { PageLifecycle, PageView } from '@operato/shell';
2
+ declare const SvProjectDetailPage_base: typeof PageView & import("@open-wc/dedupe-mixin").Constructor<import("@open-wc/scoped-elements/types/src/types").ScopedElementsHost>;
3
+ export declare class SvProjectDetailPage extends SvProjectDetailPage_base {
4
+ static styles: import("lit").CSSResult[];
5
+ get context(): {
6
+ title: string;
7
+ };
8
+ private project;
9
+ render(): import("lit-html").TemplateResult<1>;
10
+ pageUpdated(changes: any, lifecycle: PageLifecycle): Promise<void>;
11
+ initProject(projectId?: string): Promise<void>;
12
+ }
13
+ export {};