@dssp/dkpi 1.0.0-alpha.8 → 1.0.0-alpha.81

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 (252) 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/config/config.development.js +2 -1
  25. package/config/config.production.js +2 -1
  26. package/dist-client/bootstrap.d.ts +1 -0
  27. package/dist-client/bootstrap.js +75 -4
  28. package/dist-client/bootstrap.js.map +1 -1
  29. package/dist-client/components/kpi-2d-lookup-chart.d.ts +43 -0
  30. package/dist-client/components/kpi-2d-lookup-chart.js +253 -0
  31. package/dist-client/components/kpi-2d-lookup-chart.js.map +1 -0
  32. package/dist-client/components/kpi-boxplot-chart.d.ts +24 -0
  33. package/dist-client/components/kpi-boxplot-chart.js +291 -0
  34. package/dist-client/components/kpi-boxplot-chart.js.map +1 -0
  35. package/dist-client/components/kpi-lookup-chart.d.ts +53 -0
  36. package/dist-client/components/kpi-lookup-chart.js +430 -0
  37. package/dist-client/components/kpi-lookup-chart.js.map +1 -0
  38. package/dist-client/components/kpi-mini-trend-chart.d.ts +14 -0
  39. package/dist-client/components/kpi-mini-trend-chart.js +148 -0
  40. package/dist-client/components/kpi-mini-trend-chart.js.map +1 -0
  41. package/dist-client/components/kpi-radar-chart.d.ts +17 -0
  42. package/dist-client/components/kpi-radar-chart.js +259 -0
  43. package/dist-client/components/kpi-radar-chart.js.map +1 -0
  44. package/dist-client/components/kpi-single-boxplot-chart.d.ts +25 -0
  45. package/dist-client/components/kpi-single-boxplot-chart.js +398 -0
  46. package/dist-client/components/kpi-single-boxplot-chart.js.map +1 -0
  47. package/dist-client/components/kpi-trend-chart.d.ts +25 -0
  48. package/dist-client/components/kpi-trend-chart.js +220 -0
  49. package/dist-client/components/kpi-trend-chart.js.map +1 -0
  50. package/dist-client/components/sv-pagenation-control.d.ts +18 -0
  51. package/dist-client/components/sv-pagenation-control.js +142 -0
  52. package/dist-client/components/sv-pagenation-control.js.map +1 -0
  53. package/dist-client/entries/auth/checkin.d.ts +38 -0
  54. package/dist-client/entries/auth/checkin.js +546 -0
  55. package/dist-client/entries/auth/checkin.js.map +1 -0
  56. package/dist-client/google-map/common-google-map.d.ts +35 -0
  57. package/dist-client/google-map/common-google-map.js +345 -0
  58. package/dist-client/google-map/common-google-map.js.map +1 -0
  59. package/dist-client/google-map/google-map-loader.d.ts +6 -0
  60. package/dist-client/google-map/google-map-loader.js +23 -0
  61. package/dist-client/google-map/google-map-loader.js.map +1 -0
  62. package/dist-client/icons/menu-icons.d.ts +6 -0
  63. package/dist-client/icons/menu-icons.js +42 -0
  64. package/dist-client/icons/menu-icons.js.map +1 -1
  65. package/dist-client/pages/component/project-update-header.d.ts +1 -0
  66. package/dist-client/pages/component/project-update-header.js +127 -0
  67. package/dist-client/pages/component/project-update-header.js.map +1 -0
  68. package/dist-client/pages/kpi-admin/dssp-kpi-list-page.d.ts +22 -0
  69. package/dist-client/pages/kpi-admin/dssp-kpi-list-page.js +57 -0
  70. package/dist-client/pages/kpi-admin/dssp-kpi-list-page.js.map +1 -0
  71. package/dist-client/pages/kpi-admin/dssp-kpi-overview.d.ts +46 -0
  72. package/dist-client/pages/kpi-admin/dssp-kpi-overview.js +378 -0
  73. package/dist-client/pages/kpi-admin/dssp-kpi-overview.js.map +1 -0
  74. package/dist-client/pages/kpi-admin/kpi-grade-2d-editor.d.ts +20 -0
  75. package/dist-client/pages/kpi-admin/kpi-grade-2d-editor.js +445 -0
  76. package/dist-client/pages/kpi-admin/kpi-grade-2d-editor.js.map +1 -0
  77. package/dist-client/pages/kpi-admin/kpi-system-guide.d.ts +18 -0
  78. package/dist-client/pages/kpi-admin/kpi-system-guide.js +535 -0
  79. package/dist-client/pages/kpi-admin/kpi-system-guide.js.map +1 -0
  80. package/dist-client/pages/kpi-dashboard/cards/kpi-level1-card.d.ts +18 -0
  81. package/dist-client/pages/kpi-dashboard/cards/kpi-level1-card.js +259 -0
  82. package/dist-client/pages/kpi-dashboard/cards/kpi-level1-card.js.map +1 -0
  83. package/dist-client/pages/kpi-dashboard/cards/kpi-level2-comparison.d.ts +22 -0
  84. package/dist-client/pages/kpi-dashboard/cards/kpi-level2-comparison.js +346 -0
  85. package/dist-client/pages/kpi-dashboard/cards/kpi-level2-comparison.js.map +1 -0
  86. package/dist-client/pages/kpi-dashboard/cards/kpi-level3-comparison.d.ts +26 -0
  87. package/dist-client/pages/kpi-dashboard/cards/kpi-level3-comparison.js +433 -0
  88. package/dist-client/pages/kpi-dashboard/cards/kpi-level3-comparison.js.map +1 -0
  89. package/dist-client/pages/kpi-dashboard/components/kpi-chart-toggle.d.ts +8 -0
  90. package/dist-client/pages/kpi-dashboard/components/kpi-chart-toggle.js +78 -0
  91. package/dist-client/pages/kpi-dashboard/components/kpi-chart-toggle.js.map +1 -0
  92. package/dist-client/pages/kpi-dashboard/components/kpi-left-panel.d.ts +38 -0
  93. package/dist-client/pages/kpi-dashboard/components/kpi-left-panel.js +851 -0
  94. package/dist-client/pages/kpi-dashboard/components/kpi-left-panel.js.map +1 -0
  95. package/dist-client/pages/kpi-dashboard/components/kpi-map-panel.d.ts +42 -0
  96. package/dist-client/pages/kpi-dashboard/components/kpi-map-panel.js +316 -0
  97. package/dist-client/pages/kpi-dashboard/components/kpi-map-panel.js.map +1 -0
  98. package/dist-client/pages/kpi-dashboard/components/kpi-region-popup.d.ts +28 -0
  99. package/dist-client/pages/kpi-dashboard/components/kpi-region-popup.js +497 -0
  100. package/dist-client/pages/kpi-dashboard/components/kpi-region-popup.js.map +1 -0
  101. package/dist-client/pages/kpi-dashboard/kpi-alert-panel.d.ts +18 -0
  102. package/dist-client/pages/kpi-dashboard/kpi-alert-panel.js +131 -0
  103. package/dist-client/pages/kpi-dashboard/kpi-alert-panel.js.map +1 -0
  104. package/dist-client/pages/kpi-dashboard/kpi-dashboard-map.d.ts +52 -0
  105. package/dist-client/pages/kpi-dashboard/kpi-dashboard-map.js +798 -0
  106. package/dist-client/pages/kpi-dashboard/kpi-dashboard-map.js.map +1 -0
  107. package/dist-client/pages/kpi-dashboard/kpi-dashboard.d.ts +63 -0
  108. package/dist-client/pages/kpi-dashboard/kpi-dashboard.js +1089 -0
  109. package/dist-client/pages/kpi-dashboard/kpi-dashboard.js.map +1 -0
  110. package/dist-client/pages/kpi-dashboard/kpi-grade-visualization.d.ts +12 -0
  111. package/dist-client/pages/kpi-dashboard/kpi-grade-visualization.js +82 -0
  112. package/dist-client/pages/kpi-dashboard/kpi-grade-visualization.js.map +1 -0
  113. package/dist-client/pages/kpi-dashboard/kpi-history-viewer.d.ts +11 -0
  114. package/dist-client/pages/kpi-dashboard/kpi-history-viewer.js +65 -0
  115. package/dist-client/pages/kpi-dashboard/kpi-history-viewer.js.map +1 -0
  116. package/dist-client/pages/kpi-dashboard/kpi-list-summary.d.ts +13 -0
  117. package/dist-client/pages/kpi-dashboard/kpi-list-summary.js +115 -0
  118. package/dist-client/pages/kpi-dashboard/kpi-list-summary.js.map +1 -0
  119. package/dist-client/pages/kpi-dashboard/kpi-performance-summary.d.ts +15 -0
  120. package/dist-client/pages/kpi-dashboard/kpi-performance-summary.js +147 -0
  121. package/dist-client/pages/kpi-dashboard/kpi-performance-summary.js.map +1 -0
  122. package/dist-client/pages/kpi-dashboard/kpi-value-entry.d.ts +7 -0
  123. package/dist-client/pages/kpi-dashboard/kpi-value-entry.js +86 -0
  124. package/dist-client/pages/kpi-dashboard/kpi-value-entry.js.map +1 -0
  125. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.d.ts +57 -0
  126. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js +719 -0
  127. package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js.map +1 -0
  128. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.d.ts +23 -0
  129. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js +76 -0
  130. package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js.map +1 -0
  131. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.d.ts +68 -0
  132. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js +394 -0
  133. package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js.map +1 -0
  134. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.d.ts +12 -0
  135. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js +174 -0
  136. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js.map +1 -0
  137. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.d.ts +40 -0
  138. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js +190 -0
  139. package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js.map +1 -0
  140. package/dist-client/pages/kpi-value/kpi-value-importer.d.ts +23 -0
  141. package/dist-client/pages/kpi-value/kpi-value-importer.js +93 -0
  142. package/dist-client/pages/kpi-value/kpi-value-importer.js.map +1 -0
  143. package/dist-client/pages/kpi-value/kpi-value-list-page.d.ts +71 -0
  144. package/dist-client/pages/kpi-value/kpi-value-list-page.js +454 -0
  145. package/dist-client/pages/kpi-value/kpi-value-list-page.js.map +1 -0
  146. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.d.ts +61 -0
  147. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.js +766 -0
  148. package/dist-client/pages/project-complete-tabs/pc-tab1-plan.js.map +1 -0
  149. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.d.ts +16 -0
  150. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.js +341 -0
  151. package/dist-client/pages/project-complete-tabs/pc-tab2-rating.js.map +1 -0
  152. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.d.ts +21 -0
  153. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.js +335 -0
  154. package/dist-client/pages/project-complete-tabs/pc-tab3-upload.js.map +1 -0
  155. package/dist-client/pages/project-complete-tabs/pc-tab4-monthly.d.ts +42 -0
  156. package/dist-client/pages/project-complete-tabs/pc-tab4-monthly.js +641 -0
  157. package/dist-client/pages/project-complete-tabs/pc-tab4-monthly.js.map +1 -0
  158. package/dist-client/pages/sv-project-complete.d.ts +25 -0
  159. package/dist-client/pages/sv-project-complete.js +263 -0
  160. package/dist-client/pages/sv-project-complete.js.map +1 -0
  161. package/dist-client/pages/sv-project-completed-list.d.ts +27 -0
  162. package/dist-client/pages/sv-project-completed-list.js +416 -0
  163. package/dist-client/pages/sv-project-completed-list.js.map +1 -0
  164. package/dist-client/pages/sv-project-detail.d.ts +58 -0
  165. package/dist-client/pages/sv-project-detail.js +1418 -0
  166. package/dist-client/pages/sv-project-detail.js.map +1 -0
  167. package/dist-client/pages/sv-project-list.d.ts +174 -0
  168. package/dist-client/pages/sv-project-list.js +578 -0
  169. package/dist-client/pages/sv-project-list.js.map +1 -0
  170. package/dist-client/pages/sv-project-update.d.ts +86 -0
  171. package/dist-client/pages/sv-project-update.js +1331 -0
  172. package/dist-client/pages/sv-project-update.js.map +1 -0
  173. package/dist-client/pages/sv-user-management.d.ts +1 -0
  174. package/dist-client/pages/sv-user-management.js +5 -0
  175. package/dist-client/pages/sv-user-management.js.map +1 -1
  176. package/dist-client/route.d.ts +1 -1
  177. package/dist-client/route.js +50 -2
  178. package/dist-client/route.js.map +1 -1
  179. package/dist-client/shared/complete-api.d.ts +37 -0
  180. package/dist-client/shared/complete-api.js +224 -0
  181. package/dist-client/shared/complete-api.js.map +1 -0
  182. package/dist-client/shared/func.d.ts +2 -0
  183. package/dist-client/shared/func.js +22 -0
  184. package/dist-client/shared/func.js.map +1 -0
  185. package/dist-client/shared/integration-fetch.d.ts +35 -0
  186. package/dist-client/shared/integration-fetch.js +53 -0
  187. package/dist-client/shared/integration-fetch.js.map +1 -0
  188. package/dist-client/themes/dark.css +24 -24
  189. package/dist-client/themes/light.css +23 -23
  190. package/dist-client/tsconfig.tsbuildinfo +1 -1
  191. package/dist-client/viewparts/menu-tools.d.ts +40 -5
  192. package/dist-client/viewparts/menu-tools.js +292 -43
  193. package/dist-client/viewparts/menu-tools.js.map +1 -1
  194. package/dist-server/index.d.ts +2 -0
  195. package/dist-server/index.js +5 -0
  196. package/dist-server/index.js.map +1 -1
  197. package/dist-server/migrations/index.d.ts +1 -0
  198. package/dist-server/migrations/index.js +12 -0
  199. package/dist-server/migrations/index.js.map +1 -0
  200. package/dist-server/scripts/calculate-kpi-scores.d.ts +10 -0
  201. package/dist-server/scripts/calculate-kpi-scores.js +333 -0
  202. package/dist-server/scripts/calculate-kpi-scores.js.map +1 -0
  203. package/dist-server/scripts/load-grade-data-migration.d.ts +14 -0
  204. package/dist-server/scripts/load-grade-data-migration.js +279 -0
  205. package/dist-server/scripts/load-grade-data-migration.js.map +1 -0
  206. package/dist-server/scripts/propagate-parent-kpi-values.d.ts +14 -0
  207. package/dist-server/scripts/propagate-parent-kpi-values.js +786 -0
  208. package/dist-server/scripts/propagate-parent-kpi-values.js.map +1 -0
  209. package/dist-server/scripts/recalculate-by-project-name.d.ts +2 -0
  210. package/dist-server/scripts/recalculate-by-project-name.js +72 -0
  211. package/dist-server/scripts/recalculate-by-project-name.js.map +1 -0
  212. package/dist-server/service/index.d.ts +4 -0
  213. package/dist-server/service/index.js +20 -0
  214. package/dist-server/service/index.js.map +1 -0
  215. package/dist-server/service/kpi-metric-value/index.d.ts +4 -0
  216. package/dist-server/service/kpi-metric-value/index.js +8 -0
  217. package/dist-server/service/kpi-metric-value/index.js.map +1 -0
  218. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.d.ts +166 -0
  219. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.js +951 -0
  220. package/dist-server/service/kpi-metric-value/kpi-metric-value-mutation.js.map +1 -0
  221. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.d.ts +7 -0
  222. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js +59 -0
  223. package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js.map +1 -0
  224. package/dist-server/service/kpi-stat/index.d.ts +4 -0
  225. package/dist-server/service/kpi-stat/index.js +8 -0
  226. package/dist-server/service/kpi-stat/index.js.map +1 -0
  227. package/dist-server/service/kpi-stat/kpi-stat-query.d.ts +12 -0
  228. package/dist-server/service/kpi-stat/kpi-stat-query.js +663 -0
  229. package/dist-server/service/kpi-stat/kpi-stat-query.js.map +1 -0
  230. package/dist-server/service/kpi-stat/kpi-stat-types.d.ts +32 -0
  231. package/dist-server/service/kpi-stat/kpi-stat-types.js +180 -0
  232. package/dist-server/service/kpi-stat/kpi-stat-types.js.map +1 -0
  233. package/dist-server/service/kpi-value/index.d.ts +3 -0
  234. package/dist-server/service/kpi-value/index.js +7 -0
  235. package/dist-server/service/kpi-value/index.js.map +1 -0
  236. package/dist-server/service/kpi-value/kpi-value-query.d.ts +8 -0
  237. package/dist-server/service/kpi-value/kpi-value-query.js +63 -0
  238. package/dist-server/service/kpi-value/kpi-value-query.js.map +1 -0
  239. package/dist-server/tsconfig.tsbuildinfo +1 -1
  240. package/kpi-module-service-tests.md +1286 -0
  241. package/kpi-module-test-report.md +676 -0
  242. package/kpi-module-unit-test-detailed-report.md +925 -0
  243. package/kpi-module-unit-tests-detailed.md +1452 -0
  244. package/package.json +69 -55
  245. package/recalculate-batch.sh +64 -0
  246. package/recalculate-projects-range.sh +98 -0
  247. package/schema.graphql +3423 -462
  248. package/things-factory.config.js +12 -1
  249. package/translations/en.json +5 -1
  250. package/translations/ko.json +5 -1
  251. package/views/auth-page.html +0 -1
  252. package/views/public/home.html +0 -1
