requirements-confirm-card.html 53 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059
  1. <div class="info-card requirements-confirm-card">
  2. <div class="card-header">
  3. <h4>确认需求</h4>
  4. <div class="header-actions">
  5. <button class="btn-ghost btn-sm" (click)="refreshProgress()">刷新进度</button>
  6. <!-- 紧凑型流程进度卡片 -->
  7. <div class="compact-stage-indicators">
  8. <div class="stage-chain">
  9. <div class="stage-dot" [class]="getStageStatusClass('materialAnalysis')"
  10. title="素材解析 - {{ getStageStatusText('materialAnalysis') }}">
  11. <span class="stage-number">1</span>
  12. </div>
  13. <div class="stage-connector" [class]="stageCompletionStatus.materialAnalysis ? 'completed' : 'pending'"></div>
  14. <div class="stage-dot" [class]="getStageStatusClass('collaboration')"
  15. title="协作验证 - {{ getStageStatusText('collaboration') }}">
  16. <span class="stage-number">2</span>
  17. </div>
  18. <div class="stage-connector" [class]="stageCompletionStatus.collaboration ? 'completed' : 'pending'"></div>
  19. <div class="stage-dot" [class]="getStageStatusClass('progressReview')"
  20. title="进度管理 - {{ getStageStatusText('progressReview') }}">
  21. <span class="stage-number">3</span>
  22. </div>
  23. </div>
  24. </div>
  25. <div class="progress-indicator">
  26. <div class="progress-bar">
  27. <div class="progress-fill" [style.width.%]="getProgressPercentage()"></div>
  28. </div>
  29. <span class="progress-text">{{ getProgressPercentage() | number:'1.0-0' }}% 完成</span>
  30. </div>
  31. </div>
  32. </div>
  33. <!-- 标签页导航 -->
  34. <div class="tab-navigation">
  35. <button
  36. class="tab-button"
  37. [class.active]="activeTab === 'materials'"
  38. (click)="switchTab('materials')">
  39. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  40. <path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"></path>
  41. <polyline points="14,2 14,8 20,8"></polyline>
  42. </svg>
  43. 素材解析
  44. </button>
  45. <button
  46. class="tab-button"
  47. [class.active]="activeTab === 'collaboration'"
  48. (click)="switchTab('collaboration')">
  49. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  50. <path d="M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2"></path>
  51. <circle cx="9" cy="7" r="4"></circle>
  52. <path d="M23 21v-2a4 4 0 0 0-3-3.87m-4-12a4 4 0 0 1 0 7.75"></path>
  53. </svg>
  54. 协作验证
  55. </button>
  56. <button
  57. class="tab-button"
  58. [class.active]="activeTab === 'progress'"
  59. (click)="switchTab('progress')">
  60. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  61. <polyline points="22,12 18,12 15,21 9,3 6,12 2,12"></polyline>
  62. </svg>
  63. 进度管理
  64. </button>
  65. </div>
  66. <!-- 标签页内容 -->
  67. <div class="tab-content">
  68. <!-- 素材解析标签页 -->
  69. @if (activeTab === 'materials') {
  70. <div class="materials-section">
  71. <!-- 全展示模式控制和导出按钮 -->
  72. <div class="full-display-control">
  73. <button
  74. class="toggle-full-display-btn"
  75. [class.active]="isFullDisplayMode"
  76. (click)="toggleFullDisplayMode()">
  77. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  78. @if (isFullDisplayMode) {
  79. <polyline points="4 14 10 14 10 20"></polyline>
  80. <polyline points="20 10 14 10 14 4"></polyline>
  81. <line x1="14" y1="10" x2="21" y2="3"></line>
  82. <line x1="3" y1="21" x2="10" y2="14"></line>
  83. } @else {
  84. <polyline points="15 3 21 3 21 9"></polyline>
  85. <polyline points="9 21 3 21 3 15"></polyline>
  86. <line x1="21" y1="3" x2="14" y2="10"></line>
  87. <line x1="3" y1="21" x2="10" y2="14"></line>
  88. }
  89. </svg>
  90. {{ isFullDisplayMode ? '退出全展示' : '全展示模式' }}
  91. </button>
  92. <span class="mode-description">
  93. {{ isFullDisplayMode ? '已展开所有分析详情' : '快速查看所有素材的完整分析' }}
  94. </span>
  95. <!-- 导出按钮 -->
  96. <div class="export-buttons">
  97. <button class="export-btn" (click)="exportAnalysisReportHTML()" title="导出HTML报告">
  98. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  99. <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path>
  100. <polyline points="7 10 12 15 17 10"></polyline>
  101. <line x1="12" y1="15" x2="12" y2="3"></line>
  102. </svg>
  103. HTML
  104. </button>
  105. <button class="export-btn" (click)="exportAnalysisReportJSON()" title="导出JSON数据">
  106. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  107. <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path>
  108. <polyline points="7 10 12 15 17 10"></polyline>
  109. <line x1="12" y1="15" x2="12" y2="3"></line>
  110. </svg>
  111. JSON
  112. </button>
  113. </div>
  114. </div>
  115. <!-- 文本输入区域 - 独立一行 -->
  116. <div class="text-upload-section">
  117. <div class="upload-item text-item">
  118. <h5>文本描述</h5>
  119. <form [formGroup]="materialUploadForm" (ngSubmit)="onTextSubmit()">
  120. <textarea
  121. formControlName="textContent"
  122. placeholder="请描述您的装修需求,如:希望温馨的暖木色调,适合亲子家庭,需要瑜伽区收纳..."
  123. rows="4">
  124. </textarea>
  125. <button type="submit" class="btn-primary btn-sm" [disabled]="!materialUploadForm.get('textContent')?.value">
  126. 解析文本
  127. </button>
  128. </form>
  129. </div>
  130. </div>
  131. <!-- 参考图片和CAD图纸并排布局 -->
  132. <div class="file-upload-grid">
  133. <!-- 参考图片上传区域 -->
  134. <div class="upload-item image-item">
  135. <h5>参考图片</h5>
  136. <div class="file-upload-zone" (click)="imageInput.click()">
  137. <svg width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  138. <rect x="3" y="3" width="18" height="18" rx="2" ry="2"></rect>
  139. <circle cx="8.5" cy="8.5" r="1.5"></circle>
  140. <polyline points="21,15 16,10 5,21"></polyline>
  141. </svg>
  142. <p>点击上传参考图片</p>
  143. <span class="hint">支持多张图片,自动分析色调和材质</span>
  144. </div>
  145. <input #imageInput type="file" multiple accept="image/*" (change)="onFileSelected($event, 'image')" style="display: none;">
  146. </div>
  147. <!-- CAD图纸上传区域 -->
  148. <div class="upload-item cad-item">
  149. <h5>CAD图纸</h5>
  150. <div class="file-upload-zone" (click)="cadInput.click()">
  151. <svg width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  152. <path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"></path>
  153. <polyline points="14,2 14,8 20,8"></polyline>
  154. <line x1="16" y1="13" x2="8" y2="13"></line>
  155. <line x1="16" y1="17" x2="8" y2="17"></line>
  156. </svg>
  157. <p>点击上传CAD图纸</p>
  158. <span class="hint">自动识别结构和空间尺寸</span>
  159. </div>
  160. <input #cadInput type="file" accept=".dwg,.dxf,.pdf" (change)="onFileSelected($event, 'cad')" style="display: none;">
  161. </div>
  162. </div>
  163. <!-- 已上传素材列表 -->
  164. @if (materials.length > 0) {
  165. <div class="materials-list">
  166. <h5>已上传素材</h5>
  167. <div class="material-cards">
  168. @for (material of materials; track material.id) {
  169. <div class="material-card" [class]="'material-' + material.type">
  170. <div class="material-header">
  171. <span class="material-type">{{ material.type === 'text' ? '文本' : material.type === 'image' ? '图片' : 'CAD' }}</span>
  172. <button class="btn-ghost btn-xs" (click)="removeMaterial(material.id)">
  173. <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  174. <line x1="18" y1="6" x2="6" y2="18"></line>
  175. <line x1="6" y1="6" x2="18" y2="18"></line>
  176. </svg>
  177. </button>
  178. </div>
  179. <div class="material-name">{{ material.name }}</div>
  180. @if (material.type === 'image') {
  181. <button class="btn-ghost btn-xs" (click)="previewImage(material.url)">预览图片</button>
  182. } @else if (material.type === 'cad') {
  183. <button class="btn-ghost btn-xs" (click)="previewCad(material.url)">预览CAD</button>
  184. }
  185. @if (material.analysis) {
  186. <div class="parsed-info">
  187. @if (material.type === 'text') {
  188. <div class="parsed-tags">
  189. @if (material.analysis.atmosphere) {
  190. <div class="tag-group">
  191. <span class="tag-label">氛围:</span>
  192. <span class="tag">{{ material.analysis.atmosphere.description }}</span>
  193. </div>
  194. }
  195. </div>
  196. } @else if (material.type === 'image') {
  197. <div class="analysis-results">
  198. <!-- 基础色彩信息和折叠按钮 -->
  199. <div class="analysis-summary">
  200. <div class="color-info">
  201. <span class="color-temp">色温: {{ material.analysis.colorTemperature }}K</span>
  202. @if (material.analysis.mainColors && material.analysis.mainColors.length > 0) {
  203. <span class="color-count">主色: {{ material.analysis.mainColors.length }}种</span>
  204. }
  205. </div>
  206. <button
  207. class="toggle-detail-btn"
  208. (click)="toggleMaterialExpansion(material.id)"
  209. [class.expanded]="isMaterialExpanded(material.id)">
  210. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  211. <polyline [attr.points]="isMaterialExpanded(material.id) ? '18 15 12 9 6 15' : '6 9 12 15 18 9'"></polyline>
  212. </svg>
  213. {{ isMaterialExpanded(material.id) ? '收起详情' : '查看详情' }}
  214. </button>
  215. </div>
  216. <!-- 详细分析(可折叠) -->
  217. @if (isMaterialExpanded(material.id)) {
  218. <div class="detailed-analysis">
  219. <div class="analysis-section">
  220. <h6>色彩分析</h6>
  221. <!-- 色轮可视化组件 -->
  222. @if (material.analysis.mainColors && material.analysis.mainColors.length > 0) {
  223. <div class="color-wheel-visualizer-wrapper">
  224. <app-color-wheel-visualizer
  225. [colors]="material.analysis.mainColors"
  226. [size]="180"
  227. [showLabels]="true"
  228. [showPercentages]="true"
  229. ></app-color-wheel-visualizer>
  230. </div>
  231. }
  232. <!-- 色轮分析 -->
  233. @if (material.analysis.enhancedColorAnalysis?.colorWheel) {
  234. <div class="color-wheel-info">
  235. <div class="color-wheel-icon">
  236. <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  237. <circle cx="12" cy="12" r="10"></circle>
  238. <path d="M12 2a10 10 0 0 1 10 10"></path>
  239. <path d="M12 2a10 10 0 0 0-10 10"></path>
  240. <path d="M12 12l8.66-5"></path>
  241. <path d="M12 12l-8.66-5"></path>
  242. </svg>
  243. </div>
  244. <span>主色调: {{ material.analysis.enhancedColorAnalysis?.colorWheel?.primaryHue }}°</span>
  245. <span>饱和度: {{ material.analysis.enhancedColorAnalysis?.colorWheel?.saturation }}%</span>
  246. </div>
  247. }
  248. <!-- 色彩心理学 -->
  249. @if (material.analysis.enhancedColorAnalysis?.colorPsychology) {
  250. <div class="psychology-info">
  251. <span class="mood-tag">{{ material.analysis.enhancedColorAnalysis?.colorPsychology?.primaryMood }}</span>
  252. <span class="atmosphere-tag">{{ material.analysis.enhancedColorAnalysis?.colorPsychology?.atmosphere }}</span>
  253. </div>
  254. }
  255. </div>
  256. <!-- 形体分析 -->
  257. @if (material.analysis.formAnalysis) {
  258. <div class="analysis-section">
  259. <h6>形体分析</h6>
  260. <!-- 软装形体选择器 -->
  261. <div class="furniture-form-selector-wrapper">
  262. <app-furniture-form-selector
  263. (formSelected)="onFormSelected($event)"
  264. (formDeselected)="onFormDeselected($event)">
  265. </app-furniture-form-selector>
  266. </div>
  267. @if (material.analysis.formAnalysis.overallAssessment) {
  268. <div class="form-metrics">
  269. <div class="metric-item">
  270. <span class="metric-label">复杂度:</span>
  271. <div class="metric-bar">
  272. <div class="metric-fill" [style.width.%]="material.analysis.formAnalysis.overallAssessment.formComplexity"></div>
  273. </div>
  274. </div>
  275. <div class="metric-item">
  276. <span class="metric-label">视觉冲击:</span>
  277. <div class="metric-bar">
  278. <div class="metric-fill" [style.width.%]="material.analysis.formAnalysis.overallAssessment.visualImpact"></div>
  279. </div>
  280. </div>
  281. </div>
  282. }
  283. </div>
  284. }
  285. <!-- 质感分析 -->
  286. @if (material.analysis.textureAnalysis) {
  287. <div class="analysis-section">
  288. <h6>质感分析</h6>
  289. <!-- 质感对比可视化 -->
  290. @if (material.analysis.textureAnalysis.surfaceProperties && material.analysis.textureAnalysis.materialClassification?.primaryMaterial) {
  291. <div class="texture-comparison-wrapper">
  292. <app-texture-comparison-visualizer
  293. [textures]="getTextureDataForMaterial(material)">
  294. </app-texture-comparison-visualizer>
  295. </div>
  296. }
  297. @if (material.analysis.textureAnalysis.materialClassification) {
  298. <div class="material-info">
  299. <span class="material-tag">{{ material.analysis.textureAnalysis.materialClassification.primaryMaterial.category }}</span>
  300. @if (material.analysis.textureAnalysis.surfaceProperties) {
  301. <span class="surface-tag">{{ material.analysis.textureAnalysis.surfaceProperties.roughness.level }}</span>
  302. }
  303. </div>
  304. }
  305. </div>
  306. }
  307. <!-- 纹理分析 -->
  308. @if (material.analysis.patternAnalysis) {
  309. <div class="analysis-section">
  310. <h6>纹理分析</h6>
  311. <!-- 纹理可视化 -->
  312. @if (material.analysis.patternAnalysis.patternRecognition && material.analysis.patternAnalysis.patternRecognition.primaryPatterns.length > 0) {
  313. <div class="pattern-visualizer-wrapper">
  314. <app-pattern-visualizer>
  315. </app-pattern-visualizer>
  316. </div>
  317. }
  318. @if (material.analysis.patternAnalysis.patternRecognition) {
  319. <div class="pattern-info">
  320. @for (pattern of material.analysis.patternAnalysis.patternRecognition.primaryPatterns; track pattern.type) {
  321. <span class="pattern-tag">{{ pattern.type }} ({{ pattern.coverage }}%)</span>
  322. }
  323. </div>
  324. }
  325. </div>
  326. }
  327. <!-- 灯光分析 -->
  328. @if (material.analysis.lightingAnalysis) {
  329. <div class="analysis-section">
  330. <h6>灯光分析</h6>
  331. @if (material.analysis.lightingAnalysis.ambientAnalysis) {
  332. <div class="lighting-info">
  333. <span class="mood-tag">{{ material.analysis.lightingAnalysis.ambientAnalysis.lightingMood?.primary || '未知' }}</span>
  334. @if (material.analysis.lightingAnalysis.illuminationAnalysis) {
  335. <span class="brightness-tag">亮度: {{ material.analysis.lightingAnalysis.illuminationAnalysis.brightness?.overall || 0 }}%</span>
  336. }
  337. @if (material.analysis.lightingAnalysis.lightSourceIdentification) {
  338. <span class="source-tag">光源: {{ material.analysis.lightingAnalysis.lightSourceIdentification.lightingSetup?.dominantSource || '未知' }}</span>
  339. }
  340. </div>
  341. }
  342. <!-- 新增:光比分析 -->
  343. @if (material.analysis.lightingAnalysis.lightingRatio) {
  344. <div class="lighting-ratio-info">
  345. <div class="ratio-header">
  346. <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  347. <path d="M12 2v20M2 12h20"></path>
  348. </svg>
  349. <span class="label">光比分析</span>
  350. </div>
  351. <div class="ratio-details">
  352. <span class="ratio-value">{{ material.analysis.lightingAnalysis.lightingRatio.keyToFill.toFixed(1) }}:1</span>
  353. <span class="ratio-desc">({{ getRatioDescription(material.analysis.lightingAnalysis.lightingRatio.description) }})</span>
  354. </div>
  355. <div class="ratio-bars">
  356. <div class="ratio-bar-item">
  357. <span class="bar-label">主光</span>
  358. <div class="bar-container">
  359. <div class="bar-fill key-light" [style.width.%]="material.analysis.lightingAnalysis.lightingRatio.keyLightIntensity"></div>
  360. </div>
  361. <span class="bar-value">{{ material.analysis.lightingAnalysis.lightingRatio.keyLightIntensity }}%</span>
  362. </div>
  363. <div class="ratio-bar-item">
  364. <span class="bar-label">补光</span>
  365. <div class="bar-container">
  366. <div class="bar-fill fill-light" [style.width.%]="material.analysis.lightingAnalysis.lightingRatio.fillLightIntensity"></div>
  367. </div>
  368. <span class="bar-value">{{ material.analysis.lightingAnalysis.lightingRatio.fillLightIntensity }}%</span>
  369. </div>
  370. </div>
  371. </div>
  372. }
  373. <!-- 新增:光占比分析 -->
  374. @if (material.analysis.lightingAnalysis.lightDistribution) {
  375. <div class="lighting-distribution-info">
  376. <div class="distribution-header">
  377. <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  378. <circle cx="12" cy="12" r="10"></circle>
  379. <line x1="12" y1="2" x2="12" y2="12"></line>
  380. <line x1="12" y1="12" x2="18" y2="16"></line>
  381. </svg>
  382. <span class="label">光占比分析</span>
  383. </div>
  384. <div class="distribution-grid">
  385. <div class="distribution-item">
  386. <div class="dist-icon natural"></div>
  387. <div class="dist-info">
  388. <span class="dist-label">自然光</span>
  389. <span class="dist-value">{{ material.analysis.lightingAnalysis.lightDistribution.natural }}%</span>
  390. </div>
  391. </div>
  392. <div class="distribution-item">
  393. <div class="dist-icon artificial"></div>
  394. <div class="dist-info">
  395. <span class="dist-label">人工光</span>
  396. <span class="dist-value">{{ material.analysis.lightingAnalysis.lightDistribution.artificial }}%</span>
  397. </div>
  398. </div>
  399. <div class="distribution-item">
  400. <div class="dist-icon direct"></div>
  401. <div class="dist-info">
  402. <span class="dist-label">直射光</span>
  403. <span class="dist-value">{{ material.analysis.lightingAnalysis.lightDistribution.direct }}%</span>
  404. </div>
  405. </div>
  406. <div class="distribution-item">
  407. <div class="dist-icon ambient"></div>
  408. <div class="dist-info">
  409. <span class="dist-label">环境光</span>
  410. <span class="dist-value">{{ material.analysis.lightingAnalysis.lightDistribution.ambient }}%</span>
  411. </div>
  412. </div>
  413. <div class="distribution-item">
  414. <div class="dist-icon indirect"></div>
  415. <div class="dist-info">
  416. <span class="dist-label">间接光</span>
  417. <span class="dist-value">{{ material.analysis.lightingAnalysis.lightDistribution.indirect }}%</span>
  418. </div>
  419. </div>
  420. </div>
  421. <div class="distribution-summary">
  422. <span class="summary-tag">{{ getBalanceDescription(material.analysis.lightingAnalysis.lightDistribution.lightingBalance) }}</span>
  423. </div>
  424. </div>
  425. }
  426. </div>
  427. }
  428. </div>
  429. }
  430. </div>
  431. }
  432. </div>
  433. }
  434. </div>
  435. }
  436. </div>
  437. </div>
  438. }
  439. </div>
  440. }
  441. <!-- 需求映射标签页 - 已隐藏,改为后台自动触发 -->
  442. @if (false && activeTab === 'mapping') {
  443. <div class="mapping-section">
  444. <!-- 测试步骤进度 -->
  445. <div class="test-progress">
  446. <h3>需求映射进度</h3>
  447. <div class="progress-summary">
  448. @if (uploadedFiles.length > 0) {
  449. <span class="progress-badge">
  450. 📸 已同步 {{ uploadedFiles.length }} 张参考图片
  451. </span>
  452. }
  453. </div>
  454. <div class="steps-container">
  455. @for (step of testSteps; track step.id) {
  456. <div class="step-item" [class]="getStepClass(step.status)">
  457. <div class="step-icon">{{ getStepIcon(step.status) }}</div>
  458. <div class="step-info">
  459. <div class="step-name">{{ step.name }}</div>
  460. <div class="step-status">
  461. @switch (step.status) {
  462. @case ('pending') { 等待中 }
  463. @case ('in-progress') { 进行中... }
  464. @case ('completed') { 已完成 }
  465. @case ('error') { 失败 }
  466. }
  467. </div>
  468. </div>
  469. </div>
  470. }
  471. </div>
  472. </div>
  473. <!-- 文件上传区域 -->
  474. <div class="upload-section">
  475. <h3>1. 图片上传</h3>
  476. <div class="upload-area" [class.uploading]="isUploading">
  477. @if (uploadedFiles.length === 0) {
  478. <div class="upload-dropzone">
  479. <div class="upload-icon">📁</div>
  480. <div class="upload-text">选择图片文件进行测试</div>
  481. <div class="upload-hint">支持 JPG、PNG 格式,可选择多张图片</div>
  482. <input type="file"
  483. accept="image/*"
  484. multiple
  485. (change)="onFileSelectedForMapping($event)"
  486. class="file-input">
  487. </div>
  488. } @else {
  489. <div class="uploaded-files">
  490. <h4>已上传文件 ({{ uploadedFiles.length }} 张图片)</h4>
  491. <div class="upload-source-info">
  492. <span class="info-tag">✨ 来自素材分析的参考图片</span>
  493. </div>
  494. <div class="files-grid">
  495. @for (file of uploadedFiles; track file.id) {
  496. <div class="file-item">
  497. <img [src]="file.preview || file.url" [alt]="file.name" class="file-preview">
  498. <div class="file-info">
  499. <div class="file-name">{{ file.name }}</div>
  500. <div class="file-size">{{ (file.size! / 1024 / 1024).toFixed(2) }} MB</div>
  501. </div>
  502. <button class="remove-file-btn" (click)="removeUploadedFile(file.id)" title="移除">
  503. <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  504. <line x1="18" y1="6" x2="6" y2="18"></line>
  505. <line x1="6" y1="6" x2="18" y2="18"></line>
  506. </svg>
  507. </button>
  508. </div>
  509. }
  510. </div>
  511. </div>
  512. }
  513. @if (isUploading) {
  514. <div class="loading-overlay">
  515. <div class="loading-spinner"></div>
  516. <div class="loading-text">正在上传文件...</div>
  517. </div>
  518. }
  519. </div>
  520. </div>
  521. <!-- 分析结果区域 -->
  522. <div class="analysis-section" [class.disabled]="uploadedFiles.length === 0">
  523. <h3>2. 图片分析</h3>
  524. @if (isAnalyzing) {
  525. <div class="analysis-loading">
  526. <div class="loading-spinner"></div>
  527. <div class="loading-text">
  528. <h4>正在分析图片...</h4>
  529. <p>
  530. @if (uploadedFiles.length > 1) {
  531. 正在分析 {{ uploadedFiles.length }} 张图片,系统将合并分析结果
  532. } @else {
  533. 系统正在进行色彩、纹理、形态、图案和灯光分析
  534. }
  535. </p>
  536. </div>
  537. </div>
  538. } @else if (analysisError) {
  539. <div class="analysis-error">
  540. <div class="error-icon">❌</div>
  541. <div class="error-text">
  542. <h4>分析失败</h4>
  543. <p>{{ analysisError }}</p>
  544. <button class="retry-btn" (click)="retryAnalysis()">重新分析</button>
  545. </div>
  546. </div>
  547. } @else if (analysisResult) {
  548. <div class="analysis-result">
  549. <h4>分析完成 ✅</h4>
  550. <div class="analysis-summary">
  551. <div class="summary-item">
  552. <span class="label">主要颜色:</span>
  553. <span class="value">{{ analysisResult.enhancedAnalysis?.colorWheel?.colorDistribution?.length || 0 }} 种</span>
  554. </div>
  555. <div class="summary-item">
  556. <span class="label">材质类型:</span>
  557. <span class="value">{{ getMaterialName(analysisResult.textureAnalysis?.materialClassification?.primaryMaterial?.category) || '未识别' }}</span>
  558. </div>
  559. <div class="summary-item">
  560. <span class="label">灯光情绪:</span>
  561. <span class="value">{{ getLightingMoodName(analysisResult.lightingAnalysis?.ambientAnalysis?.lightingMood?.primary) || '未识别' }}</span>
  562. </div>
  563. <div class="summary-item">
  564. <span class="label">空间形态:</span>
  565. <span class="value">{{ analysisResult.formAnalysis?.geometryAnalysis?.complexity ? getComplexityName(analysisResult.formAnalysis.geometryAnalysis.complexity) : '未识别' }}</span>
  566. </div>
  567. </div>
  568. </div>
  569. } @else {
  570. <div class="analysis-placeholder">
  571. <div class="placeholder-icon">🔍</div>
  572. <div class="placeholder-text">
  573. <h4>等待分析</h4>
  574. <p>请先上传图片,系统将自动开始分析</p>
  575. </div>
  576. </div>
  577. }
  578. </div>
  579. <!-- 需求映射结果区域 -->
  580. <div class="mapping-section" [class.disabled]="!analysisResult">
  581. <h3>3. 需求映射生成</h3>
  582. @if (isGeneratingMapping) {
  583. <div class="mapping-loading">
  584. <div class="loading-spinner"></div>
  585. <div class="loading-text">
  586. <h4>正在生成需求映射...</h4>
  587. <p>基于分析结果生成场景参数和映射关系</p>
  588. </div>
  589. </div>
  590. } @else if (mappingError) {
  591. <div class="mapping-error">
  592. <div class="error-icon">❌</div>
  593. <div class="error-text">
  594. <h4>映射生成失败</h4>
  595. <p>{{ mappingError }}</p>
  596. <button class="retry-btn" (click)="retryMapping()">重新生成</button>
  597. </div>
  598. </div>
  599. } @else if (requirementMapping) {
  600. <div class="mapping-result">
  601. <h4>需求映射生成完成 ✅</h4>
  602. <!-- 场景生成信息 -->
  603. <div class="mapping-section-item">
  604. <h5>场景生成</h5>
  605. <div class="scene-info">
  606. <div class="info-row">
  607. <span class="label">基础场景:</span>
  608. <span class="value">{{ requirementMapping?.sceneGeneration?.baseScene || '未定义' }}</span>
  609. </div>
  610. @if (requirementMapping?.sceneGeneration?.atmospherePreview) {
  611. <div class="atmosphere-preview">
  612. <img [src]="requirementMapping.sceneGeneration.atmospherePreview"
  613. alt="氛围感预览图"
  614. class="preview-image">
  615. <div class="preview-label">氛围感预览图</div>
  616. </div>
  617. }
  618. </div>
  619. </div>
  620. <!-- 参数映射信息 -->
  621. <div class="mapping-section-item">
  622. <h5>参数映射</h5>
  623. <div class="params-grid">
  624. <!-- 颜色参数 -->
  625. <div class="param-group">
  626. <h6>颜色映射</h6>
  627. <div class="color-params">
  628. <div class="param-item">
  629. <span class="label">主要颜色:</span>
  630. <span class="value">{{ requirementMapping?.parameterMapping?.colorParams?.primaryColors?.length || 0 }} 种</span>
  631. </div>
  632. <div class="param-item">
  633. <span class="label">色彩和谐:</span>
  634. <span class="value">{{ requirementMapping?.parameterMapping?.colorParams ? getColorHarmonyName(requirementMapping.parameterMapping.colorParams.colorHarmony) : '未知' }}</span>
  635. </div>
  636. <div class="param-item">
  637. <span class="label">饱和度:</span>
  638. <span class="value">{{ requirementMapping?.parameterMapping?.colorParams?.saturation || 0 }}%</span>
  639. </div>
  640. <div class="param-item">
  641. <span class="label">亮度:</span>
  642. <span class="value">{{ requirementMapping?.parameterMapping?.colorParams?.brightness || 0 }}%</span>
  643. </div>
  644. </div>
  645. </div>
  646. <!-- 空间参数 -->
  647. <div class="param-group">
  648. <h6>空间映射</h6>
  649. <div class="space-params">
  650. <div class="param-item">
  651. <span class="label">布局类型:</span>
  652. <span class="value">{{ requirementMapping?.parameterMapping?.spaceParams?.layout ? getLayoutTypeName(requirementMapping.parameterMapping.spaceParams.layout.type) : '未知' }}</span>
  653. </div>
  654. <div class="param-item">
  655. <span class="label">空间流线:</span>
  656. <span class="value">{{ requirementMapping?.parameterMapping?.spaceParams?.layout ? getFlowTypeName(requirementMapping.parameterMapping.spaceParams.layout.flow) : '未知' }}</span>
  657. </div>
  658. <div class="param-item">
  659. <span class="label">家具比例:</span>
  660. <span class="value">{{ requirementMapping?.parameterMapping?.spaceParams?.scale?.furniture || 0 }}%</span>
  661. </div>
  662. <div class="param-item">
  663. <span class="label">开放度:</span>
  664. <span class="value">{{ requirementMapping?.parameterMapping?.spaceParams?.scale?.openness || 0 }}%</span>
  665. </div>
  666. </div>
  667. </div>
  668. <!-- 材质参数 -->
  669. <div class="param-group">
  670. <h6>材质映射</h6>
  671. <div class="material-params">
  672. <div class="param-item">
  673. <span class="label">纹理缩放:</span>
  674. <span class="value">{{ requirementMapping?.parameterMapping?.materialParams?.textureScale || 0 }}%</span>
  675. </div>
  676. <div class="param-item">
  677. <span class="label">反射率:</span>
  678. <span class="value">{{ requirementMapping?.parameterMapping?.materialParams?.reflectivity || 0 }}%</span>
  679. </div>
  680. <div class="param-item">
  681. <span class="label">粗糙度:</span>
  682. <span class="value">{{ requirementMapping?.parameterMapping?.materialParams?.roughness || 0 }}%</span>
  683. </div>
  684. <div class="param-item">
  685. <span class="label">金属度:</span>
  686. <span class="value">{{ requirementMapping?.parameterMapping?.materialParams?.metallic || 0 }}%</span>
  687. </div>
  688. </div>
  689. </div>
  690. </div>
  691. </div>
  692. <!-- 测试结果下载 -->
  693. <div class="test-actions">
  694. <button class="download-btn" (click)="downloadTestResult()">
  695. <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  696. <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path>
  697. <polyline points="7,10 12,15 17,10"></polyline>
  698. <line x1="12" y1="15" x2="12" y2="3"></line>
  699. </svg>
  700. 下载测试结果
  701. </button>
  702. </div>
  703. </div>
  704. } @else {
  705. <div class="mapping-placeholder">
  706. <div class="placeholder-icon">🎯</div>
  707. <div class="placeholder-text">
  708. <h4>等待映射生成</h4>
  709. <p>请先完成图片分析,系统将自动生成需求映射</p>
  710. </div>
  711. </div>
  712. }
  713. </div>
  714. </div>
  715. }
  716. <!-- 分析汇总标签页 -->
  717. <!-- 协作验证标签页 -->
  718. @if (activeTab === 'collaboration') {
  719. <div class="collaboration-section">
  720. <!-- 一致性预警 -->
  721. @if (consistencyWarnings.length > 0) {
  722. <div class="consistency-warning">
  723. <svg viewBox="0 0 24 24" fill="currentColor">
  724. <path d="M12 2L1 21h22L12 2zm0 3.99L19.53 19H4.47L12 5.99zM11 16h2v2h-2v-2zm0-6h2v4h-2v-4z"/>
  725. </svg>
  726. <div>
  727. <strong>一致性预警:</strong>
  728. @for (warning of consistencyWarnings; track warning) {
  729. <div>{{ warning }}</div>
  730. }
  731. </div>
  732. </div>
  733. }
  734. <!-- 需求列表 -->
  735. <div class="requirements-list">
  736. <div class="list-header">
  737. <h5>需求项目</h5>
  738. <div class="list-controls">
  739. <button class="btn-ghost btn-sm" (click)="sortRequirementsByPriority()">按优先级排序</button>
  740. <button class="btn-ghost btn-sm" (click)="saveCurrentState()">保存状态</button>
  741. </div>
  742. </div>
  743. @for (requirement of requirementItems; track requirement.id) {
  744. <div class="requirement-item" [class]="getStatusClass(requirement.status)">
  745. <div class="requirement-header">
  746. <div class="requirement-info">
  747. <h6>{{ requirement.title }}</h6>
  748. <p>{{ requirement.description }}</p>
  749. <div class="requirement-tags">
  750. @for (tag of requirement.tags; track tag) {
  751. <span class="tag">{{ tag }}</span>
  752. }
  753. </div>
  754. </div>
  755. <div class="requirement-meta">
  756. <select class="priority-select"
  757. [value]="requirement.priority"
  758. (change)="onPriorityChange(requirement.id, $event)">
  759. <option value="high">高优先级</option>
  760. <option value="medium">中优先级</option>
  761. <option value="low">低优先级</option>
  762. </select>
  763. <span class="status-badge" [class]="getStatusClass(requirement.status)">
  764. {{ requirement.status === 'confirmed' ? '已确认' : requirement.status === 'rejected' ? '已拒绝' : '待确认' }}
  765. </span>
  766. </div>
  767. </div>
  768. <div class="requirement-actions">
  769. @if (requirement.status === 'pending') {
  770. <button class="btn-success btn-sm" (click)="confirmRequirement(requirement.id)">确认</button>
  771. <button class="btn-danger btn-sm" (click)="rejectRequirement(requirement.id, '需要进一步讨论')">拒绝</button>
  772. }
  773. <button class="btn-ghost btn-sm" (click)="requirement.showComments = !requirement.showComments">
  774. 评论 ({{ getCommentsForRequirement(requirement.id).length }})
  775. @if (hasUnreadComments(requirement.id)) {
  776. <span class="unread-indicator"></span>
  777. }
  778. </button>
  779. </div>
  780. @if (requirement.showComments) {
  781. <div class="comments-section">
  782. @if (getCommentsForRequirement(requirement.id).length > 0) {
  783. <div class="comments-list">
  784. @for (comment of getCommentsForRequirement(requirement.id); track comment.id) {
  785. <div class="comment-item" [class.resolved]="comment.status === 'resolved'">
  786. <div class="comment-header">
  787. <span class="comment-author">{{ comment.author }}</span>
  788. <span class="comment-role">{{ comment.role === 'designer' ? '设计师' : comment.role === 'customer-service' ? '客服' : '客户' }}</span>
  789. <span class="comment-time">{{ comment.timestamp | date:'MM-dd HH:mm' }}</span>
  790. @if (comment.status === 'pending') {
  791. <button class="btn-ghost btn-xs" (click)="resolveComment(comment.id)">标记已解决</button>
  792. }
  793. </div>
  794. <div class="comment-content">{{ comment.content }}</div>
  795. </div>
  796. }
  797. </div>
  798. }
  799. <div class="add-comment">
  800. <textarea #commentInput placeholder="添加评论..." rows="2"></textarea>
  801. <button class="btn-primary btn-sm" (click)="addCommentToRequirement(requirement.id, commentInput.value); commentInput.value = ''">
  802. 发送
  803. </button>
  804. </div>
  805. </div>
  806. }
  807. </div>
  808. }
  809. </div>
  810. </div>
  811. }
  812. <!-- 进度管理标签页 -->
  813. @if (activeTab === 'progress') {
  814. <div class="progress-section">
  815. <!-- 整体进度 -->
  816. <div class="overall-progress">
  817. <div class="section-header">
  818. <h5>整体进度</h5>
  819. </div>
  820. <div class="progress-container">
  821. <div class="progress-visual">
  822. <div class="linear-progress">
  823. <div class="progress-track">
  824. <div class="progress-bar-fill" [style.width.%]="getProgressPercentage()"></div>
  825. </div>
  826. <div class="progress-text">{{ getProgressPercentage() }}%</div>
  827. </div>
  828. <div class="circular-progress">
  829. <svg viewBox="0 0 120 120" class="progress-circle">
  830. <circle cx="60" cy="60" r="54" fill="none" stroke="#e5e7eb" stroke-width="8"></circle>
  831. <circle cx="60" cy="60" r="54" fill="none" stroke="#3b82f6" stroke-width="8"
  832. stroke-linecap="round"
  833. [attr.stroke-dasharray]="339.292"
  834. [attr.stroke-dashoffset]="339.292 - (339.292 * getProgressPercentage() / 100)"
  835. transform="rotate(-90 60 60)">
  836. </circle>
  837. </svg>
  838. <div class="progress-text">
  839. <div class="progress-percentage">{{ getProgressPercentage() }}%</div>
  840. <div class="progress-label">完成度</div>
  841. </div>
  842. </div>
  843. </div>
  844. </div>
  845. </div>
  846. <!-- 进度统计 -->
  847. <div class="progress-stats">
  848. <div class="stat-item">
  849. <div class="stat-number">{{ getRequirementCountByStatus('confirmed') }}</div>
  850. <div class="stat-label">已确认</div>
  851. </div>
  852. <div class="stat-item">
  853. <div class="stat-number">{{ getRequirementCountByStatus('pending') }}</div>
  854. <div class="stat-label">待确认</div>
  855. </div>
  856. <div class="stat-item">
  857. <div class="stat-number">{{ getRequirementCountByStatus('rejected') }}</div>
  858. <div class="stat-label">已拒绝</div>
  859. </div>
  860. <div class="stat-item">
  861. <div class="stat-number">{{ getUnresolvedCommentsCount() }}</div>
  862. <div class="stat-label">待解决评论</div>
  863. </div>
  864. </div>
  865. <!-- 历史状态 -->
  866. <div class="history-section">
  867. <div class="section-header">
  868. <h5>历史状态</h5>
  869. <div class="history-controls">
  870. <button class="btn-ghost btn-sm" (click)="saveCurrentState()">保存当前状态</button>
  871. <select class="history-select" (change)="onHistoryStateChange($event)">
  872. <option value="-1">选择历史状态</option>
  873. @for (state of historyStates; track $index) {
  874. <option [value]="$index">{{ state.timestamp | date:'MM-dd HH:mm' }} - {{ state.author }}</option>
  875. }
  876. </select>
  877. </div>
  878. </div>
  879. @if (historyStates.length > 0) {
  880. <div class="history-timeline">
  881. @for (state of historyStates; track $index) {
  882. <div class="timeline-item" (click)="restoreHistoryState($index)">
  883. <div class="timeline-marker"></div>
  884. <div class="timeline-content">
  885. <div class="timeline-header">
  886. <span class="timeline-time">{{ state.timestamp | date:'MM-dd HH:mm' }}</span>
  887. <span class="timeline-author">{{ state.author }}</span>
  888. </div>
  889. <div class="timeline-summary">
  890. 状态快照:{{ state.requirementItems.length }}个需求项,
  891. {{ getRequirementCountByStatus('confirmed') }}个已确认
  892. </div>
  893. </div>
  894. </div>
  895. }
  896. </div>
  897. } @else {
  898. <div class="empty-history">
  899. <p>暂无历史状态记录</p>
  900. <button class="btn-primary btn-sm" (click)="saveCurrentState()">保存当前状态</button>
  901. </div>
  902. }
  903. </div>
  904. <!-- 需求状态分布 -->
  905. <div class="status-distribution">
  906. <h5>需求状态分布</h5>
  907. <div class="status-bars">
  908. <div class="status-bar">
  909. <div class="status-info">
  910. <span class="status-name">已确认</span>
  911. <span class="status-count">{{ getRequirementCountByStatus('confirmed') }}</span>
  912. </div>
  913. <div class="status-progress">
  914. <div class="progress-bar confirmed"
  915. [style.width.%]="getStatusPercentage('confirmed')"></div>
  916. </div>
  917. </div>
  918. <div class="status-bar">
  919. <div class="status-info">
  920. <span class="status-name">待确认</span>
  921. <span class="status-count">{{ getRequirementCountByStatus('pending') }}</span>
  922. </div>
  923. <div class="status-progress">
  924. <div class="progress-bar pending"
  925. [style.width.%]="getStatusPercentage('pending')"></div>
  926. </div>
  927. </div>
  928. <div class="status-bar">
  929. <div class="status-info">
  930. <span class="status-name">已拒绝</span>
  931. <span class="status-count">{{ getRequirementCountByStatus('rejected') }}</span>
  932. </div>
  933. <div class="status-progress">
  934. <div class="progress-bar rejected"
  935. [style.width.%]="getStatusPercentage('rejected')"></div>
  936. </div>
  937. </div>
  938. </div>
  939. </div>
  940. </div>
  941. }
  942. </div>
  943. </div>
  944. <!-- 保存状态和手动保存按钮 -->
  945. <div class="save-section">
  946. <div class="save-status">
  947. <span class="save-icon" [class]="'save-icon-' + saveStatus">{{ getSaveStatusIcon() }}</span>
  948. <span class="save-text">{{ getSaveStatusText() }}</span>
  949. </div>
  950. <div class="save-actions">
  951. <button class="btn-secondary"
  952. [disabled]="isSaving || !hasUnsavedChanges"
  953. (click)="manualSave()">
  954. @if (isSaving) {
  955. <span class="loading-spinner"></span>
  956. 保存中...
  957. } @else {
  958. 手动保存
  959. }
  960. </button>
  961. <div class="auto-save-toggle">
  962. <label class="toggle-label">
  963. <input type="checkbox"
  964. [(ngModel)]="autoSaveEnabled"
  965. class="toggle-input">
  966. <span class="toggle-slider"></span>
  967. <span class="toggle-text">自动保存</span>
  968. </label>
  969. </div>
  970. </div>
  971. </div>
  972. <!-- 上传成功弹窗 - 移到父组件以避免定位问题 -->
  973. <!-- 全局提示(支持全屏与角落两模式) -->
  974. <app-global-prompt
  975. [visible]="showGlobalPrompt"
  976. [title]="promptTitle"
  977. [message]="promptMessage"
  978. [mode]="promptMode"
  979. [position]="promptPosition"
  980. [autoDismissMs]="4000"
  981. (closed)="onPromptClose()">
  982. </app-global-prompt>
  983. <!-- 完整报告全屏覆盖层 -->
  984. <app-full-report-overlay
  985. [visible]="showFullReportOverlay"
  986. [colorResult]="colorAnalysisResult"
  987. [cadResult]="cadAnalysisResult"
  988. (close)="onCloseFullReport()">
  989. </app-full-report-overlay>
  990. <!-- 上传成功弹窗 -->
  991. <app-upload-success-modal
  992. [isVisible]="showUploadSuccessModal"
  993. [uploadedFiles]="uploadedFiles"
  994. [uploadType]="uploadType"
  995. [analysisResult]="colorAnalysisResult"
  996. [isAnalyzing]="isAnalyzingColors"
  997. (closeModal)="onModalClose()">
  998. </app-upload-success-modal>