project-files-modal.component.html 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. <!-- 模态框背景 -->
  2. <div class="modal-overlay"
  3. *ngIf="isVisible"
  4. (click)="onBackdropClick($event)">
  5. <!-- 模态框内容 -->
  6. <div class="modal-container" (click)="$event.stopPropagation()">
  7. <!-- 模态框头部 -->
  8. <div class="modal-header">
  9. <div class="header-left">
  10. <h2 class="modal-title">
  11. <svg class="title-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  12. <path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"></path>
  13. <polyline points="14,2 14,8 20,8"></polyline>
  14. <line x1="16" y1="13" x2="8" y2="13"></line>
  15. <line x1="16" y1="17" x2="8" y2="17"></line>
  16. </svg>
  17. 项目文件
  18. </h2>
  19. <div class="file-stats">
  20. <span class="stat-item">
  21. <span class="stat-number">{{ totalFiles }}</span>
  22. <span class="stat-label">文件</span>
  23. </span>
  24. <span class="stat-item">
  25. <span class="stat-number">{{ formatFileSize(totalSize) }}</span>
  26. <span class="stat-label">总大小</span>
  27. </span>
  28. @if (imageCount > 0) {
  29. <span class="stat-item">
  30. <span class="stat-number">{{ imageCount }}</span>
  31. <span class="stat-label">图片</span>
  32. </span>
  33. }
  34. @if (documentCount > 0) {
  35. <span class="stat-item">
  36. <span class="stat-number">{{ documentCount }}</span>
  37. <span class="stat-label">文档</span>
  38. </span>
  39. }
  40. </div>
  41. </div>
  42. <div class="header-right">
  43. <!-- 视图切换 -->
  44. <div class="view-toggle">
  45. <button
  46. class="toggle-btn"
  47. [class.active]="previewMode === 'grid'"
  48. (click)="previewMode = 'grid'"
  49. title="网格视图">
  50. <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  51. <rect x="3" y="3" width="7" height="7"></rect>
  52. <rect x="14" y="3" width="7" height="7"></rect>
  53. <rect x="14" y="14" width="7" height="7"></rect>
  54. <rect x="3" y="14" width="7" height="7"></rect>
  55. </svg>
  56. </button>
  57. <button
  58. class="toggle-btn"
  59. [class.active]="previewMode === 'list'"
  60. (click)="previewMode = 'list'"
  61. title="列表视图">
  62. <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  63. <line x1="8" y1="6" x2="21" y2="6"></line>
  64. <line x1="8" y1="12" x2="21" y2="12"></line>
  65. <line x1="8" y1="18" x2="21" y2="18"></line>
  66. <line x1="3" y1="6" x2="3.01" y2="6"></line>
  67. <line x1="3" y1="12" x2="3.01" y2="12"></line>
  68. <line x1="3" y1="18" x2="3.01" y2="18"></line>
  69. </svg>
  70. </button>
  71. </div>
  72. <!-- 搜索框 -->
  73. <div class="search-box">
  74. <svg class="search-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  75. <circle cx="11" cy="11" r="8"></circle>
  76. <path d="m21 21-4.35-4.35"></path>
  77. </svg>
  78. <input
  79. type="text"
  80. class="search-input"
  81. placeholder="搜索文件..."
  82. [(ngModel)]="searchQuery">
  83. </div>
  84. <!-- 过滤器 -->
  85. <select class="filter-select" [(ngModel)]="filterType">
  86. <option value="all">全部文件</option>
  87. <option value="images">图片</option>
  88. <option value="documents">文档</option>
  89. <option value="videos">视频</option>
  90. </select>
  91. <!-- 关闭按钮 -->
  92. <button class="close-btn" (click)="onClose()">
  93. <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  94. <line x1="18" y1="6" x2="6" y2="18"></line>
  95. <line x1="6" y1="6" x2="18" y2="18"></line>
  96. </svg>
  97. </button>
  98. </div>
  99. </div>
  100. <!-- 模态框内容 -->
  101. <div class="modal-content">
  102. <!-- 加载状态 -->
  103. @if (loading) {
  104. <div class="loading-state">
  105. <div class="loading-spinner"></div>
  106. <p>加载文件中...</p>
  107. </div>
  108. } @else if (getFilteredFiles().length === 0) {
  109. <!-- 空状态 -->
  110. <div class="empty-state">
  111. <svg class="empty-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  112. <path d="M13 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V9z"></path>
  113. <polyline points="13,2 13,9 20,9"></polyline>
  114. </svg>
  115. <h3>暂无文件</h3>
  116. <p>该项目还没有上传任何文件</p>
  117. </div>
  118. } @else {
  119. <!-- 文件列表 -->
  120. @if (previewMode === 'grid') {
  121. <div class="files-grid">
  122. @for (file of getFilteredFiles(); track file.id) {
  123. <div class="file-card" >
  124. <!-- 文件预览 -->
  125. <div class="file-preview" (click)="selectFile(file)">
  126. @if (isImageFile(file)) {
  127. <img [src]="file.url" [alt]="file.name" class="preview-image" />
  128. } @else {
  129. <div class="preview-placeholder">
  130. <div class="file-icon">{{ getFileIcon(file.type) }}</div>
  131. <div class="file-extension">{{ getFileExtension(file.name) }}</div>
  132. </div>
  133. }
  134. </div>
  135. <!-- 文件信息 -->
  136. <div class="file-info">
  137. <h4 class="file-name" [title]="file.originalName">{{ file.originalName }}</h4>
  138. <div class="file-meta">
  139. <span class="file-size">{{ formatFileSize(file.size) }}</span>
  140. <span class="file-date">{{ file.uploadedAt | date:'MM-dd HH:mm' }}</span>
  141. </div>
  142. <div class="file-description" *ngIf="file.description">
  143. <p>{{ file.description }}</p>
  144. </div>
  145. <div class="file-footer">
  146. <div class="uploader-info">
  147. @if (file.uploadedBy?.avatar) {
  148. <img [src]="file.uploadedBy?.avatar" [alt]="file.uploadedBy?.name" class="uploader-avatar" />
  149. }
  150. <span class="uploader-name">{{ file.uploadedBy?.name }}</span>
  151. <span class="source-badge" [class]="getSourceBadgeClass(file.source)">
  152. {{ getSourceLabel(file.source) }}
  153. </span>
  154. </div>
  155. <div class="file-actions">
  156. @if(file && isImageFile(file)){
  157. <button class="analyze-btn" (click)="openColorAnalysis(file)">色彩分析</button>
  158. }
  159. <button
  160. class="action-btn download-btn"
  161. (click)="downloadFile(file); $event.stopPropagation()"
  162. title="下载">
  163. <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  164. <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path>
  165. <polyline points="7,10 12,15 17,10"></polyline>
  166. <line x1="12" y1="15" x2="12" y2="3"></line>
  167. </svg>
  168. </button>
  169. </div>
  170. </div>
  171. </div>
  172. </div>
  173. }
  174. </div>
  175. } @else {
  176. <!-- 列表视图 -->
  177. <div class="files-list">
  178. @for (file of getFilteredFiles(); track file.id) {
  179. <div class="file-list-item" >
  180. <div class="list-file-icon" (click)="selectFile(file)">
  181. @if (isImageFile(file)) {
  182. <img [src]="file.url" [alt]="file.name" class="list-preview-image" />
  183. } @else {
  184. <span class="list-icon">{{ getFileIcon(file.type) }}</span>
  185. }
  186. </div>
  187. <div class="list-file-info">
  188. <div class="list-file-name">{{ file.originalName }}</div>
  189. <div class="list-file-meta">
  190. <span>{{ formatFileSize(file.size) }}</span>
  191. <span>{{ file.uploadedAt | date:'MM-dd HH:mm' }}</span>
  192. <span>{{ file.uploadedBy?.name }}</span>
  193. <span class="source-badge" [class]="getSourceBadgeClass(file.source)">
  194. {{ getSourceLabel(file.source) }}
  195. </span>
  196. </div>
  197. @if (file.description) {
  198. <div class="list-file-description">{{ file.description }}</div>
  199. }
  200. </div>
  201. <div class="list-file-actions">
  202. @if(isImageFile(file)){
  203. <button class="analyze-btn" (click)="openColorAnalysis(file)">色彩分析</button>
  204. }
  205. <button
  206. class="action-btn download-btn"
  207. (click)="downloadFile(file); $event.stopPropagation()"
  208. title="下载">
  209. <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  210. <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path>
  211. <polyline points="7,10 12,15 17,10"></polyline>
  212. <line x1="12" y1="15" x2="12" y2="3"></line>
  213. </svg>
  214. </button>
  215. </div>
  216. </div>
  217. }
  218. </div>
  219. }
  220. }
  221. </div>
  222. </div>
  223. <!-- 文件预览模态框 -->
  224. @if (selectedFile) {
  225. <div class="preview-overlay" (click)="closePreview()">
  226. <div class="preview-container" (click)="$event.stopPropagation()">
  227. <div class="preview-header">
  228. <h3>{{ selectedFile.originalName }}</h3>
  229. <button class="preview-close" (click)="closePreview()">
  230. <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  231. <line x1="18" y1="6" x2="6" y2="18"></line>
  232. <line x1="6" y1="6" x2="18" y2="18"></line>
  233. </svg>
  234. </button>
  235. </div>
  236. <div class="preview-content">
  237. @if (isImageFile(selectedFile)) {
  238. <img [src]="selectedFile.url" [alt]="selectedFile.originalName" class="preview-full-image" />
  239. } @else if (isVideoFile(selectedFile)) {
  240. <video [src]="selectedFile.url" controls class="preview-video"></video>
  241. } @else {
  242. <div class="preview-no-preview">
  243. <div class="preview-icon-large">{{ getFileIcon(selectedFile.type) }}</div>
  244. <p>该文件类型不支持预览</p>
  245. <button
  246. class="preview-download-btn"
  247. (click)="downloadFile(selectedFile)">
  248. 下载文件
  249. </button>
  250. </div>
  251. }
  252. </div>
  253. <div class="preview-footer">
  254. <div class="preview-meta">
  255. <span>{{ formatFileSize(selectedFile.size) }}</span>
  256. <span>{{ selectedFile.uploadedAt | date:'yyyy-MM-dd HH:mm:ss' }}</span>
  257. <span>上传者: {{ selectedFile.uploadedBy?.name }}</span>
  258. <span class="source-badge" [class]="getSourceBadgeClass(selectedFile.source)">
  259. {{ getSourceLabel(selectedFile.source) }}
  260. </span>
  261. </div>
  262. <div class="preview-description">
  263. <textarea
  264. class="description-textarea"
  265. placeholder="添加文件说明..."
  266. [(ngModel)]="selectedFile.description"
  267. (blur)="updateFileDescription(selectedFile, selectedFile.description || '')">
  268. </textarea>
  269. </div>
  270. </div>
  271. </div>
  272. </div>
  273. }
  274. </div>