@dssp/dkpi 1.0.0-alpha.7 → 1.0.0-alpha.71

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 (232) hide show
  1. package/KPI-STATISTICS-SERVICE.md +233 -0
  2. package/_index.html +0 -5
  3. package/assets/favicon.ico +0 -0
  4. package/assets/images/project-image.png +0 -0
  5. package/assets/manifest/apple-1024.png +0 -0
  6. package/assets/manifest/apple-120.png +0 -0
  7. package/assets/manifest/apple-152.png +0 -0
  8. package/assets/manifest/apple-167.png +0 -0
  9. package/assets/manifest/apple-180.png +0 -0
  10. package/assets/manifest/apple-touch-icon.png +0 -0
  11. package/assets/manifest/badge-128x128.png +0 -0
  12. package/assets/manifest/chrome-splashscreen-icon-384x384.png +0 -0
  13. package/assets/manifest/chrome-touch-icon-192x192.png +0 -0
  14. package/assets/manifest/icon-128x128.png +0 -0
  15. package/assets/manifest/icon-192x192.png +0 -0
  16. package/assets/manifest/icon-512x512.png +0 -0
  17. package/assets/manifest/icon-72x72.png +0 -0
  18. package/assets/manifest/icon-96x96.png +0 -0
  19. package/assets/manifest/image-metaog.png +0 -0
  20. package/assets/manifest/maskable_icon.png +0 -0
  21. package/assets/manifest/ms-icon-144x144.png +0 -0
  22. package/assets/manifest/ms-touch-icon-144x144-precomposed.png +0 -0
  23. package/assets/videos/intro.mp4 +0 -0
  24. package/dist-client/bootstrap.js +64 -4
  25. package/dist-client/bootstrap.js.map +1 -1
  26. package/dist-client/components/kpi-2d-lookup-chart.d.ts +43 -0
  27. package/dist-client/components/kpi-2d-lookup-chart.js +253 -0
  28. package/dist-client/components/kpi-2d-lookup-chart.js.map +1 -0
  29. package/dist-client/components/kpi-boxplot-chart.d.ts +24 -0
  30. package/dist-client/components/kpi-boxplot-chart.js +291 -0
  31. package/dist-client/components/kpi-boxplot-chart.js.map +1 -0
  32. package/dist-client/components/kpi-lookup-chart.d.ts +53 -0
  33. package/dist-client/components/kpi-lookup-chart.js +430 -0
  34. package/dist-client/components/kpi-lookup-chart.js.map +1 -0
  35. package/dist-client/components/kpi-mini-trend-chart.d.ts +14 -0
  36. package/dist-client/components/kpi-mini-trend-chart.js +148 -0
  37. package/dist-client/components/kpi-mini-trend-chart.js.map +1 -0
  38. package/dist-client/components/kpi-radar-chart.d.ts +17 -0
  39. package/dist-client/components/kpi-radar-chart.js +259 -0
  40. package/dist-client/components/kpi-radar-chart.js.map +1 -0
  41. package/dist-client/components/kpi-single-boxplot-chart.d.ts +24 -0
  42. package/dist-client/components/kpi-single-boxplot-chart.js +391 -0
  43. package/dist-client/components/kpi-single-boxplot-chart.js.map +1 -0
  44. package/dist-client/components/kpi-trend-chart.d.ts +25 -0
  45. package/dist-client/components/kpi-trend-chart.js +220 -0
  46. package/dist-client/components/kpi-trend-chart.js.map +1 -0
  47. package/dist-client/components/sv-pagenation-control.d.ts +18 -0
  48. package/dist-client/components/sv-pagenation-control.js +142 -0
  49. package/dist-client/components/sv-pagenation-control.js.map +1 -0
  50. package/dist-client/google-map/common-google-map.d.ts +35 -0
  51. package/dist-client/google-map/common-google-map.js +345 -0
  52. package/dist-client/google-map/common-google-map.js.map +1 -0
  53. package/dist-client/google-map/google-map-loader.d.ts +6 -0
  54. package/dist-client/google-map/google-map-loader.js +23 -0
  55. package/dist-client/google-map/google-map-loader.js.map +1 -0
  56. package/dist-client/icons/menu-icons.d.ts +6 -0
  57. package/dist-client/icons/menu-icons.js +42 -0
  58. package/dist-client/icons/menu-icons.js.map +1 -1
  59. package/dist-client/pages/kpi-admin/dssp-kpi-list-page.d.ts +22 -0
  60. package/dist-client/pages/kpi-admin/dssp-kpi-list-page.js +57 -0
  61. package/dist-client/pages/kpi-admin/dssp-kpi-list-page.js.map +1 -0
  62. package/dist-client/pages/kpi-admin/dssp-kpi-overview.d.ts +46 -0
  63. package/dist-client/pages/kpi-admin/dssp-kpi-overview.js +378 -0
  64. package/dist-client/pages/kpi-admin/dssp-kpi-overview.js.map +1 -0
  65. package/dist-client/pages/kpi-admin/kpi-grade-2d-editor.d.ts +20 -0
  66. package/dist-client/pages/kpi-admin/kpi-grade-2d-editor.js +445 -0
  67. package/dist-client/pages/kpi-admin/kpi-grade-2d-editor.js.map +1 -0
  68. package/dist-client/pages/kpi-admin/kpi-system-guide.d.ts +18 -0
  69. package/dist-client/pages/kpi-admin/kpi-system-guide.js +535 -0
  70. package/dist-client/pages/kpi-admin/kpi-system-guide.js.map +1 -0
  71. package/dist-client/pages/kpi-dashboard/cards/kpi-level1-card.d.ts +18 -0
  72. package/dist-client/pages/kpi-dashboard/cards/kpi-level1-card.js +259 -0
  73. package/dist-client/pages/kpi-dashboard/cards/kpi-level1-card.js.map +1 -0
  74. package/dist-client/pages/kpi-dashboard/cards/kpi-level2-comparison.d.ts +22 -0
  75. package/dist-client/pages/kpi-dashboard/cards/kpi-level2-comparison.js +346 -0
  76. package/dist-client/pages/kpi-dashboard/cards/kpi-level2-comparison.js.map +1 -0
  77. package/dist-client/pages/kpi-dashboard/cards/kpi-level3-comparison.d.ts +26 -0
  78. package/dist-client/pages/kpi-dashboard/cards/kpi-level3-comparison.js +433 -0
  79. package/dist-client/pages/kpi-dashboard/cards/kpi-level3-comparison.js.map +1 -0
  80. package/dist-client/pages/kpi-dashboard/components/kpi-chart-toggle.d.ts +8 -0
  81. package/dist-client/pages/kpi-dashboard/components/kpi-chart-toggle.js +78 -0
  82. package/dist-client/pages/kpi-dashboard/components/kpi-chart-toggle.js.map +1 -0
  83. package/dist-client/pages/kpi-dashboard/components/kpi-left-panel.d.ts +38 -0
  84. package/dist-client/pages/kpi-dashboard/components/kpi-left-panel.js +851 -0
  85. package/dist-client/pages/kpi-dashboard/components/kpi-left-panel.js.map +1 -0
  86. package/dist-client/pages/kpi-dashboard/components/kpi-map-panel.d.ts +42 -0
  87. package/dist-client/pages/kpi-dashboard/components/kpi-map-panel.js +316 -0
  88. package/dist-client/pages/kpi-dashboard/components/kpi-map-panel.js.map +1 -0
  89. package/dist-client/pages/kpi-dashboard/components/kpi-region-popup.d.ts +28 -0
  90. package/dist-client/pages/kpi-dashboard/components/kpi-region-popup.js +497 -0
  91. package/dist-client/pages/kpi-dashboard/components/kpi-region-popup.js.map +1 -0
  92. package/dist-client/pages/kpi-dashboard/kpi-alert-panel.d.ts +18 -0
  93. package/dist-client/pages/kpi-dashboard/kpi-alert-panel.js +131 -0
  94. package/dist-client/pages/kpi-dashboard/kpi-alert-panel.js.map +1 -0
  95. package/dist-client/pages/kpi-dashboard/kpi-dashboard-map.d.ts +52 -0
  96. package/dist-client/pages/kpi-dashboard/kpi-dashboard-map.js +798 -0
  97. package/dist-client/pages/kpi-dashboard/kpi-dashboard-map.js.map +1 -0
  98. package/dist-client/pages/kpi-dashboard/kpi-dashboard.d.ts +63 -0
  99. package/dist-client/pages/kpi-dashboard/kpi-dashboard.js +1089 -0
  100. package/dist-client/pages/kpi-dashboard/kpi-dashboard.js.map +1 -0
  101. package/dist-client/pages/kpi-dashboard/kpi-grade-visualization.d.ts +12 -0
  102. package/dist-client/pages/kpi-dashboard/kpi-grade-visualization.js +82 -0
  103. package/dist-client/pages/kpi-dashboard/kpi-grade-visualization.js.map +1 -0
  104. package/dist-client/pages/kpi-dashboard/kpi-history-viewer.d.ts +11 -0
  105. package/dist-client/pages/kpi-dashboard/kpi-history-viewer.js +65 -0
  106. package/dist-client/pages/kpi-dashboard/kpi-history-viewer.js.map +1 -0
  107. package/dist-client/pages/kpi-dashboard/kpi-list-summary.d.ts +13 -0
  108. package/dist-client/pages/kpi-dashboard/kpi-list-summary.js +115 -0
  109. package/dist-client/pages/kpi-dashboard/kpi-list-summary.js.map +1 -0
  110. package/dist-client/pages/kpi-dashboard/kpi-performance-summary.d.ts +15 -0
  111. package/dist-client/pages/kpi-dashboard/kpi-performance-summary.js +147 -0
  112. package/dist-client/pages/kpi-dashboard/kpi-performance-summary.js.map +1 -0
  113. package/dist-client/pages/kpi-dashboard/kpi-value-entry.d.ts +7 -0
  114. package/dist-client/pages/kpi-dashboard/kpi-value-entry.js +86 -0
  115. package/dist-client/pages/kpi-dashboard/kpi-value-entry.js.map +1 -0
  116. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.d.ts +57 -0
  117. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js +719 -0
  118. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js.map +1 -0
  119. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.d.ts +23 -0
  120. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js +76 -0
  121. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js.map +1 -0
  122. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.d.ts +68 -0
  123. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js +380 -0
  124. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js.map +1 -0
  125. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.d.ts +12 -0
  126. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js +174 -0
  127. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js.map +1 -0
  128. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.d.ts +40 -0
  129. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js +190 -0
  130. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js.map +1 -0
  131. package/dist-client/pages/kpi-value/kpi-value-importer.d.ts +23 -0
  132. package/dist-client/pages/kpi-value/kpi-value-importer.js +93 -0
  133. package/dist-client/pages/kpi-value/kpi-value-importer.js.map +1 -0
  134. package/dist-client/pages/kpi-value/kpi-value-list-page.d.ts +71 -0
  135. package/dist-client/pages/kpi-value/kpi-value-list-page.js +464 -0
  136. package/dist-client/pages/kpi-value/kpi-value-list-page.js.map +1 -0
  137. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.d.ts +14 -0
  138. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.js +339 -0
  139. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.js.map +1 -0
  140. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.d.ts +14 -0
  141. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.js +276 -0
  142. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.js.map +1 -0
  143. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.d.ts +18 -0
  144. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.js +307 -0
  145. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.js.map +1 -0
  146. package/dist-client/pages/project-complete-tabs/pc-tab4-monthly.d.ts +18 -0
  147. package/dist-client/pages/project-complete-tabs/pc-tab4-monthly.js +433 -0
  148. package/dist-client/pages/project-complete-tabs/pc-tab4-monthly.js.map +1 -0
  149. package/dist-client/pages/sv-project-complete.d.ts +21 -0
  150. package/dist-client/pages/sv-project-complete.js +213 -0
  151. package/dist-client/pages/sv-project-complete.js.map +1 -0
  152. package/dist-client/pages/sv-project-completed-list.d.ts +27 -0
  153. package/dist-client/pages/sv-project-completed-list.js +416 -0
  154. package/dist-client/pages/sv-project-completed-list.js.map +1 -0
  155. package/dist-client/pages/sv-project-detail.d.ts +46 -0
  156. package/dist-client/pages/sv-project-detail.js +1236 -0
  157. package/dist-client/pages/sv-project-detail.js.map +1 -0
  158. package/dist-client/pages/sv-project-list.d.ts +165 -0
  159. package/dist-client/pages/sv-project-list.js +488 -0
  160. package/dist-client/pages/sv-project-list.js.map +1 -0
  161. package/dist-client/route.d.ts +1 -1
  162. package/dist-client/route.js +35 -1
  163. package/dist-client/route.js.map +1 -1
  164. package/dist-client/shared/complete-api.d.ts +8 -0
  165. package/dist-client/shared/complete-api.js +177 -0
  166. package/dist-client/shared/complete-api.js.map +1 -0
  167. package/dist-client/shared/func.d.ts +2 -0
  168. package/dist-client/shared/func.js +22 -0
  169. package/dist-client/shared/func.js.map +1 -0
  170. package/dist-client/themes/dark.css +24 -24
  171. package/dist-client/themes/light.css +23 -23
  172. package/dist-client/tsconfig.tsbuildinfo +1 -1
  173. package/dist-client/viewparts/menu-tools.d.ts +40 -5
  174. package/dist-client/viewparts/menu-tools.js +289 -34
  175. package/dist-client/viewparts/menu-tools.js.map +1 -1
  176. package/dist-server/index.d.ts +2 -0
  177. package/dist-server/index.js +5 -0
  178. package/dist-server/index.js.map +1 -1
  179. package/dist-server/migrations/index.d.ts +1 -0
  180. package/dist-server/migrations/index.js +12 -0
  181. package/dist-server/migrations/index.js.map +1 -0
  182. package/dist-server/scripts/calculate-kpi-scores.d.ts +10 -0
  183. package/dist-server/scripts/calculate-kpi-scores.js +333 -0
  184. package/dist-server/scripts/calculate-kpi-scores.js.map +1 -0
  185. package/dist-server/scripts/load-grade-data-migration.d.ts +14 -0
  186. package/dist-server/scripts/load-grade-data-migration.js +279 -0
  187. package/dist-server/scripts/load-grade-data-migration.js.map +1 -0
  188. package/dist-server/scripts/propagate-parent-kpi-values.d.ts +14 -0
  189. package/dist-server/scripts/propagate-parent-kpi-values.js +786 -0
  190. package/dist-server/scripts/propagate-parent-kpi-values.js.map +1 -0
  191. package/dist-server/scripts/recalculate-by-project-name.d.ts +2 -0
  192. package/dist-server/scripts/recalculate-by-project-name.js +72 -0
  193. package/dist-server/scripts/recalculate-by-project-name.js.map +1 -0
  194. package/dist-server/service/index.d.ts +4 -0
  195. package/dist-server/service/index.js +20 -0
  196. package/dist-server/service/index.js.map +1 -0
  197. package/dist-server/service/kpi-metric-value/index.d.ts +4 -0
  198. package/dist-server/service/kpi-metric-value/index.js +8 -0
  199. package/dist-server/service/kpi-metric-value/index.js.map +1 -0
  200. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.d.ts +74 -0
  201. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.js +687 -0
  202. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.js.map +1 -0
  203. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.d.ts +7 -0
  204. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js +52 -0
  205. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js.map +1 -0
  206. package/dist-server/service/kpi-stat/index.d.ts +4 -0
  207. package/dist-server/service/kpi-stat/index.js +8 -0
  208. package/dist-server/service/kpi-stat/index.js.map +1 -0
  209. package/dist-server/service/kpi-stat/kpi-stat-query.d.ts +12 -0
  210. package/dist-server/service/kpi-stat/kpi-stat-query.js +662 -0
  211. package/dist-server/service/kpi-stat/kpi-stat-query.js.map +1 -0
  212. package/dist-server/service/kpi-stat/kpi-stat-types.d.ts +32 -0
  213. package/dist-server/service/kpi-stat/kpi-stat-types.js +180 -0
  214. package/dist-server/service/kpi-stat/kpi-stat-types.js.map +1 -0
  215. package/dist-server/service/kpi-value/index.d.ts +3 -0
  216. package/dist-server/service/kpi-value/index.js +7 -0
  217. package/dist-server/service/kpi-value/index.js.map +1 -0
  218. package/dist-server/service/kpi-value/kpi-value-query.d.ts +8 -0
  219. package/dist-server/service/kpi-value/kpi-value-query.js +69 -0
  220. package/dist-server/service/kpi-value/kpi-value-query.js.map +1 -0
  221. package/dist-server/tsconfig.tsbuildinfo +1 -1
  222. package/kpi-module-service-tests.md +1286 -0
  223. package/kpi-module-test-report.md +676 -0
  224. package/kpi-module-unit-test-detailed-report.md +925 -0
  225. package/kpi-module-unit-tests-detailed.md +1452 -0
  226. package/package.json +67 -55
  227. package/recalculate-batch.sh +64 -0
  228. package/recalculate-projects-range.sh +98 -0
  229. package/schema.graphql +2514 -455
  230. package/things-factory.config.js +11 -1
  231. package/views/auth-page.html +0 -1
  232. package/views/public/home.html +0 -1
