dashboard.html 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. <div class="dashboard-container">
  2. <header class="dashboard-header">
  3. <h1>设计师工作台</h1>
  4. <!-- 顶部导航 -->
  5. <nav class="dashboard-nav">
  6. <button class="nav-btn" [class.active]="activeDashboard === 'main'" (click)="switchDashboard('main')">工作台</button>
  7. <button class="nav-btn" [class.active]="activeDashboard === 'skills'" (click)="switchDashboard('skills')">能力雷达</button>
  8. <button class="nav-btn" [class.active]="activeDashboard === 'personal'" (click)="switchDashboard('personal')">个人看板</button>
  9. </nav>
  10. </header>
  11. <!-- 主要内容区域 - 工作台 -->
  12. <div *ngIf="activeDashboard === 'main'" class="dashboard-main">
  13. <!-- 视图切换按钮 -->
  14. <div class="view-toggle">
  15. <button class="toggle-btn" (click)="toggleView()">
  16. {{ viewMode === 'card' ? '切换为列表' : '切换为卡片' }}
  17. </button>
  18. </div>
  19. <!-- 卡片视图 -->
  20. @if (viewMode === 'card') {
  21. <!-- 核心信息卡片区域 - 每行列3张卡片 -->
  22. <section class="core-cards-section">
  23. <div class="cards-grid">
  24. <!-- 紧急任务卡片 -->
  25. <div *ngFor="let task of urgentTasks" class="core-card urgent-card">
  26. <div class="card-header">
  27. <span class="card-badge urgent">紧急</span>
  28. <h3>{{ task.title }}</h3>
  29. </div>
  30. <div class="card-content">
  31. <p class="project-name">项目: {{ task.projectName }}</p>
  32. <p class="countdown">剩余: {{ getTaskCountdown(task.id) }}</p>
  33. </div>
  34. <div class="card-actions">
  35. <button [routerLink]="['/designer/project-detail', task.projectId]" class="btn-primary">
  36. 立即处理
  37. </button>
  38. </div>
  39. </div>
  40. <!-- 待办任务卡片 -->
  41. <div *ngFor="let task of getTopTasks(6 - urgentTasks.length)" class="core-card task-card">
  42. <div class="card-header">
  43. <span class="card-badge" [class.overdue]="task.isOverdue">
  44. {{ task.stage }}
  45. <span *ngIf="task.isOverdue">/超期</span>
  46. </span>
  47. <h3>{{ task.title }}</h3>
  48. </div>
  49. <div class="card-content">
  50. <p class="project-name">项目: {{ task.projectName }}</p>
  51. <p class="deadline" [class.overdue]="task.isOverdue">
  52. 截止: {{ task.deadline | date:'yyyy-MM-dd HH:mm' }}
  53. </p>
  54. <!-- 进度条 -->
  55. <div class="task-progress" *ngIf="task.stage !== '投诉处理' && !task.isCompleted">
  56. <div class="progress-bar">
  57. <div class="progress-fill" [style.width]="getTaskStageProgress(task.id) + '%'">
  58. </div>
  59. </div>
  60. <p class="progress-text">{{ getTaskStageProgress(task.id) }}%</p>
  61. </div>
  62. </div>
  63. <div class="card-actions">
  64. <button *ngIf="!task.isCompleted" (click)="markTaskAsCompleted(task.id)" class="btn-secondary">
  65. 标记完成
  66. </button>
  67. <button [routerLink]="['/designer/project-detail', task.projectId]" class="btn-primary">
  68. 查看详情
  69. </button>
  70. </div>
  71. </div>
  72. <!-- 项目饱和度卡片 -->
  73. <div class="core-card workload-card" *ngIf="(urgentTasks.length + tasks.length) < 6">
  74. <div class="card-header">
  75. <span class="card-badge workload">饱和度</span>
  76. <h3>当前工作量</h3>
  77. </div>
  78. <div class="card-content">
  79. <div class="workload-indicator">
  80. <div class="workload-circle" [style.background]="getWorkloadColor()">
  81. <span class="workload-percentage">{{ workloadPercentage }}%</span>
  82. </div>
  83. </div>
  84. <p class="workload-status">{{ getWorkloadStatus() }}</p>
  85. </div>
  86. <div class="card-actions">
  87. <button class="btn-secondary" (click)="switchDashboard('personal')">
  88. 查看详情
  89. </button>
  90. </div>
  91. </div>
  92. </div>
  93. </section>
  94. <!-- 附加信息区域 -->
  95. <section class="additional-info-section">
  96. <!-- 待处理反馈区域 -->
  97. <div class="info-column" *ngIf="pendingFeedbacks.length > 0">
  98. <div class="section-header">
  99. <h2>待处理反馈</h2>
  100. </div>
  101. <div class="feedback-list">
  102. <div *ngFor="let item of pendingFeedbacks" class="feedback-item">
  103. <div class="feedback-content">
  104. <p class="feedback-title">⚠️ {{ item.task.title }} - 客户反馈</p>
  105. <p class="feedback-project">项目: {{ item.task.projectName }}</p>
  106. <p class="feedback-summary">反馈: {{ !item.feedback.isSatisfied ? '不满意' : '满意' }}</p>
  107. </div>
  108. <div class="feedback-actions">
  109. <button (click)="handleFeedback(item.task.id)" class="btn-handle-feedback">
  110. 处理反馈
  111. </button>
  112. </div>
  113. </div>
  114. </div>
  115. </div>
  116. <!-- 代班信息区域 -->
  117. <div class="info-column" *ngIf="shiftTasks.length > 0">
  118. <div class="section-header">
  119. <h2>👥 代班信息</h2>
  120. <button class="add-shift-btn" (click)="openShiftModal()">
  121. 添加代班任务
  122. </button>
  123. </div>
  124. <div class="shift-list">
  125. <div class="shift-item" *ngFor="let shift of shiftTasks">
  126. <div class="shift-header">
  127. <div class="shift-project">{{ shift.projectName }}</div>
  128. <div class="shift-priority" [class.priority-high]="shift.priority === '高'" [class.priority-medium]="shift.priority === '中'" [class.priority-low]="shift.priority === '低'">
  129. {{ shift.priority }}级
  130. </div>
  131. </div>
  132. <div class="shift-details">
  133. <div class="shift-task">{{ shift.taskDescription }}</div>
  134. <div class="shift-time">代班时间: {{ shift.shiftDate }}</div>
  135. </div>
  136. </div>
  137. </div>
  138. </div>
  139. <!-- 时间预警区域 -->
  140. <div class="info-column" *ngIf="overdueTasks.length > 0">
  141. <div class="section-header">
  142. <h2>⏰ 时间预警</h2>
  143. </div>
  144. <div class="warning-list">
  145. <div *ngFor="let task of overdueTasks" class="warning-item">
  146. <div class="warning-content">
  147. <p class="warning-title">{{ task.title }} - 已超期</p>
  148. <p class="warning-detail">项目: {{ task.projectName }}</p>
  149. </div>
  150. </div>
  151. </div>
  152. </div>
  153. </section>
  154. }
  155. <!-- 列表视图 -->
  156. @if (viewMode === 'list') {
  157. <section class="list-section">
  158. <div class="list-header">
  159. <div class="col urgency-col">紧急度</div>
  160. <div class="col project-col">项目</div>
  161. <div class="col title-col">任务</div>
  162. <div class="col stage-col">阶段</div>
  163. <div class="col deadline-col">截止时间</div>
  164. <div class="col left-col">剩余</div>
  165. <div class="col actions-col">操作</div>
  166. </div>
  167. <div class="list-body">
  168. @for (task of getTasksSortedByUrgency(); track task.id) {
  169. <div class="list-row">
  170. <div class="col urgency-col">
  171. <span class="urgency-dot" [ngClass]="getUrgencyClass(task)"></span>
  172. <span class="urgency-text">{{ getUrgencyLevel(task) }}</span>
  173. </div>
  174. <div class="col project-col">{{ task.projectName }}</div>
  175. <div class="col title-col">{{ task.title }}</div>
  176. <div class="col stage-col">{{ task.stage }}</div>
  177. <div class="col deadline-col">{{ task.deadline | date:'yyyy-MM-dd HH:mm' }}</div>
  178. <div class="col left-col">{{ getListTimeLeft(task) }}</div>
  179. <div class="col actions-col">
  180. <button class="btn-link" [routerLink]="['/designer/project-detail', task.projectId]">详情</button>
  181. <button class="btn-link" *ngIf="!task.isCompleted" (click)="markTaskAsCompleted(task.id)">完成</button>
  182. </div>
  183. </div>
  184. }
  185. </div>
  186. </section>
  187. }
  188. </div>
  189. <!-- 能力雷达单独视图 -->
  190. <div *ngIf="activeDashboard === 'skills'" class="skills-view">
  191. <app-skill-radar></app-skill-radar>
  192. </div>
  193. <!-- 个人看板单独视图 -->
  194. <div *ngIf="activeDashboard === 'personal'" class="personal-view">
  195. <app-personal-board></app-personal-board>
  196. </div>
  197. <!-- 提醒话术弹窗 -->
  198. <div *ngIf="reminderMessage" class="reminder-modal">
  199. <div class="modal-content">
  200. <h3>提醒话术</h3>
  201. <p>{{ reminderMessage }}</p>
  202. <button (click)="clearReminder()" class="btn-close">关闭</button>
  203. </div>
  204. </div>
  205. </div>