@@ -0,0 +1,578 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import '@material/web/icon/icon.js';
3
+ import '../components/kpi-single-boxplot-chart';
4
+ import { navigate, PageView } 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 { openPopup } from '@operato/layout';
10
+ import gql from 'graphql-tag';
11
+ import '../components/sv-pagenation-control';
12
+ import '@dssp/project/dist-client/pages/project/popup/popup-project-create';
13
+ import { tenantHeaders } from '@dssp/project/dist-client/shared/domain-context';
14
+ import moment from 'moment-timezone';
15
+ export var ProjectState;
16
+ (function (ProjectState) {
17
+ ProjectState["ONGOING"] = "10";
18
+ ProjectState["COMPLETED"] = "20";
19
+ })(ProjectState || (ProjectState = {}));
20
+ export const PROJECT_STATE = {
21
+ [ProjectState.ONGOING]: '진행중',
22
+ [ProjectState.COMPLETED]: '완료'
23
+ };
24
+ export var BuildingInspectionStatus;
25
+ (function (BuildingInspectionStatus) {
26
+ BuildingInspectionStatus["WAIT"] = "WAIT";
27
+ BuildingInspectionStatus["OVERALL_WAIT"] = "OVERALL_WAIT";
28
+ BuildingInspectionStatus["REQUEST"] = "REQUEST";
29
+ BuildingInspectionStatus["OVERALL_REQUEST"] = "OVERALL_REQUEST";
30
+ BuildingInspectionStatus["PASS"] = "PASS";
31
+ BuildingInspectionStatus["FAIL"] = "FAIL";
32
+ })(BuildingInspectionStatus || (BuildingInspectionStatus = {}));
33
+ export const BUILDING_INSPECTION_STATUS = {
34
+ [BuildingInspectionStatus.WAIT]: '작업 완료',
35
+ [BuildingInspectionStatus.OVERALL_WAIT]: '작업 완료',
36
+ [BuildingInspectionStatus.REQUEST]: '감리 대기',
37
+ [BuildingInspectionStatus.OVERALL_REQUEST]: '감리 대기',
38
+ [BuildingInspectionStatus.PASS]: '검측 완료',
39
+ [BuildingInspectionStatus.FAIL]: '조치 필요'
40
+ };
41
+ export var ProjectType;
42
+ (function (ProjectType) {
43
+ ProjectType["DSSP"] = "DSSP";
44
+ ProjectType["DCSP"] = "DCSP";
45
+ ProjectType["DKPI"] = "DKPI";
46
+ })(ProjectType || (ProjectType = {}));
47
+ let SvProjectListPage = class SvProjectListPage extends ScopedElementsMixin(PageView) {
48
+ constructor() {
49
+ super(...arguments);
50
+ this.projectName = '';
51
+ this.projectList = [];
52
+ this.projectCount = 0;
53
+ this.currentPage = 1;
54
+ /** projectId → 미입력 KPI 건수 */
55
+ this.pendingCounts = {};
56
+ this.kpiComprehensiveStats = [];
57
+ this.pageLimit = 20;
58
+ }
59
+ get context() {
60
+ return {
61
+ title: '진행중 프로젝트'
62
+ };
63
+ }
64
+ get totalPages() {
65
+ return Math.max(1, Math.ceil((this.projectCount || 0) / this.pageLimit));
66
+ }
67
+ render() {
68
+ var _a;
69
+ return html `
70
+ <div header>
71
+ <div search>
72
+ <md-icon>manage_search</md-icon>
73
+ <input
74
+ type="search"
75
+ name="projectName"
76
+ placeholder="프로젝트명"
77
+ .value=${this.projectName}
78
+ @input=${this._onInputChange}
79
+ @keypress=${this._onKeypress}
80
+ />
81
+ </div>
82
+
83
+ <span class="spacer"></span>
84
+ <div count>총 ${this.projectCount}건</div>
85
+
86
+ <button add-project @click=${this._openCreateProjectPopup}>
87
+ <md-icon>add</md-icon>
88
+ 신규 프로젝트 생성
89
+ </button>
90
+ </div>
91
+
92
+ <div table>
93
+ <div table-header>
94
+ <div col></div>
95
+ <div col>현장상태</div>
96
+ <div col>현장정보</div>
97
+ <div col>기타정보</div>
98
+ <div col>입력요청</div>
99
+ <div col>종합KPI</div>
100
+ </div>
101
+ <div table-body>
102
+ ${(_a = this.projectList) === null || _a === void 0 ? void 0 : _a.map((project) => {
103
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l;
104
+ 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()) || '-'} ㎡`;
105
+ 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()) || '-'} 억원`;
106
+ 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()) || '-'} 세대`;
107
+ return html `
108
+ <div tr @click=${() => navigate(`project-detail/${project.id}`)}>
109
+ <div td-pics>
110
+ <img pic src=${((_g = project.mainPhoto) === null || _g === void 0 ? void 0 : _g.fullpath) || '/assets/images/project-image.png'} alt="${project.name}" />
111
+ </div>
112
+ <div td-status>${PROJECT_STATE[project.state]}</div>
113
+ <div td-info>
114
+ <div class="name">${project.name}</div>
115
+ <div class="sub">주소 : ${((_h = project.buildingComplex) === null || _h === void 0 ? void 0 : _h.address) || ''}</div>
116
+ <div class="sub">착공~준공 : ${project.startDate} ~ ${project.endDate}</div>
117
+ </div>
118
+ <div td-etc>${areaText + '\n' + costText + '\n' + householdCountText}</div>
119
+ <div td-request>KPI입력 요청 ${(_j = this.pendingCounts[project.id]) !== null && _j !== void 0 ? _j : '-'}건</div>
120
+ <div td-kpi>
121
+ <kpi-single-boxplot-chart .data=${this.getBoxPlotDataForProject(project)}></kpi-single-boxplot-chart>
122
+ <span class="kpi-value">${(_l = (_k = project.kpi) === null || _k === void 0 ? void 0 : _k.toFixed(0)) !== null && _l !== void 0 ? _l : '-'}</span>
123
+ </div>
124
+ </div>
125
+ `;
126
+ })}
127
+ </div>
128
+ </div>
129
+ <sv-pagenation-control
130
+ .currentPage=${this.currentPage}
131
+ .totalItems=${this.projectCount}
132
+ .pageLimit=${this.pageLimit}
133
+ @page-change=${(e) => this._changePage(e.detail.page)}
134
+ ></sv-pagenation-control>
135
+ `;
136
+ }
137
+ async pageUpdated(changes, lifecycle) {
138
+ if (this.active) {
139
+ this.getProjectList();
140
+ this.getKpiComprehensiveStats();
141
+ }
142
+ }
143
+ async getProjectList() {
144
+ var _a, _b;
145
+ const response = await client.query({
146
+ query: gql `
147
+ query Projects($filters: [Filter!], $sortings: [Sorting!], $pagination: Pagination) {
148
+ projects(filters: $filters, sortings: $sortings, pagination: $pagination) {
149
+ items {
150
+ id
151
+ code
152
+ name
153
+ state
154
+ startDate
155
+ endDate
156
+ mainPhoto {
157
+ fullpath
158
+ }
159
+ totalProgress
160
+ weeklyProgress
161
+ kpi
162
+ inspPassRate
163
+ robotProgressRate
164
+ structuralSafetyRate
165
+ buildingComplex {
166
+ address
167
+ area
168
+ clientCompany
169
+ constructionCost
170
+ householdCount
171
+ }
172
+ }
173
+ total
174
+ }
175
+ }
176
+ `,
177
+ variables: {
178
+ filters: [
179
+ { name: 'name', operator: 'search', value: `%${this.projectName}%` },
180
+ { name: 'state', operator: 'eq', value: ProjectState.ONGOING }
181
+ ],
182
+ sortings: [{ name: 'createdAt', desc: true }],
183
+ pagination: { page: this.currentPage, limit: this.pageLimit }
184
+ }
185
+ });
186
+ this.projectList = ((_a = response.data.projects) === null || _a === void 0 ? void 0 : _a.items) || [];
187
+ this.projectCount = ((_b = response.data.projects) === null || _b === void 0 ? void 0 : _b.total) || 0;
188
+ // 프로젝트별 "입력 대기 중" 건수 비동기 로드 — 결과 도착 시 그 row 만 업데이트
189
+ this._loadPendingCounts();
190
+ }
191
+ /**
192
+ * 각 프로젝트의 미입력 KPI metric 건수 산정 — sv-project-detail._loadPendingMetrics 와 동일 기준.
193
+ * - metric 정의: admin 도메인 — 헤더 없이 한 번 조회
194
+ * - 각 프로젝트의 KpiMetricValue: 그 프로젝트 tenant 도메인 — 프로젝트별로 헤더 적용해 병렬 조회
195
+ * 결과는 this.pendingCounts 에 캐시, 렌더에서 표시.
196
+ */
197
+ async _loadPendingCounts() {
198
+ var _a, _b;
199
+ const projects = this.projectList || [];
200
+ if (projects.length === 0)
201
+ return;
202
+ try {
203
+ // 1) 사용자 직접 입력 대상 metric 만 (admin 도메인 한 번 조회)
204
+ const metricsResp = await client.query({
205
+ query: gql `
206
+ query KpiMetrics {
207
+ kpiMetrics {
208
+ items {
209
+ id
210
+ periodType
211
+ collectType
212
+ active
213
+ }
214
+ }
215
+ }
216
+ `
217
+ });
218
+ const targets = (((_b = (_a = metricsResp.data) === null || _a === void 0 ? void 0 : _a.kpiMetrics) === null || _b === void 0 ? void 0 : _b.items) || []).filter((m) => m.active && m.collectType === 'MANUAL');
219
+ if (targets.length === 0) {
220
+ this.pendingCounts = {};
221
+ return;
222
+ }
223
+ const lastMonthYm = moment().tz('Asia/Seoul').subtract(1, 'month').format('YYYY-MM');
224
+ const hasValid = (v) => v.value !== null && v.value !== undefined;
225
+ // 2) 프로젝트별 KpiMetricValue 병렬 조회 — 각자의 tenant 도메인 헤더 적용
226
+ const entries = await Promise.all(projects.map(async (p) => {
227
+ var _a, _b;
228
+ if (!(p === null || p === void 0 ? void 0 : p.id))
229
+ return [p === null || p === void 0 ? void 0 : p.id, 0];
230
+ try {
231
+ const valuesResp = await client.query({
232
+ query: gql `
233
+ query KpiMetricValues($filters: [Filter!]) {
234
+ kpiMetricValues(filters: $filters) {
235
+ items {
236
+ metricId
237
+ periodType
238
+ valueDate
239
+ value
240
+ }
241
+ }
242
+ }
243
+ `,
244
+ variables: { filters: [{ name: 'org', operator: 'eq', value: p.id }] },
245
+ context: { headers: tenantHeaders(p.code) }
246
+ });
247
+ const values = ((_b = (_a = valuesResp.data) === null || _a === void 0 ? void 0 : _a.kpiMetricValues) === null || _b === void 0 ? void 0 : _b.items) || [];
248
+ const pending = targets.filter((m) => {
249
+ if (m.periodType === 'MONTH') {
250
+ return !values.some(v => v.metricId === m.id &&
251
+ v.periodType === 'MONTH' &&
252
+ (v.valueDate || '').startsWith(lastMonthYm) &&
253
+ hasValid(v));
254
+ }
255
+ return !values.some(v => v.metricId === m.id && v.periodType === m.periodType && hasValid(v));
256
+ });
257
+ return [p.id, pending.length];
258
+ }
259
+ catch (_c) {
260
+ return [p.id, 0];
261
+ }
262
+ }));
263
+ this.pendingCounts = Object.fromEntries(entries);
264
+ }
265
+ catch (e) {
266
+ console.error('[project-list] pending count 조회 실패', e);
267
+ this.pendingCounts = {};
268
+ }
269
+ }
270
+ // Input 요소의 값이 변경될 때 호출되는 콜백 함수
271
+ _onInputChange(event) {
272
+ const target = event.target;
273
+ this[target.name] = target.value;
274
+ if (target.name === 'projectName') {
275
+ this.currentPage = 1;
276
+ }
277
+ }
278
+ // 검색창에서 엔터입력시 검색
279
+ _onKeypress(event) {
280
+ if (event.code === 'Enter') {
281
+ this.currentPage = 1;
282
+ this.getProjectList();
283
+ }
284
+ }
285
+ _changePage(page) {
286
+ const nextPage = Math.min(Math.max(1, page), this.totalPages);
287
+ if (nextPage !== this.currentPage) {
288
+ this.currentPage = nextPage;
289
+ this.getProjectList();
290
+ }
291
+ }
292
+ async getKpiComprehensiveStats() {
293
+ try {
294
+ const response = await client.query({
295
+ query: gql `
296
+ query GetKpiZValueComprehensiveStats {
297
+ totalKpiZValueComprehensiveStats {
298
+ minVal
299
+ q1Val
300
+ medVal
301
+ q3Val
302
+ maxVal
303
+ }
304
+ }
305
+ `
306
+ });
307
+ this.kpiComprehensiveStats = response.data.totalKpiZValueComprehensiveStats || [];
308
+ }
309
+ catch (error) {
310
+ console.error('Failed to fetch KPI comprehensive stats:', error);
311
+ this.kpiComprehensiveStats = [];
312
+ }
313
+ }
314
+ _openCreateProjectPopup() {
315
+ openPopup(html `<popup-project-create .refreshFn=${this.getProjectList.bind(this)}></popup-project-create>`, {
316
+ backdrop: true,
317
+ size: 'small',
318
+ title: '신규 프로젝트 생성'
319
+ });
320
+ }
321
+ getBoxPlotDataForProject(project) {
322
+ var _a;
323
+ // 프로젝트의 지역(geo_group)에 해당하는 통계 찾기
324
+ const stats = (_a = this.kpiComprehensiveStats) === null || _a === void 0 ? void 0 : _a[0];
325
+ if (!stats) {
326
+ // 전체 통계의 평균값 사용 또는 기본값 반환
327
+ const defaultStats = this.kpiComprehensiveStats.length > 0 ? this.kpiComprehensiveStats[0] : null;
328
+ if (defaultStats) {
329
+ return {
330
+ min: defaultStats.minVal,
331
+ max: defaultStats.maxVal,
332
+ mean: (defaultStats.q1Val + defaultStats.q3Val) / 2,
333
+ median: defaultStats.medVal,
334
+ q1: defaultStats.q1Val,
335
+ q3: defaultStats.q3Val,
336
+ value: project.kpi || 0
337
+ };
338
+ }
339
+ // 완전한 기본값
340
+ return {
341
+ min: 0,
342
+ max: 100,
343
+ mean: 50,
344
+ median: 50,
345
+ q1: 25,
346
+ q3: 75,
347
+ value: project.kpi || 0
348
+ };
349
+ }
350
+ return {
351
+ min: stats.minVal * 100,
352
+ max: stats.maxVal * 100,
353
+ mean: ((stats.q1Val + stats.q3Val) / 2) * 100, // 대략적 평균값
354
+ median: stats.medVal * 100,
355
+ q1: stats.q1Val * 100,
356
+ q3: stats.q3Val * 100,
357
+ value: project.kpi || 0
358
+ };
359
+ }
360
+ };
361
+ SvProjectListPage.styles = [
362
+ css `
363
+ :host {
364
+ display: flex;
365
+ flex-direction: column;
366
+ overflow-y: auto;
367
+
368
+ width: 100%;
369
+ height: 100%;
370
+ background-color: var(--md-sys-color-background, #f6f6f6);
371
+
372
+ --grid-record-emphasized-background-color: red;
373
+ --grid-record-emphasized-color: yellow;
374
+ }
375
+
376
+ /* Material Symbols Outlined (stroke style) */
377
+ md-icon {
378
+ font-variation-settings:
379
+ 'FILL' 0,
380
+ 'wght' 400,
381
+ 'GRAD' 0,
382
+ 'opsz' 24;
383
+ }
384
+
385
+ /* Search bar (Figma: content > search bar) */
386
+ div[header] {
387
+ display: flex;
388
+ align-items: center;
389
+ gap: 10px;
390
+ margin: var(--spacing-large, 12px);
391
+ margin-bottom: 0;
392
+ padding: 12px 11px;
393
+ border-radius: 7px;
394
+ background-color: rgba(46, 164, 223, 0.1);
395
+ border: 1px solid rgba(46, 164, 223, 0.3);
396
+ }
397
+ div[header] div[search] {
398
+ display: flex;
399
+ align-items: center;
400
+ gap: 6px;
401
+ min-width: 186px;
402
+ }
403
+ div[header] div[search] md-icon {
404
+ color: #212529;
405
+ }
406
+ div[header] div[search] input[type='search'] {
407
+ border: none;
408
+ outline: none;
409
+ background: transparent;
410
+ font-size: 16px;
411
+ color: #212529;
412
+ padding: 2px 0;
413
+ border-bottom: 1px solid rgba(0, 0, 0, 0.2);
414
+ }
415
+ div[header] .spacer {
416
+ flex: 1;
417
+ }
418
+ div[header] div[count] {
419
+ font-size: 16px;
420
+ color: #000000;
421
+ margin-right: 6px;
422
+ }
423
+ div[header] md-icon[view] {
424
+ color: #35618e;
425
+ cursor: pointer;
426
+ }
427
+ div[header] md-icon[add] {
428
+ color: #4cbb49;
429
+ cursor: pointer;
430
+ }
431
+ div[header] button[add-project] {
432
+ display: flex;
433
+ align-items: center;
434
+ gap: 4px;
435
+ padding: 8px 12px;
436
+ background-color: #24be7b;
437
+ color: white;
438
+ border: none;
439
+ border-radius: 7px;
440
+ font-size: 14px;
441
+ font-weight: bold;
442
+ cursor: pointer;
443
+ margin-left: 8px;
444
+ }
445
+ div[header] button[add-project]:hover {
446
+ background-color: #1fa968;
447
+ }
448
+ div[header] button[add-project] md-icon {
449
+ color: white;
450
+ font-size: 18px;
451
+ }
452
+
453
+ /* Table (Figma: content > table) */
454
+ div[table] {
455
+ margin: var(--spacing-large, 12px);
456
+ border: 1px solid rgba(0, 0, 0, 0.15);
457
+ border-top: 2px solid #0c4da2;
458
+ border-radius: var(--md-sys-shape-corner-small, 5px);
459
+ overflow: hidden;
460
+ background: #ffffff;
461
+ min-height: fit-content;
462
+ }
463
+ div[table-header] {
464
+ display: grid;
465
+ grid-template-columns: 150px 80px 1fr 220px 150px 100px;
466
+ align-items: center;
467
+ justify-items: center;
468
+ height: 35px;
469
+ background: #f3f3fa;
470
+ padding: 0 25px;
471
+ column-gap: 30px;
472
+ }
473
+ div[table-header] div[col] {
474
+ font-size: 16px;
475
+ color: #212529;
476
+ line-height: 1.875em;
477
+ }
478
+ div[table-body] {
479
+ display: block;
480
+ }
481
+ div[tr] {
482
+ display: grid;
483
+ grid-template-columns: 150px 80px 1fr 220px 150px 100px;
484
+ align-items: center;
485
+ justify-items: center;
486
+ column-gap: 30px;
487
+ padding: 12px 25px;
488
+ background: #ffffff;
489
+ border-top: 1px solid rgba(0, 0, 0, 0.15);
490
+ cursor: pointer;
491
+ }
492
+ /* 좌측 정렬 요구사항 */
493
+ div[td-info],
494
+ div[td-etc] {
495
+ justify-self: start;
496
+ text-align: left;
497
+ }
498
+ div[td-pics] img[pic] {
499
+ width: 150px;
500
+ height: 90px;
501
+ object-fit: fill;
502
+ background: #e5e7eb;
503
+ border-radius: 2px;
504
+ }
505
+ div[td-status] {
506
+ font-weight: 700;
507
+ font-size: 16px;
508
+ color: #35618e;
509
+ line-height: 1;
510
+ }
511
+ div[td-info] {
512
+ display: flex;
513
+ flex-direction: column;
514
+ gap: 4px;
515
+ }
516
+ div[td-info] .name {
517
+ font-weight: 700;
518
+ font-size: 14px;
519
+ color: #000000;
520
+ }
521
+ div[td-info] .sub {
522
+ font-size: 14px;
523
+ color: #000000;
524
+ line-height: 1.7;
525
+ }
526
+ div[td-etc] {
527
+ font-size: 16px;
528
+ color: #212529;
529
+ white-space: pre-line;
530
+ }
531
+ div[td-request] {
532
+ font-weight: 700;
533
+ font-size: 16px;
534
+ color: #35618e;
535
+ white-space: pre-line;
536
+ }
537
+ div[td-kpi] {
538
+ display: flex;
539
+ align-items: center;
540
+ height: 100%;
541
+ }
542
+ div[td-kpi] .kpi-value {
543
+ font-weight: 700;
544
+ font-size: 22px;
545
+ color: #35618e;
546
+ width: 30px;
547
+ }
548
+ `
549
+ ];
550
+ __decorate([
551
+ state(),
552
+ __metadata("design:type", String)
553
+ ], SvProjectListPage.prototype, "projectName", void 0);
554
+ __decorate([
555
+ state(),
556
+ __metadata("design:type", Array)
557
+ ], SvProjectListPage.prototype, "projectList", void 0);
558
+ __decorate([
559
+ state(),
560
+ __metadata("design:type", Number)
561
+ ], SvProjectListPage.prototype, "projectCount", void 0);
562
+ __decorate([
563
+ state(),
564
+ __metadata("design:type", Number)
565
+ ], SvProjectListPage.prototype, "currentPage", void 0);
566
+ __decorate([
567
+ state(),
568
+ __metadata("design:type", Object)
569
+ ], SvProjectListPage.prototype, "pendingCounts", void 0);
570
+ __decorate([
571
+ state(),
572
+ __metadata("design:type", Array)
573
+ ], SvProjectListPage.prototype, "kpiComprehensiveStats", void 0);
574
+ SvProjectListPage = __decorate([
575
+ customElement('sv-project-list')
576
+ ], SvProjectListPage);
577
+ export { SvProjectListPage };
578
+ //# sourceMappingURL=sv-project-list.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sv-project-list.js","sourceRoot":"","sources":["../../client/pages/sv-project-list.ts"],"names":[],"mappings":"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","sourcesContent":["import '@material/web/icon/icon.js'\nimport '../components/kpi-single-boxplot-chart'\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'\nimport { client } from '@operato/graphql'\nimport { openPopup } from '@operato/layout'\nimport gql from 'graphql-tag'\nimport { Attachment } from '@things-factory/attachment-base'\nimport type { FileUpload } from 'graphql-upload/GraphQLUpload.js'\nimport '../components/sv-pagenation-control'\nimport '@dssp/project/dist-client/pages/project/popup/popup-project-create'\nimport { tenantHeaders } from '@dssp/project/dist-client/shared/domain-context'\nimport moment from 'moment-timezone'\n\nexport enum ProjectState {\n 'ONGOING' = '10',\n 'COMPLETED' = '20'\n}\nexport const PROJECT_STATE = {\n [ProjectState.ONGOING]: '진행중',\n [ProjectState.COMPLETED]: '완료'\n}\nexport enum BuildingInspectionStatus {\n WAIT = 'WAIT',\n OVERALL_WAIT = 'OVERALL_WAIT',\n REQUEST = 'REQUEST',\n OVERALL_REQUEST = 'OVERALL_REQUEST',\n PASS = 'PASS',\n FAIL = 'FAIL'\n}\nexport const BUILDING_INSPECTION_STATUS = {\n [BuildingInspectionStatus.WAIT]: '작업 완료',\n [BuildingInspectionStatus.OVERALL_WAIT]: '작업 완료',\n [BuildingInspectionStatus.REQUEST]: '감리 대기',\n [BuildingInspectionStatus.OVERALL_REQUEST]: '감리 대기',\n [BuildingInspectionStatus.PASS]: '검측 완료',\n [BuildingInspectionStatus.FAIL]: '조치 필요'\n}\n\nexport enum ProjectType {\n DSSP = 'DSSP',\n DCSP = 'DCSP',\n DKPI = 'DKPI'\n}\nexport interface KpiValuesObject {\n kpiName: string\n value: number\n}\nexport interface Project {\n id?: string\n name: string\n state: ProjectState\n startDate?: string\n endDate?: string\n mainPhoto?: Attachment\n mainPhotoUpload?: FileUpload\n totalProgress?: number\n weeklyProgress?: number\n kpi?: number\n kpiValues?: KpiValuesObject[]\n inspPassRate?: number\n robotProgressRate?: number\n structuralSafetyRate?: number\n robotCount?: number\n scheduleTable?: Attachment\n buildingComplex: BuildingComplex\n projectType?: ProjectType\n}\nexport interface BuildingComplex {\n id?: string\n address?: string\n latitude?: number\n longitude?: number\n area?: number\n constructionCompany?: string\n clientCompany?: string\n designCompany?: string\n supervisoryCompany?: string\n drawing?: Attachment\n drawingUpload?: FileUpload\n constructionType?: string\n constructionCost?: number\n etc?: string\n householdCount?: number\n buildingCount?: number\n notice?: string\n planXScale?: number\n planYScale?: number\n overallConstructorEmails?: string[]\n taskConstructorEmails?: string[]\n overallSupervisoryEmails?: string[]\n taskSupervisoryEmails?: string[]\n buildings?: Building[]\n}\nexport interface Building {\n id?: string\n name: string | undefined\n floorCount: number | undefined\n hasBasement?: boolean\n basementFloorCount?: number\n drawing?: Attachment\n drawingUpload?: FileUpload\n buildingLevels?: BuildingLevel[]\n}\n\nexport interface BuildingLevel {\n id?: string\n floor?: number\n floorDisplayName?: string\n mainDrawing?: Attachment\n mainDrawingImage?: string\n mainDrawingThumbnail?: string\n mainDrawingUpload?: FileUpload\n elevationDrawing?: Attachment\n elevationDrawingThumbnail?: string\n elevationDrawingUpload?: FileUpload\n rebarDistributionDrawing?: Attachment\n rebarDistributionDrawingThumbnail?: string\n rebarDistributionDrawingUpload?: FileUpload\n etcDrawings?: Attachment[]\n etcDrawingsUpload?: FileUpload[]\n building?: Building\n buildingInspections?: BuildingInspection[]\n}\n\nexport interface BuildingInspection {\n id?: string\n attatchments?: Attachment[]\n // buildingInspectionAttachments?: BuildingInspectionAttachment[]\n status?: BuildingInspectionStatus\n requestDate?: Date\n buildingLevel?: BuildingLevel\n checklist?: Checklist\n createdAt?: Date\n updatedAt?: Date\n deletedAt?: Date\n}\n\nexport interface Checklist {\n id: string\n name?: string\n documentNo?: string\n constructionType?: string\n constructionDetailType?: string\n location?: string\n constructionInspectorDate?: Date\n supervisorInspectorDate?: Date\n overallConstructorSignature?: string\n taskConstructorSignature?: string\n overallSupervisorySignature?: string\n taskSupervisorySignature?: string\n inspectionParts?: string[]\n // checklistItems?: ChecklistItem[]\n}\n\n@customElement('sv-project-list')\nexport class SvProjectListPage 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 /* Material Symbols Outlined (stroke style) */\n md-icon {\n font-variation-settings:\n 'FILL' 0,\n 'wght' 400,\n 'GRAD' 0,\n 'opsz' 24;\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 div[header] button[add-project] {\n display: flex;\n align-items: center;\n gap: 4px;\n padding: 8px 12px;\n background-color: #24be7b;\n color: white;\n border: none;\n border-radius: 7px;\n font-size: 14px;\n font-weight: bold;\n cursor: pointer;\n margin-left: 8px;\n }\n div[header] button[add-project]:hover {\n background-color: #1fa968;\n }\n div[header] button[add-project] md-icon {\n color: white;\n font-size: 18px;\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 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: #35618e;\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 /** projectId → 미입력 KPI 건수 */\n @state() private pendingCounts: Record<string, number> = {}\n @state() private kpiComprehensiveStats: any[] = []\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\n <button add-project @click=${this._openCreateProjectPopup}>\n <md-icon>add</md-icon>\n 신규 프로젝트 생성\n </button>\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/project-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입력 요청 ${this.pendingCounts[project.id!] ?? '-'}건</div>\n <div td-kpi>\n <kpi-single-boxplot-chart .data=${this.getBoxPlotDataForProject(project)}></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 this.getKpiComprehensiveStats()\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 code\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.ONGOING }\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 // 프로젝트별 \"입력 대기 중\" 건수 비동기 로드 — 결과 도착 시 그 row 만 업데이트\n this._loadPendingCounts()\n }\n\n /**\n * 각 프로젝트의 미입력 KPI metric 건수 산정 — sv-project-detail._loadPendingMetrics 와 동일 기준.\n * - metric 정의: admin 도메인 — 헤더 없이 한 번 조회\n * - 각 프로젝트의 KpiMetricValue: 그 프로젝트 tenant 도메인 — 프로젝트별로 헤더 적용해 병렬 조회\n * 결과는 this.pendingCounts 에 캐시, 렌더에서 표시.\n */\n private async _loadPendingCounts() {\n const projects = this.projectList || []\n if (projects.length === 0) return\n\n try {\n // 1) 사용자 직접 입력 대상 metric 만 (admin 도메인 한 번 조회)\n const metricsResp = await client.query({\n query: gql`\n query KpiMetrics {\n kpiMetrics {\n items {\n id\n periodType\n collectType\n active\n }\n }\n }\n `\n })\n const targets = (metricsResp.data?.kpiMetrics?.items || []).filter(\n (m: any) => m.active && m.collectType === 'MANUAL'\n )\n if (targets.length === 0) {\n this.pendingCounts = {}\n return\n }\n\n const lastMonthYm = moment().tz('Asia/Seoul').subtract(1, 'month').format('YYYY-MM')\n const hasValid = (v: any) => v.value !== null && v.value !== undefined\n\n // 2) 프로젝트별 KpiMetricValue 병렬 조회 — 각자의 tenant 도메인 헤더 적용\n const entries = await Promise.all(\n projects.map(async (p: any) => {\n if (!p?.id) return [p?.id, 0] as [string, number]\n try {\n const valuesResp = await client.query({\n query: gql`\n query KpiMetricValues($filters: [Filter!]) {\n kpiMetricValues(filters: $filters) {\n items {\n metricId\n periodType\n valueDate\n value\n }\n }\n }\n `,\n variables: { filters: [{ name: 'org', operator: 'eq', value: p.id }] },\n context: { headers: tenantHeaders(p.code) }\n })\n const values: any[] = valuesResp.data?.kpiMetricValues?.items || []\n const pending = targets.filter((m: any) => {\n if (m.periodType === 'MONTH') {\n return !values.some(\n v =>\n v.metricId === m.id &&\n v.periodType === 'MONTH' &&\n (v.valueDate || '').startsWith(lastMonthYm) &&\n hasValid(v)\n )\n }\n return !values.some(v => v.metricId === m.id && v.periodType === m.periodType && hasValid(v))\n })\n return [p.id, pending.length] as [string, number]\n } catch {\n return [p.id, 0] as [string, number]\n }\n })\n )\n this.pendingCounts = Object.fromEntries(entries)\n } catch (e) {\n console.error('[project-list] pending count 조회 실패', e)\n this.pendingCounts = {}\n }\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 async getKpiComprehensiveStats() {\n try {\n const response = await client.query({\n query: gql`\n query GetKpiZValueComprehensiveStats {\n totalKpiZValueComprehensiveStats {\n minVal\n q1Val\n medVal\n q3Val\n maxVal\n }\n }\n `\n })\n\n this.kpiComprehensiveStats = response.data.totalKpiZValueComprehensiveStats || []\n } catch (error) {\n console.error('Failed to fetch KPI comprehensive stats:', error)\n this.kpiComprehensiveStats = []\n }\n }\n\n private _openCreateProjectPopup() {\n openPopup(html`<popup-project-create .refreshFn=${this.getProjectList.bind(this)}></popup-project-create>`, {\n backdrop: true,\n size: 'small',\n title: '신규 프로젝트 생성'\n })\n }\n\n private getBoxPlotDataForProject(project: Project) {\n // 프로젝트의 지역(geo_group)에 해당하는 통계 찾기\n const stats = this.kpiComprehensiveStats?.[0]\n\n if (!stats) {\n // 전체 통계의 평균값 사용 또는 기본값 반환\n const defaultStats = this.kpiComprehensiveStats.length > 0 ? this.kpiComprehensiveStats[0] : null\n if (defaultStats) {\n return {\n min: defaultStats.minVal,\n max: defaultStats.maxVal,\n mean: (defaultStats.q1Val + defaultStats.q3Val) / 2,\n median: defaultStats.medVal,\n q1: defaultStats.q1Val,\n q3: defaultStats.q3Val,\n value: project.kpi || 0\n }\n }\n\n // 완전한 기본값\n return {\n min: 0,\n max: 100,\n mean: 50,\n median: 50,\n q1: 25,\n q3: 75,\n value: project.kpi || 0\n }\n }\n\n return {\n min: stats.minVal * 100,\n max: stats.maxVal * 100,\n mean: ((stats.q1Val + stats.q3Val) / 2) * 100, // 대략적 평균값\n median: stats.medVal * 100,\n q1: stats.q1Val * 100,\n q3: stats.q3Val * 100,\n value: project.kpi || 0\n }\n }\n}\n"]}