@@ -0,0 +1,213 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import '@material/web/icon/icon.js';
3
+ import { navigate, PageView } from '@operato/shell';
4
+ import { css, html } from 'lit';
5
+ import { customElement, state } from 'lit/decorators.js';
6
+ import { ScopedElementsMixin } from '@open-wc/scoped-elements';
7
+ import './project-complete-tabs/pc-tab1-plan';
8
+ import './project-complete-tabs/pc-tab2-rating';
9
+ import './project-complete-tabs/pc-tab3-upload';
10
+ import './project-complete-tabs/pc-tab4-monthly';
11
+ import { getProject, updateProjectCompleteFinalize } from '../shared/complete-api';
12
+ import { notify } from '@operato/layout';
13
+ import { OxPrompt } from '@operato/popup';
14
+ let SvProjectCompletePage = class SvProjectCompletePage extends ScopedElementsMixin(PageView) {
15
+ constructor() {
16
+ super(...arguments);
17
+ this.activeTab = 1;
18
+ this.projectId = '';
19
+ this.project = {};
20
+ // 탭 클릭
21
+ this._onTabClick = async (tabNumber) => {
22
+ if (tabNumber === this.activeTab)
23
+ return;
24
+ this.activeTab = tabNumber;
25
+ };
26
+ // 완공 처리
27
+ this._onComplete = async () => {
28
+ if (await OxPrompt.open({ title: '완공 처리를 하시겠습니까?', confirmButton: { text: '확인' }, cancelButton: { text: '취소' } })) {
29
+ const result = await updateProjectCompleteFinalize(this.projectId);
30
+ if (!result.errors) {
31
+ notify({ message: '완공 처리가 완료되었습니다.', level: 'info' });
32
+ navigate(`project-detail/${this.projectId}`);
33
+ }
34
+ }
35
+ };
36
+ }
37
+ get context() {
38
+ var _a;
39
+ return {
40
+ title: this.project ? `프로젝트 완공 처리 - ${(_a = this.project) === null || _a === void 0 ? void 0 : _a.name}` : '프로젝트 완공 처리'
41
+ };
42
+ }
43
+ render() {
44
+ return html `
45
+ <div class="page-header">
46
+ <div class="page-title">프로젝트 완공 처리</div>
47
+ <div class="triangle"></div>
48
+ <span class="spacer"></span>
49
+ <div class="ghost-btn" @click=${() => navigate(`project-detail/${this.projectId}`)}>
50
+ <md-icon>arrow_back</md-icon>
51
+ <div>상세로 돌아가기</div>
52
+ </div>
53
+ <div class="ghost-btn complete" @click=${this._onComplete}>
54
+ <md-icon>check_circle</md-icon>
55
+ <div>완공 처리</div>
56
+ </div>
57
+ </div>
58
+
59
+ <div class="card">
60
+ <div class="tabs">
61
+ <div class="tab" ?active=${this.activeTab === 1} @click=${() => this._onTabClick(1)}>
62
+ Step1. 프로젝트 기본정보 현행화
63
+ </div>
64
+ <div class="tab" ?active=${this.activeTab === 2} @click=${() => this._onTabClick(2)}>Step2. 프로젝트 완료 평가</div>
65
+ <div class="tab" ?active=${this.activeTab === 3} @click=${() => this._onTabClick(3)}>Step3. 준공 문서 업로드</div>
66
+ <div class="tab" ?active=${this.activeTab === 4} @click=${() => this._onTabClick(4)}>Step4. 월별 데이터 입력</div>
67
+ </div>
68
+
69
+ <div class="tab-body">
70
+ ${this.activeTab === 1
71
+ ? html `<sv-pc-tab1-plan .project=${this.project}></sv-pc-tab1-plan>`
72
+ : this.activeTab === 2
73
+ ? html `<sv-pc-tab2-rating .project=${this.project}></sv-pc-tab2-rating>`
74
+ : this.activeTab === 3
75
+ ? html `<sv-pc-tab3-upload .project=${this.project}></sv-pc-tab3-upload>`
76
+ : html `<sv-pc-tab4-monthly .project=${this.project}></sv-pc-tab4-monthly>`}
77
+ </div>
78
+ </div>
79
+ `;
80
+ }
81
+ async pageUpdated(changes, lifecycle) {
82
+ if (this.active) {
83
+ this.projectId = lifecycle.resourceId || '';
84
+ this.project = await getProject(this.projectId); // View용 데이터
85
+ }
86
+ this.updateContext();
87
+ }
88
+ };
89
+ SvProjectCompletePage.styles = [
90
+ css `
91
+ :host {
92
+ display: flex;
93
+ flex-direction: column;
94
+ overflow-y: auto;
95
+
96
+ width: 100%;
97
+ height: 100%;
98
+ background-color: var(--md-sys-color-background, #f6f6f6);
99
+ }
100
+
101
+ .page-header {
102
+ display: flex;
103
+ align-items: center;
104
+ gap: 8px;
105
+ padding: 16px 20px 8px 20px;
106
+ }
107
+ .page-title {
108
+ color: #35618e;
109
+ font-weight: 700;
110
+ font-size: 22px;
111
+ letter-spacing: -0.05em;
112
+ }
113
+ .triangle {
114
+ width: 0;
115
+ height: 0;
116
+ border-left: 6px solid transparent;
117
+ border-right: 6px solid transparent;
118
+ border-top: 8px solid #35618e;
119
+ }
120
+
121
+ .card {
122
+ background: #ffffff;
123
+ border-radius: 8px;
124
+ padding: 12px;
125
+ box-shadow: 2px 2px 2px 0 rgba(0, 0, 0, 0.08);
126
+ margin: 0 20px 20px 20px;
127
+ }
128
+
129
+ .tabs {
130
+ display: flex;
131
+ justify-content: center;
132
+ padding: 0 12px 8px 12px;
133
+ border-bottom: 1px dashed rgba(0, 0, 0, 0.15);
134
+ }
135
+ .tab {
136
+ display: inline-flex;
137
+ align-items: center;
138
+ padding: 6px 14px;
139
+ border: 1px solid rgba(0, 0, 0, 0.12);
140
+ color: #35618e;
141
+ background: #f3f6f9;
142
+ cursor: pointer;
143
+ font-size: 14px;
144
+ letter-spacing: -0.02em;
145
+ }
146
+ .tab:first-child {
147
+ border-top-left-radius: 10px;
148
+ border-bottom-left-radius: 10px;
149
+ }
150
+ .tab:last-child {
151
+ border-top-right-radius: 10px;
152
+ border-bottom-right-radius: 10px;
153
+ }
154
+ /* 중간 탭 이중 보더 방지: 좌측 보더 제거 */
155
+ .tab + .tab {
156
+ border-left: 0;
157
+ }
158
+ .tab[active] {
159
+ background: #02a8a2;
160
+ color: #ffffff;
161
+ border-color: rgba(148, 163, 184, 0.5);
162
+ font-weight: 700;
163
+ }
164
+
165
+ .tab-body {
166
+ padding: 12px 8px 4px 8px;
167
+ }
168
+
169
+ .button-line {
170
+ display: flex;
171
+ justify-content: space-between;
172
+ align-items: center;
173
+ margin: 0 20px 10px 20px;
174
+ }
175
+ .spacer {
176
+ flex: 1;
177
+ }
178
+ .ghost-btn {
179
+ display: inline-flex;
180
+ align-items: center;
181
+ gap: 6px;
182
+ padding: 7px 12px;
183
+ background: #35618e;
184
+ color: #ffffff;
185
+ border-radius: 5px;
186
+ box-shadow: 2px 2px 2px 0 rgba(0, 0, 0, 0.1);
187
+ cursor: pointer;
188
+ }
189
+ .ghost-btn.secondary {
190
+ background: #3395f1;
191
+ }
192
+ .ghost-btn.complete {
193
+ background: #16a085;
194
+ }
195
+ `
196
+ ];
197
+ __decorate([
198
+ state(),
199
+ __metadata("design:type", Number)
200
+ ], SvProjectCompletePage.prototype, "activeTab", void 0);
201
+ __decorate([
202
+ state(),
203
+ __metadata("design:type", String)
204
+ ], SvProjectCompletePage.prototype, "projectId", void 0);
205
+ __decorate([
206
+ state(),
207
+ __metadata("design:type", Object)
208
+ ], SvProjectCompletePage.prototype, "project", void 0);
209
+ SvProjectCompletePage = __decorate([
210
+ customElement('sv-project-complete')
211
+ ], SvProjectCompletePage);
212
+ export { SvProjectCompletePage };
213
+ //# sourceMappingURL=sv-project-complete.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sv-project-complete.js","sourceRoot":"","sources":["../../client/pages/sv-project-complete.ts"],"names":[],"mappings":"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","sourcesContent":["import '@material/web/icon/icon.js'\n\nimport { navigate, PageView } from '@operato/shell'\nimport { css, html } from 'lit'\nimport { customElement, state } from 'lit/decorators.js'\nimport { ScopedElementsMixin } from '@open-wc/scoped-elements'\n\nimport './project-complete-tabs/pc-tab1-plan'\nimport './project-complete-tabs/pc-tab2-rating'\nimport './project-complete-tabs/pc-tab3-upload'\nimport './project-complete-tabs/pc-tab4-monthly'\nimport { getProject, getKpiCategories, updateProjectCompleteFinalize } from '../shared/complete-api'\nimport { notify } from '@operato/layout'\nimport { OxPrompt } from '@operato/popup'\n\n@customElement('sv-project-complete')\nexport class SvProjectCompletePage 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\n .page-header {\n display: flex;\n align-items: center;\n gap: 8px;\n padding: 16px 20px 8px 20px;\n }\n .page-title {\n color: #35618e;\n font-weight: 700;\n font-size: 22px;\n letter-spacing: -0.05em;\n }\n .triangle {\n width: 0;\n height: 0;\n border-left: 6px solid transparent;\n border-right: 6px solid transparent;\n border-top: 8px solid #35618e;\n }\n\n .card {\n background: #ffffff;\n border-radius: 8px;\n padding: 12px;\n box-shadow: 2px 2px 2px 0 rgba(0, 0, 0, 0.08);\n margin: 0 20px 20px 20px;\n }\n\n .tabs {\n display: flex;\n justify-content: center;\n padding: 0 12px 8px 12px;\n border-bottom: 1px dashed rgba(0, 0, 0, 0.15);\n }\n .tab {\n display: inline-flex;\n align-items: center;\n padding: 6px 14px;\n border: 1px solid rgba(0, 0, 0, 0.12);\n color: #35618e;\n background: #f3f6f9;\n cursor: pointer;\n font-size: 14px;\n letter-spacing: -0.02em;\n }\n .tab:first-child {\n border-top-left-radius: 10px;\n border-bottom-left-radius: 10px;\n }\n .tab:last-child {\n border-top-right-radius: 10px;\n border-bottom-right-radius: 10px;\n }\n /* 중간 탭 이중 보더 방지: 좌측 보더 제거 */\n .tab + .tab {\n border-left: 0;\n }\n .tab[active] {\n background: #02a8a2;\n color: #ffffff;\n border-color: rgba(148, 163, 184, 0.5);\n font-weight: 700;\n }\n\n .tab-body {\n padding: 12px 8px 4px 8px;\n }\n\n .button-line {\n display: flex;\n justify-content: space-between;\n align-items: center;\n margin: 0 20px 10px 20px;\n }\n .spacer {\n flex: 1;\n }\n .ghost-btn {\n display: inline-flex;\n align-items: center;\n gap: 6px;\n padding: 7px 12px;\n background: #35618e;\n color: #ffffff;\n border-radius: 5px;\n box-shadow: 2px 2px 2px 0 rgba(0, 0, 0, 0.1);\n cursor: pointer;\n }\n .ghost-btn.secondary {\n background: #3395f1;\n }\n .ghost-btn.complete {\n background: #16a085;\n }\n `\n ]\n\n get context() {\n return {\n title: this.project ? `프로젝트 완공 처리 - ${this.project?.name}` : '프로젝트 완공 처리'\n }\n }\n\n @state() private activeTab: number = 1\n @state() private projectId: string = ''\n @state() private project: any = {}\n\n render() {\n return html`\n <div class=\"page-header\">\n <div class=\"page-title\">프로젝트 완공 처리</div>\n <div class=\"triangle\"></div>\n <span class=\"spacer\"></span>\n <div class=\"ghost-btn\" @click=${() => navigate(`project-detail/${this.projectId}`)}>\n <md-icon>arrow_back</md-icon>\n <div>상세로 돌아가기</div>\n </div>\n <div class=\"ghost-btn complete\" @click=${this._onComplete}>\n <md-icon>check_circle</md-icon>\n <div>완공 처리</div>\n </div>\n </div>\n\n <div class=\"card\">\n <div class=\"tabs\">\n <div class=\"tab\" ?active=${this.activeTab === 1} @click=${() => this._onTabClick(1)}>\n Step1. 프로젝트 기본정보 현행화\n </div>\n <div class=\"tab\" ?active=${this.activeTab === 2} @click=${() => this._onTabClick(2)}>Step2. 프로젝트 완료 평가</div>\n <div class=\"tab\" ?active=${this.activeTab === 3} @click=${() => this._onTabClick(3)}>Step3. 준공 문서 업로드</div>\n <div class=\"tab\" ?active=${this.activeTab === 4} @click=${() => this._onTabClick(4)}>Step4. 월별 데이터 입력</div>\n </div>\n\n <div class=\"tab-body\">\n ${this.activeTab === 1\n ? html`<sv-pc-tab1-plan .project=${this.project}></sv-pc-tab1-plan>`\n : this.activeTab === 2\n ? html`<sv-pc-tab2-rating .project=${this.project}></sv-pc-tab2-rating>`\n : this.activeTab === 3\n ? html`<sv-pc-tab3-upload .project=${this.project}></sv-pc-tab3-upload>`\n : html`<sv-pc-tab4-monthly .project=${this.project}></sv-pc-tab4-monthly>`}\n </div>\n </div>\n `\n }\n\n async pageUpdated(changes: any, lifecycle: any) {\n if (this.active) {\n this.projectId = lifecycle.resourceId || ''\n\n this.project = await getProject(this.projectId) // View용 데이터\n }\n\n this.updateContext()\n }\n\n // 탭 클릭\n private _onTabClick = async (tabNumber: number) => {\n if (tabNumber === this.activeTab) return\n this.activeTab = tabNumber\n }\n\n // 완공 처리\n private _onComplete = async () => {\n if (\n await OxPrompt.open({ title: '완공 처리를 하시겠습니까?', confirmButton: { text: '확인' }, cancelButton: { text: '취소' } })\n ) {\n const result = await updateProjectCompleteFinalize(this.projectId)\n if (!result.errors) {\n notify({ message: '완공 처리가 완료되었습니다.', level: 'info' })\n navigate(`project-detail/${this.projectId}`)\n }\n }\n }\n}\n"]}
@@ -0,0 +1,27 @@
1
+ import '@material/web/icon/icon.js';
2
+ import '../components/kpi-single-boxplot-chart';
3
+ import { PageView } from '@operato/shell';
4
+ import '../components/sv-pagenation-control';
5
+ declare const SvProjectCompletedListPage_base: typeof PageView & import("@open-wc/dedupe-mixin").Constructor<import("@open-wc/scoped-elements/types/src/types").ScopedElementsHost>;
6
+ export declare class SvProjectCompletedListPage extends SvProjectCompletedListPage_base {
7
+ static styles: import("lit").CSSResult[];
8
+ get context(): {
9
+ title: string;
10
+ };
11
+ private projectName;
12
+ private projectList;
13
+ private projectCount;
14
+ private currentPage;
15
+ private kpiComprehensiveStats;
16
+ private readonly pageLimit;
17
+ get totalPages(): number;
18
+ render(): import("lit-html").TemplateResult<1>;
19
+ pageUpdated(changes: any, lifecycle: any): Promise<void>;
20
+ getProjectList(): Promise<void>;
21
+ getKpiComprehensiveStats(): Promise<void>;
22
+ private _onInputChange;
23
+ private _onKeypress;
24
+ private _changePage;
25
+ private getBoxPlotDataForProject;
26
+ }
27
+ export {};
@@ -0,0 +1,416 @@
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.kpiComprehensiveStats = [];
20
+ this.pageLimit = 20;
21
+ }
22
+ get context() {
23
+ return {
24
+ title: '완료 프로젝트'
25
+ };
26
+ }
27
+ get totalPages() {
28
+ return Math.max(1, Math.ceil((this.projectCount || 0) / this.pageLimit));
29
+ }
30
+ render() {
31
+ var _a;
32
+ return html `
33
+ <div header>
34
+ <div search>
35
+ <md-icon>manage_search</md-icon>
36
+ <input
37
+ type="search"
38
+ name="projectName"
39
+ placeholder="프로젝트명"
40
+ .value=${this.projectName}
41
+ @input=${this._onInputChange}
42
+ @keypress=${this._onKeypress}
43
+ />
44
+ </div>
45
+
46
+ <span class="spacer"></span>
47
+ <div count>총 ${this.projectCount}건</div>
48
+ </div>
49
+
50
+ <div table>
51
+ <div table-header>
52
+ <div col></div>
53
+ <div col>현장상태</div>
54
+ <div col>현장정보</div>
55
+ <div col>기타정보</div>
56
+ <div col>입력요청</div>
57
+ <div col>종합KPI</div>
58
+ </div>
59
+ <div table-body>
60
+ ${(_a = this.projectList) === null || _a === void 0 ? void 0 : _a.map((project) => {
61
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j;
62
+ 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()) || '-'} ㎡`;
63
+ 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()) || '-'} 억원`;
64
+ 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()) || '-'} 세대`;
65
+ const kpiValue = project.kpi ? project.kpi : 0;
66
+ return html `
67
+ <div tr @click=${() => navigate(`project-detail/${project.id}`)}>
68
+ <div td-pics>
69
+ <img pic src=${((_g = project.mainPhoto) === null || _g === void 0 ? void 0 : _g.fullpath) || '/assets/images/project-image.png'} alt="${project.name}" />
70
+ </div>
71
+ <div td-status>${PROJECT_STATE[project.state]}</div>
72
+ <div td-info>
73
+ <div class="name">${project.name}</div>
74
+ <div class="sub">주소 : ${((_h = project.buildingComplex) === null || _h === void 0 ? void 0 : _h.address) || ''}</div>
75
+ <div class="sub">착공~준공 : ${project.startDate} ~ ${project.endDate}</div>
76
+ </div>
77
+ <div td-etc>${areaText + '\n' + costText + '\n' + householdCountText}</div>
78
+ <div td-request>KPI입력 요청 - 건</div>
79
+ <div td-kpi>
80
+ <kpi-single-boxplot-chart .data=${this.getBoxPlotDataForProject(project)}></kpi-single-boxplot-chart>
81
+ <span class="kpi-value">${(_j = kpiValue === null || kpiValue === void 0 ? void 0 : kpiValue.toFixed(0)) !== null && _j !== void 0 ? _j : '-'}</span>
82
+ </div>
83
+ </div>
84
+ `;
85
+ })}
86
+ </div>
87
+ </div>
88
+ <sv-pagenation-control
89
+ .currentPage=${this.currentPage}
90
+ .totalItems=${this.projectCount}
91
+ .pageLimit=${this.pageLimit}
92
+ @page-change=${(e) => this._changePage(e.detail.page)}
93
+ ></sv-pagenation-control>
94
+ `;
95
+ }
96
+ async pageUpdated(changes, lifecycle) {
97
+ if (this.active) {
98
+ this.getProjectList();
99
+ this.getKpiComprehensiveStats();
100
+ }
101
+ }
102
+ async getProjectList() {
103
+ var _a, _b;
104
+ const response = await client.query({
105
+ query: gql `
106
+ query Projects($filters: [Filter!], $sortings: [Sorting!], $pagination: Pagination) {
107
+ projects(filters: $filters, sortings: $sortings, pagination: $pagination) {
108
+ items {
109
+ id
110
+ name
111
+ state
112
+ startDate
113
+ endDate
114
+ geoGroup
115
+ mainPhoto {
116
+ fullpath
117
+ }
118
+ totalProgress
119
+ weeklyProgress
120
+ kpi
121
+ inspPassRate
122
+ robotProgressRate
123
+ structuralSafetyRate
124
+ buildingComplex {
125
+ address
126
+ area
127
+ clientCompany
128
+ constructionCost
129
+ householdCount
130
+ }
131
+ }
132
+ total
133
+ }
134
+ }
135
+ `,
136
+ variables: {
137
+ filters: [
138
+ { name: 'name', operator: 'search', value: `%${this.projectName}%` },
139
+ { name: 'state', operator: 'eq', value: ProjectState.COMPLETED }
140
+ ],
141
+ sortings: [{ name: 'createdAt', desc: true }],
142
+ pagination: { page: this.currentPage, limit: this.pageLimit }
143
+ }
144
+ });
145
+ this.projectList = ((_a = response.data.projects) === null || _a === void 0 ? void 0 : _a.items) || [];
146
+ this.projectCount = ((_b = response.data.projects) === null || _b === void 0 ? void 0 : _b.total) || 0;
147
+ }
148
+ async getKpiComprehensiveStats() {
149
+ try {
150
+ const response = await client.query({
151
+ query: gql `
152
+ query GetKpiZValueComprehensiveStats {
153
+ totalKpiZValueComprehensiveStats {
154
+ minVal
155
+ q1Val
156
+ medVal
157
+ q3Val
158
+ maxVal
159
+ }
160
+ }
161
+ `
162
+ });
163
+ this.kpiComprehensiveStats = response.data.totalKpiZValueComprehensiveStats || [];
164
+ }
165
+ catch (error) {
166
+ console.error('Failed to fetch KPI comprehensive stats:', error);
167
+ this.kpiComprehensiveStats = [];
168
+ }
169
+ }
170
+ // Input 요소의 값이 변경될 때 호출되는 콜백 함수
171
+ _onInputChange(event) {
172
+ const target = event.target;
173
+ this[target.name] = target.value;
174
+ if (target.name === 'projectName') {
175
+ this.currentPage = 1;
176
+ }
177
+ }
178
+ // 검색창에서 엔터입력시 검색
179
+ _onKeypress(event) {
180
+ if (event.code === 'Enter') {
181
+ this.currentPage = 1;
182
+ this.getProjectList();
183
+ }
184
+ }
185
+ _changePage(page) {
186
+ const nextPage = Math.min(Math.max(1, page), this.totalPages);
187
+ if (nextPage !== this.currentPage) {
188
+ this.currentPage = nextPage;
189
+ this.getProjectList();
190
+ }
191
+ }
192
+ getBoxPlotDataForProject(project) {
193
+ var _a;
194
+ // 전체 프로젝트의 통계 찾기
195
+ const stats = (_a = this.kpiComprehensiveStats) === null || _a === void 0 ? void 0 : _a[0];
196
+ if (!stats) {
197
+ // 전체 통계의 평균값 사용 또는 기본값 반환
198
+ const defaultStats = this.kpiComprehensiveStats.length > 0 ? this.kpiComprehensiveStats[0] : null;
199
+ if (defaultStats) {
200
+ return {
201
+ min: defaultStats.minVal,
202
+ max: defaultStats.maxVal,
203
+ mean: (defaultStats.q1Val + defaultStats.q3Val) / 2,
204
+ median: defaultStats.medVal,
205
+ q1: defaultStats.q1Val,
206
+ q3: defaultStats.q3Val,
207
+ value: project.kpi || 0
208
+ };
209
+ }
210
+ // 완전한 기본값
211
+ return {
212
+ min: 0,
213
+ max: 100,
214
+ mean: 50,
215
+ median: 50,
216
+ q1: 25,
217
+ q3: 75,
218
+ value: project.kpi || 0
219
+ };
220
+ }
221
+ return {
222
+ min: stats.minVal * 100,
223
+ max: stats.maxVal * 100,
224
+ mean: ((stats.q1Val + stats.q3Val) / 2) * 100, // 대략적 평균값
225
+ median: stats.medVal * 100,
226
+ q1: stats.q1Val * 100,
227
+ q3: stats.q3Val * 100,
228
+ value: project.kpi || 0
229
+ };
230
+ }
231
+ };
232
+ SvProjectCompletedListPage.styles = [
233
+ css `
234
+ :host {
235
+ display: flex;
236
+ flex-direction: column;
237
+ overflow-y: auto;
238
+
239
+ width: 100%;
240
+ height: 100%;
241
+ background-color: var(--md-sys-color-background, #f6f6f6);
242
+
243
+ --grid-record-emphasized-background-color: red;
244
+ --grid-record-emphasized-color: yellow;
245
+ }
246
+
247
+ /* Search bar (Figma: content > search bar) */
248
+ div[header] {
249
+ display: flex;
250
+ align-items: center;
251
+ gap: 10px;
252
+ margin: var(--spacing-large, 12px);
253
+ margin-bottom: 0;
254
+ padding: 12px 11px;
255
+ border-radius: 7px;
256
+ background-color: rgba(46, 164, 223, 0.1);
257
+ border: 1px solid rgba(46, 164, 223, 0.3);
258
+ }
259
+ div[header] div[search] {
260
+ display: flex;
261
+ align-items: center;
262
+ gap: 6px;
263
+ min-width: 186px;
264
+ }
265
+ div[header] div[search] md-icon {
266
+ color: #212529;
267
+ }
268
+ div[header] div[search] input[type='search'] {
269
+ border: none;
270
+ outline: none;
271
+ background: transparent;
272
+ font-size: 16px;
273
+ color: #212529;
274
+ padding: 2px 0;
275
+ border-bottom: 1px solid rgba(0, 0, 0, 0.2);
276
+ }
277
+ div[header] .spacer {
278
+ flex: 1;
279
+ }
280
+ div[header] div[count] {
281
+ font-size: 16px;
282
+ color: #000000;
283
+ margin-right: 6px;
284
+ }
285
+ div[header] md-icon[view] {
286
+ color: #35618e;
287
+ cursor: pointer;
288
+ }
289
+ div[header] md-icon[add] {
290
+ color: #4cbb49;
291
+ cursor: pointer;
292
+ }
293
+
294
+ /* Table (Figma: content > table) */
295
+ div[table] {
296
+ margin: var(--spacing-large, 12px);
297
+ border: 1px solid rgba(0, 0, 0, 0.15);
298
+ border-top: 2px solid #0c4da2;
299
+ border-radius: var(--md-sys-shape-corner-small, 5px);
300
+ overflow: hidden;
301
+ background: #ffffff;
302
+ overflow-y: auto;
303
+ min-height: fit-content;
304
+ }
305
+ div[table-header] {
306
+ display: grid;
307
+ grid-template-columns: 150px 80px 1fr 220px 150px 100px;
308
+ align-items: center;
309
+ justify-items: center;
310
+ height: 35px;
311
+ background: #f3f3fa;
312
+ padding: 0 25px;
313
+ column-gap: 30px;
314
+ }
315
+ div[table-header] div[col] {
316
+ font-size: 16px;
317
+ color: #212529;
318
+ line-height: 1.875em;
319
+ }
320
+ div[table-body] {
321
+ display: block;
322
+ }
323
+ div[tr] {
324
+ display: grid;
325
+ grid-template-columns: 150px 80px 1fr 220px 150px 100px;
326
+ align-items: center;
327
+ justify-items: center;
328
+ column-gap: 30px;
329
+ padding: 12px 25px;
330
+ background: #ffffff;
331
+ border-top: 1px solid rgba(0, 0, 0, 0.15);
332
+ cursor: pointer;
333
+ }
334
+ /* 좌측 정렬 요구사항 */
335
+ div[td-info],
336
+ div[td-etc] {
337
+ justify-self: start;
338
+ text-align: left;
339
+ }
340
+ div[td-pics] img[pic] {
341
+ width: 150px;
342
+ height: 90px;
343
+ object-fit: fill;
344
+ background: #e5e7eb;
345
+ border-radius: 2px;
346
+ }
347
+ div[td-status] {
348
+ font-weight: 700;
349
+ font-size: 16px;
350
+ color: #4cbb49;
351
+ line-height: 1;
352
+ }
353
+ div[td-info] {
354
+ display: flex;
355
+ flex-direction: column;
356
+ gap: 4px;
357
+ }
358
+ div[td-info] .name {
359
+ font-weight: 700;
360
+ font-size: 14px;
361
+ color: #000000;
362
+ }
363
+ div[td-info] .sub {
364
+ font-size: 14px;
365
+ color: #000000;
366
+ line-height: 1.7;
367
+ }
368
+ div[td-etc] {
369
+ font-size: 16px;
370
+ color: #212529;
371
+ white-space: pre-line;
372
+ }
373
+ div[td-request] {
374
+ font-weight: 700;
375
+ font-size: 16px;
376
+ color: #35618e;
377
+ white-space: pre-line;
378
+ }
379
+ div[td-kpi] {
380
+ display: flex;
381
+ align-items: center;
382
+ height: 100%;
383
+ }
384
+ div[td-kpi] .kpi-value {
385
+ font-weight: 700;
386
+ font-size: 22px;
387
+ color: #35618e;
388
+ width: 30px;
389
+ }
390
+ `
391
+ ];
392
+ __decorate([
393
+ state(),
394
+ __metadata("design:type", String)
395
+ ], SvProjectCompletedListPage.prototype, "projectName", void 0);
396
+ __decorate([
397
+ state(),
398
+ __metadata("design:type", Array)
399
+ ], SvProjectCompletedListPage.prototype, "projectList", void 0);
400
+ __decorate([
401
+ state(),
402
+ __metadata("design:type", Number)
403
+ ], SvProjectCompletedListPage.prototype, "projectCount", void 0);
404
+ __decorate([
405
+ state(),
406
+ __metadata("design:type", Number)
407
+ ], SvProjectCompletedListPage.prototype, "currentPage", void 0);
408
+ __decorate([
409
+ state(),
410
+ __metadata("design:type", Array)
411
+ ], SvProjectCompletedListPage.prototype, "kpiComprehensiveStats", void 0);
412
+ SvProjectCompletedListPage = __decorate([
413
+ customElement('sv-project-completed-list')
414
+ ], SvProjectCompletedListPage);
415
+ export { SvProjectCompletedListPage };
416
+ //# sourceMappingURL=sv-project-completed-list.js.map