| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809 |
- <!-- 基于Product表的报价编辑器组件 -->
- <div class="quotation-editor">
- <!-- 加载状态 -->
- @if (loading) {
- <div class="loading-container">
- <div class="spinner">
- <div class="spinner-circle"></div>
- </div>
- <p>加载报价数据...</p>
- </div>
- }
- @if (!loading) {
- <!-- 产品管理区域 -->
- @if (canEdit) {
- <div class="product-management">
- <div class="product-header">
- <h3>设计产品 ({{ products.length }}个空间)</h3>
- <div class="product-actions">
- <button class="btn-primary" (click)="generateQuotationFromProducts()">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M447.1 96h-288C131.3 96 112 115.3 112 140.4v231.2C112 396.7 131.3 416 159.1 416h288c27.6 0 48-19.3 48-44.4V140.4C495.1 115.3 475.6 96 447.1 96zM447.1 144v192h-288V144H447.1z"/>
- </svg>
- 生成报价
- </button>
- <button class="btn-secondary" (click)="openAddProductModal()">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM256 48c114.7 0 208 93.31 208 208s-93.31 208-208 208S48 370.7 48 256S141.3 48 256 48zM256 336c13.25 0 24-10.75 24-24V280h32c13.25 0 24-10.75 24-24s-10.75-24-24-24h-32V176c0-13.25-10.75-24-24-24s-24 10.75-24 24v56H200c-13.25 0-24 10.75-24 24s10.75 24 24 24h32v32C232 325.3 242.8 336 256 336z"/>
- </svg>
- 添加产品
- </button>
- <button class="btn-outline" (click)="cleanupDuplicateProducts()" title="清理重复产品">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M432 64c26.5 0 48 21.5 48 48v288c0 26.5-21.5 48-48 48H80c-26.5 0-48-21.5-48-48V112c0-26.5 21.5-48 48-48h352zm0-32H80C35.8 32 0 67.8 0 112v288c0 44.2 35.8 80 80 80h352c44.2 0 80-35.8 80-80V112c0-44.2-35.8-80-80-80zM362.7 184l-82.7 82.7L197.3 184c-6.2-6.2-16.4-6.2-22.6 0-6.2 6.2-6.2 16.4 0 22.6l82.7 82.7-82.7 82.7c-6.2 6.2-6.2 16.4 0 22.6 6.2 6.2 16.4 6.2 22.6 0l82.7-82.7 82.7 82.7c6.2 6.2 16.4 6.2 22.6 0 6.2-6.2 6.2-16.4 0-22.6L303.3 289.3l82.7-82.7c6.2-6.2 6.2-16.4 0-22.6-6.2-6.2-16.4-6.2-22.6 0z"/>
- </svg>
- 清理重复
- </button>
- <button class="btn-outline" (click)="saveQuotation()">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M504.1 141C490.6 121.4 471.1 107.5 447.8 96C424.6 84.51 400.8 80 376.1 80H136c-24.74 0-48.48 4.511-71.79 16.01C40.88 107.5 21.36 121.4 7.85 141C-5.654 160.6-1.466 180.2 11.66 195.7L144.1 353c11.14 13.4 27.62 21 44.8 21h124.3c17.18 0 33.66-7.6 44.8-21l133.3-157.4C504.5 180.2 508.6 160.6 504.1 141z"/>
- </svg>
- 保存报价
- </button>
- </div>
- </div>
- </div>
- }
- @if (quotation.spaces.length === 0 && products.length > 0) {
- <!-- 空状态 - 有产品但未生成报价 -->
- <div class="empty-state">
- <svg class="icon empty-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M447.1 96h-288C131.3 96 112 115.3 112 140.4v231.2C112 396.7 131.3 416 159.1 416h288c27.6 0 48-19.3 48-44.4V140.4C495.1 115.3 475.6 96 447.1 96zM447.1 144v192h-288V144H447.1z"/>
- </svg>
- <p class="empty-message">尚未生成报价</p>
- <p class="empty-hint">已加载 {{ products.length }} 个设计空间,请点击"生成报价"按钮</p>
- @if (canEdit) {
- <button class="btn-primary" (click)="generateQuotationFromProducts()">立即生成报价</button>
- }
- </div>
- } @else if (quotation.spaces.length === 0) {
- <!-- 完全空状态 - 无产品 -->
- <div class="empty-state">
- <svg class="icon empty-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 0C114.6 0 0 114.6 0 256s114.6 256 256 256s256-114.6 256-256S397.4 0 256 0zM256 464c-114.7 0-208-93.31-208-208S141.3 48 256 48s208 93.31 208 208S370.7 464 256 464z"/>
- </svg>
- <p class="empty-message">暂无设计产品</p>
- <p class="empty-hint">该项目还没有创建任何设计产品</p>
- @if (canEdit) {
- <button class="btn-primary" (click)="openAddProductModal()">创建第一个产品</button>
- }
- </div>
- } @else {
- <!-- 报价工具栏 -->
- <div class="quotation-toolbar">
- <div class="toolbar-left">
- <h4 class="toolbar-title">报价明细 ({{ quotation.spaces.length }}个设计空间)</h4>
- <div class="toolbar-meta">
- @if (quotation.generatedAt) {
- <span class="generate-time">生成于: {{ quotation.generatedAt | date:'MM-dd HH:mm' }}</span>
- }
- @if (quotation.validUntil) {
- <span class="valid-until">有效期至: {{ quotation.validUntil | date:'yyyy-MM-dd' }}</span>
- }
- </div>
- </div>
- <div class="toolbar-right">
- <button class="btn-icon" (click)="expandAll()" title="展开全部">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M112 184l144 144 144-144M256 328V88"/>
- </svg>
- </button>
- <button class="btn-icon" (click)="collapseAll()" title="折叠全部">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M112 328l144-144 144 144M256 184v240"/>
- </svg>
- </button>
- </div>
- </div>
- <!-- 卡片视图 -->
- @if (viewMode === 'card') {
- <div class="quotation-products">
- @for (space of quotation.spaces; track space.name) {
- <div class="product-card" [class.expanded]="isProductExpanded(space.name)">
- <!-- 产品头部 -->
- <div class="product-header" (click)="toggleProductExpand(space.name)">
- <div class="product-info">
- <div class="product-title">
- <div class="product-icon">
- <i class="icon-{{ getProductIconForSpace(space.name) }}"></i>
- </div>
- <div class="product-details">
- <h3 class="product-name">{{ space.name }}</h3>
- <div class="product-meta">
- <span class="badge" [attr.data-color]="getStatusColorForSpace(space.productId)">
- {{ getStatusTextForSpace(space.productId) }}
- </span>
- <span class="designer-name">{{ getDesignerNameForSpace(space.productId) }}</span>
- </div>
- </div>
- </div>
- <div class="product-pricing">
- <p class="product-subtotal">{{ formatPrice(calculateSpaceSubtotal(space)) }}</p>
- @if (quotation.spaceBreakdown?.length > 1) {
- <span class="percentage">{{ formatPercentage(getSpacePercentage(space.productId)) }}</span>
- }
- </div>
- </div>
- <div class="product-actions">
- @if (canEdit) {
- <button class="btn-icon" (click)="openEditProductModal(space.productId); $event.stopPropagation()" title="编辑">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M362.7 19.32C387.7-5.678 428.3-5.678 453.3 19.32l39.38 39.38c25 25 25 65.62 0 90.62l-109.5 109.5c-25 25-65.62 25-90.62 0l-109.5-109.5c-25-25-25-65.62 0-90.62L272.1 19.32z"/>
- </svg>
- </button>
- <button class="btn-icon danger" (click)="deleteProduct(space.productId); $event.stopPropagation()" title="删除">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M96 480c0 17.67 14.33 32 32 32h256c17.67 0 32-14.33 32-32V128H96V480zM312 192c13.25 0 24 10.75 24 24v208c0 13.25-10.75 24-24 24c-13.25 0-24-10.75-24-24V216C288 202.8 298.8 192 312 192zM200 192c13.25 0 24 10.75 24 24v208c0 13.25-10.75 24-24 24s-24-10.75-24-24V216C176 202.8 186.8 192 200 192z"/>
- </svg>
- </button>
- }
- <div class="product-toggle">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 294.1L383 167c9.4-9.4 24.6-9.4 33.9 0s9.3 24.6 0 34L273 345c-9.1 9.1-23.7 9.3-33.1.7L95 201.1c-4.7-4.7-7-10.9-7-17s2.3-12.3 7-17c9.4-9.4 24.6-9.4 33.9 0l127.1 127z"/>
- </svg>
- </div>
- </div>
- </div>
- <!-- 产品详情 -->
- @if (isProductExpanded(space.name)) {
- <div class="product-content">
- <!-- 产品信息与价格明细 -->
- @if (getProductForSpace(space.productId)) {
- <div class="product-details-section">
- <h5 class="section-title">
- <svg class="title-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM216 336h24V272H216c-13.3 0-24-10.7-24-24s10.7-24 24-24h48c13.3 0 24 10.7 24 24v88h8c13.3 0 24 10.7 24 24s-10.7 24-24 24H216c-13.3 0-24-10.7-24-24s10.7-24 24-24z"/>
- </svg>
- 空间信息与报价明细
- </h5>
-
- <!-- 价格卡片 -->
- <div class="price-cards-grid">
- <div class="price-card base-price">
- <div class="price-card-icon">
- <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M512 80c0 18-14.3 34.6-38.4 48c-29.1 16.1-72.5 27.5-122.3 30.9c-3.7-1.8-7.4-3.5-11.3-5C300.6 137.4 248.2 128 192 128c-8.3 0-16.4 .2-24.5 .6l-1.1-.6C142.3 114.6 128 98 128 80c0-44.2 86-80 192-80S512 35.8 512 80zM160.7 161.1c10.2-.7 20.7-1.1 31.3-1.1c62.2 0 117.4 12.3 152.5 31.4C369.3 204.9 384 221.7 384 240c0 4-.7 7.9-2.1 11.7c-4.6 13.2-17 25.3-35 35.5c0 0 0 0 0 0c-.1 .1-.3 .1-.4 .2l0 0 0 0c-.3 .2-.6 .3-.9 .5c-35 19.4-90.8 32-153.6 32c-59.6 0-112.9-11.3-148.2-29.1c-1.9-.9-3.7-1.9-5.5-2.9C14.3 274.6 0 258 0 240c0-34.8 53.4-64.5 128-75.4c10.5-1.5 21.4-2.7 32.7-3.5zM416 240c0-21.9-10.6-39.9-24.1-53.4c28.3-4.4 54.2-11.4 76.2-20.5c16.3-6.8 31.5-15.2 43.9-25.5V176c0 19.3-16.5 37.1-43.8 50.9c-14.6 7.4-32.4 13.7-52.4 18.5c.1-1.8 .2-3.5 .2-5.3zm-32 96c0 18-14.3 34.6-38.4 48c-1.8 1-3.6 1.9-5.5 2.9C304.9 404.7 251.6 416 192 416c-62.8 0-118.6-12.6-153.6-32C14.3 370.6 0 354 0 336V300.6c12.5 10.3 27.6 18.7 43.9 25.5C83.4 342.6 135.8 352 192 352s108.6-9.4 148.1-25.9c7.8-3.2 15.3-6.9 22.4-10.9c6.1-3.4 11.8-7.2 17.2-11.2c1.5-1.1 2.9-2.3 4.3-3.4V304v5.7V336zm32 0V304 278.1c19-4.2 36.5-9.5 52.1-16c16.3-6.8 31.5-15.2 43.9-25.5V272c0 10.5-5 21-14.9 30.9c-16.3 16.3-45 29.7-81.3 38.4c.1-1.7 .2-3.5 .2-5.3zM192 448c56.2 0 108.6-9.4 148.1-25.9c16.3-6.8 31.5-15.2 43.9-25.5V432c0 44.2-86 80-192 80S0 476.2 0 432V396.6c12.5 10.3 27.6 18.7 43.9 25.5C83.4 438.6 135.8 448 192 448z"/>
- </svg>
- </div>
- <div class="price-card-content">
- <div class="price-card-label">基础报价</div>
- <div class="price-card-value">{{ formatPrice(getProductForSpace(space.productId)?.get('quotation')?.basePrice || 0) }}</div>
- </div>
- </div>
- <div class="price-card total-price">
- <div class="price-card-icon">
- <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M64 32C28.7 32 0 60.7 0 96V416c0 35.3 28.7 64 64 64H448c35.3 0 64-28.7 64-64V192c0-35.3-28.7-64-64-64H80c-8.8 0-16-7.2-16-16s7.2-16 16-16H448c17.7 0 32-14.3 32-32s-14.3-32-32-32H64zM416 272c17.7 0 32 14.3 32 32s-14.3 32-32 32H352c-17.7 0-32-14.3-32-32s14.3-32 32-32h64z"/>
- </svg>
- </div>
- <div class="price-card-content">
- <div class="price-card-label">空间总价</div>
- <div class="price-card-value highlight">{{ formatPrice(calculateSpaceSubtotal(space)) }}</div>
- </div>
- </div>
- </div>
- <!-- 详细信息 -->
- <div class="detail-grid">
- <div class="detail-item">
- <svg class="detail-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M0 96C0 60.7 28.7 32 64 32H448c35.3 0 64 28.7 64 64V416c0 35.3-28.7 64-64 64H64c-35.3 0-64-28.7-64-64V96z"/>
- </svg>
- <div class="detail-content">
- <span class="detail-label">产品类型</span>
- <span class="detail-value">{{ getProductForSpace(space.productId)?.get('productType') }}</span>
- </div>
- </div>
- <div class="detail-item">
- <svg class="detail-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M448 96V224H288V96H448zm0 192V416H288V288H448zM224 224H64V96H224V224zM64 288H224V416H64V288zM64 32C28.7 32 0 60.7 0 96V416c0 35.3 28.7 64 64 64H448c35.3 0 64-28.7 64-64V96c0-35.3-28.7-64-64-64H64z"/>
- </svg>
- <div class="detail-content">
- <span class="detail-label">空间面积</span>
- <span class="detail-value">{{ getProductForSpace(space.productId)?.get('space')?.area || 0 }}㎡</span>
- </div>
- </div>
- <div class="detail-item">
- <svg class="detail-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM369 209L241 337c-9.4 9.4-24.6 9.4-33.9 0l-64-64c-9.4-9.4-9.4-24.6 0-33.9s24.6-9.4 33.9 0l47 47L335 175c9.4-9.4 24.6-9.4 33.9 0s9.4 24.6 0 33.9z"/>
- </svg>
- <div class="detail-content">
- <span class="detail-label">复杂度</span>
- <span class="detail-value">{{ getProductForSpace(space.productId)?.get('space')?.complexity || 'medium' }}</span>
- </div>
- </div>
- </div>
- </div>
- }
- <!-- 协作分工管理 -->
- @if (canEdit) {
- <div class="collaboration-section">
- <h5 class="section-title">
- <svg class="title-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 0c53 0 96 43 96 96v3.6c0 15.7-12.7 28.4-28.4 28.4H188.4c-15.7 0-28.4-12.7-28.4-28.4V96c0-53 43-96 96-96zM41.4 105.4c12.5-12.5 32.8-12.5 45.3 0l64 64c.7 .7 1.3 1.4 1.9 2.1c14.2-7.3 30.4-11.4 47.5-11.4H312c17.1 0 33.2 4.1 47.5 11.4c.6-.7 1.2-1.4 1.9-2.1l64-64c12.5-12.5 32.8-12.5 45.3 0s12.5 32.8 0 45.3l-64 64c-.7 .7-1.4 1.3-2.1 1.9c6.2 12 10.1 25.3 11.1 39.5H480c17.7 0 32 14.3 32 32s-14.3 32-32 32H416c0 24.6-5.5 47.8-15.4 68.6c2.2 1.3 4.2 2.9 6 4.8l64 64c12.5 12.5 12.5 32.8 0 45.3s-32.8 12.5-45.3 0l-63.1-63.1c-24.5 21.8-55.8 36.2-90.3 39.6V240c0-8.8-7.2-16-16-16s-16 7.2-16 16V479.2c-34.5-3.4-65.8-17.8-90.3-39.6L86.6 502.6c-12.5 12.5-32.8 12.5-45.3 0s-12.5-32.8 0-45.3l64-64c1.9-1.9 3.9-3.4 6-4.8C101.5 367.8 96 344.6 96 320H32c-17.7 0-32-14.3-32-32s14.3-32 32-32H96.3c1.1-14.1 5-27.5 11.1-39.5c-.7-.6-1.4-1.2-2.1-1.9l-64-64c-12.5-12.5-12.5-32.8 0-45.3z"/>
- </svg>
- 协作分工管理
- <span class="section-subtitle">(少数需协作情况可手动设置)</span>
- </h5>
-
- <button class="btn-add-collaboration" (click)="openCollaborationModal(space)">
- <svg class="btn-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM232 344V280H168c-13.3 0-24-10.7-24-24s10.7-24 24-24h64V168c0-13.3 10.7-24 24-24s24 10.7 24 24v64h64c13.3 0 24 10.7 24 24s-10.7 24-24 24H280v64c0 13.3-10.7 24-24 24s-24-10.7-24-24z"/>
- </svg>
- 添加协作人员
- </button>
- @if (getSpaceCollaborators(space.productId).length > 0) {
- <div class="collaborators-list">
- @for (collab of getSpaceCollaborators(space.productId); track collab.id) {
- <div class="collaborator-card">
- <div class="collaborator-info">
- <div class="collaborator-avatar">
- @if (collab.profile?.get('avatar')) {
- <img [src]="collab.profile.get('avatar')" [alt]="collab.profile.get('realName')">
- } @else {
- <div class="avatar-placeholder">{{ collab.profile?.get('realName')?.charAt(0) || '?' }}</div>
- }
- </div>
- <div class="collaborator-details">
- <div class="collaborator-name">{{ collab.profile?.get('realName') || '未知' }}</div>
- <div class="collaborator-role">{{ collab.role }}</div>
- </div>
- </div>
- <div class="collaborator-allocation">
- <div class="allocation-input-group">
- <label>工作占比</label>
- <input
- type="number"
- [(ngModel)]="collab.workload"
- (ngModelChange)="onCollaborationChange(space)"
- min="0"
- max="100"
- class="workload-input">
- <span class="unit">%</span>
- </div>
- <div class="allocation-amount">
- <label>分配金额</label>
- <div class="amount-display">¥{{ calculateCollaboratorAmount(space, collab.workload) }}</div>
- </div>
- </div>
- @if (canEdit) {
- <button class="btn-remove" (click)="removeCollaborator(space, collab.id)" title="移除">
- <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 48C141.1 48 48 141.1 48 256s93.1 208 208 208 208-93.1 208-208S370.9 48 256 48zm52.7 283.3L256 278.6l-52.7 52.7c-6.2 6.2-16.4 6.2-22.6 0-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3l52.7-52.7-52.7-52.7c-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3 6.2-6.2 16.4-6.2 22.6 0l52.7 52.7 52.7-52.7c6.2-6.2 16.4-6.2 22.6 0 6.2 6.2 6.2 16.4 0 22.6L278.6 256l52.7 52.7c6.2 6.2 6.2 16.4 0 22.6-6.2 6.3-16.4 6.3-22.6 0z"/>
- </svg>
- </button>
- }
- </div>
- }
- </div>
- }
- </div>
- }
- <!-- 内部分配明细(3个分配:建模阶段10%、软装渲染40%、公司分配50%) -->
- <div class="allocation-section-detail">
- <h5 class="section-title">内部执行分配 <span class="section-subtitle">(基于设计图总价自动分配)</span></h5>
- <div class="allocation-grid-detail">
- @for (allocationType of allocationTypes; track allocationType.key) {
- <div
- class="allocation-item-detail"
- [class.enabled]="isProcessEnabled(space, allocationType.key)"
- [attr.data-type]="allocationType.key">
- <div class="allocation-header-detail">
- <div class="allocation-left">
- <label class="checkbox-wrapper">
- <input
- type="checkbox"
- class="checkbox-input"
- [checked]="isProcessEnabled(space, allocationType.key)"
- (change)="canEdit && toggleProcess(space, allocationType.key)"
- [disabled]="!canEdit" />
- <span class="checkbox-custom"></span>
- </label>
- <div class="allocation-info-detail">
- <span class="allocation-name-detail">{{ allocationType.name }}</span>
- <span class="allocation-desc-detail">{{ allocationType.description }}</span>
- </div>
- </div>
- <span class="allocation-percentage-badge">{{ allocationType.percentage }}%</span>
- </div>
- @if (isProcessEnabled(space, allocationType.key)) {
- <div class="allocation-input-section">
- <label class="input-label-small">分配金额(可编辑)</label>
- <div class="input-with-currency">
- <span class="currency-symbol">¥</span>
- <input
- class="input-field amount-input"
- type="number"
- [ngModel]="getAllocationAmount(space, allocationType.key)"
- (ngModelChange)="setAllocationAmount(space, allocationType.key, $event); onProcessChange()"
- [disabled]="!canEdit"
- placeholder="0" />
- </div>
- <div class="allocation-hint">
- 建议金额: {{ forSpacePrice(space,allocationType) }}
- </div>
- </div>
- }
- </div>
- }
- </div>
- </div>
- </div>
- }
- </div>
- }
- </div>
- }
- <!-- 报价汇总 -->
- <div class="quotation-summary">
- <div class="summary-header">
- <h4>报价汇总</h4>
- @if (quotation.spaceBreakdown?.length > 1) {
- <div class="breakdown-toggle">
- <button class="btn-text" (click)="showBreakdown = !showBreakdown">
- {{ showBreakdown ? '隐藏' : '显示'}}明细
- </button>
- </div>
- }
- </div>
- @if (quotation.spaceBreakdown?.length > 1 && showBreakdown) {
- <div class="breakdown-list">
- @for (item of quotation.spaceBreakdown; track item.spaceId) {
- <div class="breakdown-item">
- <span class="breakdown-name">{{ item.spaceName }}</span>
- <span class="breakdown-amount">{{ formatPrice(item.amount) }}</span>
- <span class="breakdown-percentage">{{ formatPercentage(item.percentage) }}</span>
- </div>
- }
- </div>
- }
- <!-- 内部执行分配 -->
- @if (quotation.allocation) {
- <div class="allocation-section">
- <div class="allocation-header">
- <h4>内部执行分配</h4>
- <button class="btn-text" (click)="toggleAllocation()">
- {{ showAllocation ? '隐藏' : '显示' }}
- </button>
- </div>
- @if (showAllocation) {
- <div class="allocation-list">
- <!-- 建模阶段 -->
- <div class="allocation-item modeling">
- <div class="allocation-icon">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M234.5 5.709C248.4 .7377 263.6 .7377 277.5 5.709L469.5 74.28C494.1 83.38 512 107.5 512 134.6C512 161.7 494.1 185.8 469.5 194.9L277.5 263.5C263.6 268.5 248.4 268.5 234.5 263.5L42.47 194.9C17.05 185.8 0 161.7 0 134.6C0 107.5 17.05 83.38 42.47 74.28L234.5 5.709z"/>
- </svg>
- </div>
- <div class="allocation-info">
- <span class="allocation-name">{{ allocationRules.modeling.label }}</span>
- <span class="allocation-desc">{{ allocationRules.modeling.description }}</span>
- </div>
- <div class="allocation-values">
- <span class="allocation-percentage">{{ quotation.allocation.modeling.percentage }}%</span>
- <span class="allocation-amount">{{ formatPrice(quotation.allocation.modeling.amount) }}</span>
- </div>
- </div>
- <!-- 软装渲染 -->
- <div class="allocation-item decoration">
- <div class="allocation-icon">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M384 160C366.3 160 352 145.7 352 128C352 110.3 366.3 96 384 96C401.7 96 416 110.3 416 128C416 145.7 401.7 160 384 160z"/>
- </svg>
- </div>
- <div class="allocation-info">
- <span class="allocation-name">{{ allocationRules.decoration.label }}</span>
- <span class="allocation-desc">{{ allocationRules.decoration.description }}</span>
- </div>
- <div class="allocation-values">
- <span class="allocation-percentage">{{ quotation.allocation.decoration.percentage }}%</span>
- <span class="allocation-amount">{{ formatPrice(quotation.allocation.decoration.amount) }}</span>
- </div>
- </div>
- <!-- 公司分配 -->
- <div class="allocation-item company">
- <div class="allocation-icon">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M320 32c-8.1 0-16.1 1.4-23.7 4.1L15.8 137.4C6.3 140.9 0 149.9 0 160s6.3 19.1 15.8 22.6l57.9 20.9C57.3 229.3 48 259.8 48 291.9v28.1c0 28.4-10.8 57.7-22.3 80.8c-6.5 13-13.9 25.8-22.5 37.6C0 442.7-.9 448.3 .9 453.4s6 8.9 11.2 10.2l64 16c4.2 1.1 8.7 .3 12.4-2s6.3-6.1 7.1-10.4c8.6-42.8 4.3-81.2-2.1-108.7C90.3 344.3 86 329.8 80 316.5V295.6c0-4.2 .3-8.4 1-12.4L288.1 406.6c8.4 3.2 17.4 4.8 26.6 4.8H448c17.7 0 32-14.3 32-32V256c0-17.7-14.3-32-32-32H448c-17.7 0-32-14.3-32-32s14.3-32 32-32h32c9.7 0 18.5-5.8 22.2-14.8s1.7-19.3-5.2-26.2l-64-64c-12.5-12.5-32.8-12.5-45.3 0l-64 64c-6.9 6.9-8.9 17.2-5.2 26.2s12.5 14.8 22.2 14.8h32c17.7 0 32 14.3 32 32s-14.3 32-32 32H320z"/>
- </svg>
- </div>
- <div class="allocation-info">
- <span class="allocation-name">{{ allocationRules.company.label }}</span>
- <span class="allocation-desc">{{ allocationRules.company.description }}</span>
- </div>
- <div class="allocation-values">
- <span class="allocation-percentage">{{ quotation.allocation.company.percentage }}%</span>
- <span class="allocation-amount">{{ formatPrice(quotation.allocation.company.amount) }}</span>
- </div>
- </div>
- </div>
- <div class="allocation-note">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM216 336h24V272H216c-13.3 0-24-10.7-24-24s10.7-24 24-24h48c13.3 0 24 10.7 24 24v88h8c13.3 0 24 10.7 24 24s-10.7 24-24 24H216c-13.3 0-24-10.7-24-24s10.7-24 24-24z"/>
- </svg>
- <span>内部执行分配为系统自动计算,基于报价总额按固定比例拆分,所有金额均为整数</span>
- </div>
- }
- </div>
- }
- <div class="total-section">
- <div class="total-row">
- <span class="total-label">报价总额</span>
- <span class="total-amount">{{ formatPrice(quotation.total) }}</span>
- </div>
- @if (quotation.generatedAt) {
- <div class="total-meta">
- <span class="generate-info">生成于 {{ quotation.generatedAt | date:'yyyy-MM-dd HH:mm' }}</span>
- @if (quotation.validUntil) {
- <span class="valid-info">有效期至 {{ quotation.validUntil | date:'yyyy-MM-dd' }}</span>
- }
- </div>
- }
- </div>
- </div>
- }
- }
- </div>
- <!-- 协作人员选择模态框 -->
- @if (showCollaborationModal) {
- <div class="modal-overlay" (click)="closeCollaborationModal()">
- <div class="modal-container collaboration-modal" (click)="$event.stopPropagation()">
- <div class="modal-header">
- <h3>添加协作人员</h3>
- <button class="close-btn" (click)="closeCollaborationModal()">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 48C141.1 48 48 141.1 48 256s93.1 208 208 208 208-93.1 208-208S370.9 48 256 48zm52.7 283.3L256 278.6l-52.7 52.7c-6.2 6.2-16.4 6.2-22.6 0-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3l52.7-52.7-52.7-52.7c-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3 6.2-6.2 16.4-6.2 22.6 0l52.7 52.7 52.7-52.7c6.2-6.2 16.4-6.2 22.6 0 6.2 6.2 6.2 16.4 0 22.6L278.6 256l52.7 52.7c6.2 6.2 6.2 16.4 0 22.6-6.2 6.3-16.4 6.3-22.6 0z"/>
- </svg>
- </button>
- </div>
- <div class="modal-body">
- @if (loadingCollaborators) {
- <div class="loading-state">
- <div class="spinner"></div>
- <p>加载团队成员中...</p>
- </div>
- } @else {
- <!-- 搜索框 -->
- <div class="search-box">
- <svg class="search-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M416 208c0 45.9-14.9 88.3-40 122.7L502.6 457.4c12.5 12.5 12.5 32.8 0 45.3s-32.8 12.5-45.3 0L330.7 376c-34.4 25.2-76.8 40-122.7 40C93.1 416 0 322.9 0 208S93.1 0 208 0S416 93.1 416 208zM208 352c79.5 0 144-64.5 144-144s-64.5-144-144-144S64 128.5 64 208s64.5 144 144 144z"/>
- </svg>
- <input
- type="text"
- [(ngModel)]="collaboratorSearchTerm"
- (ngModelChange)="filterCollaborators()"
- placeholder="搜索团队成员..."
- class="search-input">
- </div>
- <!-- 团队成员列表 -->
- <div class="members-list">
- @if (filteredAvailableCollaborators.length === 0) {
- <div class="empty-state-small">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM216 336h24V272H216c-13.3 0-24-10.7-24-24s10.7-24 24-24h48c13.3 0 24 10.7 24 24v88h8c13.3 0 24 10.7 24 24s-10.7 24-24 24H216c-13.3 0-24-10.7-24-24s10.7-24 24-24z"/>
- </svg>
- <p>没有找到可用的团队成员</p>
- </div>
- } @else {
- @for (member of filteredAvailableCollaborators; track member.id) {
- <div
- class="member-item"
- [class.selected]="isCollaboratorSelected(member.id)"
- (click)="toggleCollaboratorSelection(member)">
- <div class="member-avatar">
- @if (member.get('avatar')) {
- <img [src]="member.get('avatar')" [alt]="member.get('realName')">
- } @else {
- <div class="avatar-placeholder">{{ member.get('realName')?.charAt(0) || '?' }}</div>
- }
- </div>
- <div class="member-info">
- <div class="member-name">{{ member.get('realName') || '未知' }}</div>
- <div class="member-department">{{ member.get('department')?.get('name') || '未分配部门' }}</div>
- </div>
- <div class="member-checkbox">
- <svg *ngIf="isCollaboratorSelected(member.id)" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 512c141.4 0 256-114.6 256-256S397.4 0 256 0S0 114.6 0 256S114.6 512 256 512zM369 209L241 337c-9.4 9.4-24.6 9.4-33.9 0l-64-64c-9.4-9.4-9.4-24.6 0-33.9s24.6-9.4 33.9 0l47 47L335 175c9.4-9.4 24.6-9.4 33.9 0s9.4 24.6 0 33.9z"/>
- </svg>
- </div>
- </div>
- }
- }
- </div>
- <!-- 已选择的成员 -->
- @if (selectedCollaborators.length > 0) {
- <div class="selected-section">
- <h4>已选择 ({{ selectedCollaborators.length }}人)</h4>
- <div class="selected-list">
- @for (collab of selectedCollaborators; track collab.member.id) {
- <div class="selected-item">
- <div class="selected-info">
- <div class="selected-avatar">
- @if (collab.member.get('avatar')) {
- <img [src]="collab.member.get('avatar')" [alt]="collab.member.get('realName')">
- } @else {
- <div class="avatar-placeholder">{{ collab.member.get('realName')?.charAt(0) }}</div>
- }
- </div>
- <span>{{ collab.member.get('realName') }}</span>
- </div>
- <div class="selected-inputs">
- <div class="input-group-small">
- <label>角色</label>
- <select [(ngModel)]="collab.role" class="role-select">
- <option value="协作设计师">协作设计师</option>
- <option value="建模师">建模师</option>
- <option value="渲染师">渲染师</option>
- <option value="软装师">软装师</option>
- </select>
- </div>
- <div class="input-group-small">
- <label>占比</label>
- <input
- type="number"
- [(ngModel)]="collab.workload"
- min="0"
- max="100"
- class="workload-input-small">
- <span class="unit">%</span>
- </div>
- </div>
- <button class="btn-remove-small" (click)="removeSelectedCollaborator(collab.member.id)">
- <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 48C141.1 48 48 141.1 48 256s93.1 208 208 208 208-93.1 208-208S370.9 48 256 48zm52.7 283.3L256 278.6l-52.7 52.7c-6.2 6.2-16.4 6.2-22.6 0-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3l52.7-52.7-52.7-52.7c-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3 6.2-6.2 16.4-6.2 22.6 0l52.7 52.7 52.7-52.7c6.2-6.2 16.4-6.2 22.6 0 6.2 6.2 6.2 16.4 0 22.6L278.6 256l52.7 52.7c6.2 6.2 6.2 16.4 0 22.6-6.2 6.3-16.4 6.3-22.6 0z"/>
- </svg>
- </button>
- </div>
- }
- </div>
- </div>
- }
- }
- </div>
- <div class="modal-footer">
- <button class="btn-secondary" (click)="closeCollaborationModal()">取消</button>
- <button
- class="btn-primary"
- (click)="confirmCollaborators()"
- [disabled]="selectedCollaborators.length === 0">
- 确认添加 ({{ selectedCollaborators.length }})
- </button>
- </div>
- </div>
- </div>
- }
- <!-- 产品添加/编辑模态框 -->
- @if (showAddProductModal) {
- <div class="modal-overlay" (click)="closeAddProductModal()">
- <div class="modal-container add-product-modal" (click)="$event.stopPropagation()">
- <div class="modal-header">
- <h3>{{ isEditMode ? '编辑设计产品' : '添加设计产品' }}</h3>
- <button class="close-btn" (click)="closeAddProductModal()">
- <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
- <path fill="currentColor" d="M256 48C141.1 48 48 141.1 48 256s93.1 208 208 208 208-93.1 208-208S370.9 48 256 48zm52.7 283.3L256 278.6l-52.7 52.7c-6.2 6.2-16.4 6.2-22.6 0-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3l52.7-52.7-52.7-52.7c-3.1-3.1-4.7-7.2-4.7-11.3 0-4.1 1.6-8.2 4.7-11.3 6.2-6.2 16.4-6.2 22.6 0l52.7 52.7 52.7-52.7c6.2-6.2 16.4-6.2 22.6 0 6.2 6.2 6.2 16.4 0 22.6L278.6 256l52.7 52.7c6.2 6.2 6.2 16.4 0 22.6-6.2 6.3-16.4 6.3-22.6 0z"/>
- </svg>
- </button>
- </div>
- <div class="modal-body">
- <!-- 预设场景选择 -->
- <div class="form-group">
- <label class="form-label">选择空间场景</label>
- <div class="scene-grid">
- @for (scene of getPresetScenes(); track scene) {
- <button
- class="scene-card"
- [class.selected]="newProduct.sceneName === scene"
- (click)="selectScene(scene)">
- <span class="scene-name">{{ scene }}</span>
- </button>
- }
- <button
- class="scene-card custom"
- [class.selected]="newProduct.isCustom"
- (click)="selectCustomScene()">
- <span class="scene-name">自定义</span>
- </button>
- </div>
- </div>
- <!-- 自定义名称 -->
- @if (newProduct.isCustom) {
- <div class="form-group">
- <label class="form-label">产品名称</label>
- <input
- type="text"
- [(ngModel)]="newProduct.productName"
- placeholder="例如:客厅、主卧、大堂等"
- class="form-input">
- </div>
- }
- <!-- 家装专属配置 -->
- @if (projectInfo.projectType === '家装') {
- <div class="form-group">
- <label class="form-label">空间类型</label>
- <div class="radio-group">
- <label class="radio-label">
- <input type="radio" name="spaceType" value="平层" [(ngModel)]="newProduct.spaceType">
- <span>平层</span>
- </label>
- <label class="radio-label">
- <input type="radio" name="spaceType" value="跃层" [(ngModel)]="newProduct.spaceType">
- <span>跃层</span>
- </label>
- <label class="radio-label">
- <input type="radio" name="spaceType" value="挑空" [(ngModel)]="newProduct.spaceType">
- <span>挑空</span>
- </label>
- <label class="radio-label">
- <input type="radio" name="spaceType" value="卧室" [(ngModel)]="newProduct.spaceType">
- <span>卧室</span>
- </label>
- </div>
- </div>
- <div class="form-group">
- <label class="form-label">风格等级</label>
- <select [(ngModel)]="newProduct.styleLevel" class="form-select">
- <option value="基础风格组">基础风格组</option>
- <option value="中级风格组">中级风格组</option>
- <option value="高级风格组">高级风格组</option>
- <option value="顶级风格组">顶级风格组</option>
- </select>
- </div>
- }
- <!-- 工装专属配置 -->
- @if (projectInfo.projectType === '工装') {
- <div class="form-group">
- <label class="form-label">业态类型</label>
- <select [(ngModel)]="newProduct.businessType" class="form-select">
- <option value="办公空间">办公空间</option>
- <option value="商业空间">商业空间</option>
- <option value="娱乐空间">娱乐空间</option>
- <option value="酒店餐厅">酒店餐厅</option>
- <option value="公共空间">公共空间</option>
- </select>
- </div>
- <div class="form-group">
- <label class="form-label">空间类型</label>
- <div class="radio-group">
- <label class="radio-label">
- <input type="radio" name="spaceType" value="门厅空间" [(ngModel)]="newProduct.spaceType">
- <span>门厅空间</span>
- </label>
- <label class="radio-label">
- <input type="radio" name="spaceType" value="封闭空间" [(ngModel)]="newProduct.spaceType">
- <span>封闭空间</span>
- </label>
- </div>
- </div>
- }
- <!-- 建筑类专属配置 -->
- @if (projectInfo.projectType === '建筑类') {
- <div class="form-group">
- <label class="form-label">建筑类型</label>
- <select [(ngModel)]="newProduct.architectureType" class="form-select">
- <option value="门头">门头</option>
- <option value="小型单体">小型单体</option>
- <option value="大型单体">大型单体</option>
- <option value="鸟瞰">鸟瞰</option>
- </select>
- </div>
- }
- <!-- 加价规则配置 -->
- <div class="pricing-adjustments">
- <h5 class="section-title">加价规则配置(可选)</h5>
- @if (projectInfo.projectType === '家装' || projectInfo.projectType === '工装') {
- <div class="form-group">
- <label class="form-label">额外功能区数量</label>
- <div class="input-with-hint">
- <input
- type="number"
- [(ngModel)]="newProduct.adjustments.extraFunction"
- min="0"
- class="form-input"
- placeholder="0">
- <span class="input-hint">
- @if (projectInfo.projectType === '家装') {
- 每增加一个功能区 +100元
- } @else {
- 每增加一个功能区 +400元
- }
- </span>
- </div>
- </div>
- }
- <div class="form-group">
- <label class="form-label">造型复杂度加价</label>
- <div class="input-with-hint">
- <input
- type="number"
- [(ngModel)]="newProduct.adjustments.complexity"
- min="0"
- max="200"
- class="form-input"
- placeholder="0">
- <span class="input-hint">立面和吊顶造型复杂: 0-200元</span>
- </div>
- </div>
- <div class="form-group">
- <label class="checkbox-label">
- <input type="checkbox" [(ngModel)]="newProduct.adjustments.design">
- <span>需要我们设计(原价×2)</span>
- </label>
- </div>
- @if (projectInfo.projectType === '工装') {
- <div class="form-group">
- <label class="checkbox-label">
- <input type="checkbox" [(ngModel)]="newProduct.adjustments.panoramic">
- <span>需要全景渲染(原价×2)</span>
- </label>
- </div>
- }
- </div>
- <!-- 价格预览 -->
- <div class="price-preview">
- <div class="price-preview-row">
- <span class="label">基础价格:</span>
- <span class="price">{{ formatPrice(calculatePreviewBasePrice()) }}</span>
- </div>
- @if (calculatePreviewAdjustmentTotal() > 0) {
- <div class="price-preview-row adjustment">
- <span class="label">加价合计:</span>
- <span class="price">+{{ formatPrice(calculatePreviewAdjustmentTotal()) }}</span>
- </div>
- }
- <div class="price-preview-row total">
- <span class="label">最终价格:</span>
- <span class="price">{{ formatPrice(calculatePreviewFinalPrice()) }}</span>
- </div>
- </div>
- </div>
- <div class="modal-footer">
- <button class="btn-secondary" (click)="closeAddProductModal()">取消</button>
- <button
- class="btn-primary"
- (click)="confirmAddProduct()"
- [disabled]="!isNewProductValid()">
- {{ isEditMode ? '确认修改' : '确认添加' }}
- </button>
- </div>
- </div>
- </div>
- }
|