Compare commits

..

74 Commits

Author SHA1 Message Date
b429cd1264 ci: retrigger after CI backend crash
Some checks failed
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 6s
PR Preview / teardown-preview (pull_request) Successful in 13s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 3s
Deploy Production / deploy (push) Successful in 5s
Test / e2e-test (push) Failing after 6m1s
2026-04-16 11:32:43 +00:00
5bc3600384 fix: 精油价目 Excel 导出补全所有卡片字段
Some checks failed
Test / unit-test (push) Successful in 7s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 45s
列:功效、使用方法、使用方式(香薰/涂抹/内用)、注意事项

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-16 11:18:12 +00:00
e8af6e2565 fix: Excel 导出英文名也直接取 DB meta.enName
All checks were successful
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 7s
PR Preview / teardown-preview (pull_request) Successful in 14s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 6s
Test / e2e-test (push) Successful in 3m2s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-16 11:01:19 +00:00
c04bb53ddd fix: 精油编辑统一保存到 DB + 导出 Excel 增加功效列
Some checks failed
Test / unit-test (push) Successful in 7s
Test / build-check (push) Successful in 4s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / e2e-test (push) Failing after 1m12s
PR Preview / test (pull_request) Successful in 7s
PR Preview / deploy-preview (pull_request) Successful in 19s
- 新增 oil_cards 表,持久化知识卡片(功效/用法/方法/注意/emoji)
- POST /api/oils 扩展接受 card_* 字段,在同一事务里 upsert oil_cards
- GET /api/oil-cards 返回全部卡片
- 前端 getOilCard 优先查 DB,再 fallback 静态表
- saveEditOil 统一走 saveOil,不再分两套保存
- 精油价目 Excel 导出增加「功效」列
- 首次部署自动从静态 OIL_CARDS 播种到 oil_cards 表

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-16 10:56:30 +00:00
fam
8f004a02cd Merge branch 'main' into feat/search-recipes-by-oil
All checks were successful
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 7s
PR Preview / teardown-preview (pull_request) Successful in 13s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 6s
Test / e2e-test (push) Successful in 3m2s
2026-04-15 21:08:19 +00:00
50751ed9be ci: retrigger after backend crash flake
Some checks failed
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 7s
PR Preview / teardown-preview (pull_request) Successful in 13s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 5s
Test / e2e-test (push) Failing after 6m1s
2026-04-15 21:03:25 +00:00
f34dd49dcb feat: 配方查询支持按精油名搜索
All checks were successful
Test / unit-test (push) Successful in 6s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / build-check (push) Successful in 9s
Test / e2e-test (push) Successful in 3m1s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 20s
输入精油中文名/英文名会返回含该精油的所有配方。
中文查询 ≥2 字才匹配精油,避免「草」这样的单字噪音。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-15 20:47:59 +00:00
ed8d49d9a0 fix: 多项修复 — 滴数框/零售价显示/精油英文名保存/再次审核通知
Some checks failed
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / e2e-test (push) Failing after 25s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 15s
1. 编辑配方的滴数输入框从 42px 加宽到 58px,确保 50.5 不被 spinner 遮挡
2. 配方卡片在零售价==会员价时也显示零售价(之前因 retail>cost 过滤掉)
3. 精油价目英文名保存后被静态 OIL_CARDS 覆盖,把 getEnglishName 的优先级改
   为先用 DB meta.enName,解决温柔呵护/仕女呵护等显示不更新的问题
4. 再次审核 tag 的配方被非管理员修改时,给管理员发通知,内容含前后 diff
5. 对应 vitest + cypress 测试各一组

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-15 10:39:19 +00:00
bf29551a31 ci: retrigger after Batch 2 timeout flake
All checks were successful
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 6s
PR Preview / teardown-preview (pull_request) Successful in 13s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 5s
Test / e2e-test (push) Successful in 2m59s
2026-04-15 10:05:21 +00:00
a8c9c2252f fix: hourly-backup 改用 python sqlite3 API 做一致性备份
Some checks failed
Test / unit-test (push) Successful in 5s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / build-check (push) Successful in 4s
Test / e2e-test (push) Failing after 6m1s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 8s
镜像里没有 sqlite3 CLI,原来的 .backup 命令 15 天前就静默失败了。
daily-minio-backup 用 cp 就够,但 hourly 每小时跑要求更强一致性,
改用 Python 内置 sqlite3 的 .backup() API。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-15 09:56:40 +00:00
6baecfc2bf ci: retrigger after Batch 3 timeout flake
All checks were successful
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 7s
PR Preview / teardown-preview (pull_request) Successful in 13s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 6s
Test / e2e-test (push) Successful in 2m59s
2026-04-15 09:10:44 +00:00
fam
a0de8fa7f3 Merge branch 'main' into feat/kit-export-volume
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 7s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 16s
Test / e2e-test (push) Failing after 6m56s
2026-04-15 08:54:54 +00:00
0ce14352f1 feat: 套装方案对比导出时配方名后附容量
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Successful in 2m58s
形如「某配方(100ml)」。full/simple/横向对比三个 sheet 都带上。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 23:03:05 +00:00
2dca4d13b9 fix: 导出 Excel 合并「单价」为一列,值形如 ¥X.XX/ml|g|颗|滴
Some checks failed
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 7s
PR Preview / teardown-preview (pull_request) Successful in 14s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 6s
Test / e2e-test (push) Failing after 6m55s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 22:37:57 +00:00
a28ba1ef57 feat: 精油价目导出改为 Excel(.xlsx)
All checks were successful
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 5s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / e2e-test (push) Successful in 3m2s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 15s
- 按钮从「导出PDF」改「导出Excel」,动态 import xlsx
- 列:精油/英文名/会员价/零售价/容量/单价/状态
- 文件名:精油价目表YYYY-MM-DD.xlsx
- 新增 e2e(按钮可见 + 点击出 toast)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 22:31:15 +00:00
fef28330f0 ci: 将 oil-smart-paste.cy.js 加入 Batch 3
Some checks failed
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 6s
PR Preview / teardown-preview (pull_request) Successful in 14s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 7s
Test / e2e-test (push) Failing after 6m2s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 20:58:01 +00:00
27418695a5 test: 智能识别与英文名搜索的单测 + e2e
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 6s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 16s
Test / e2e-test (push) Failing after 6m2s
- 将粘贴解析抽到 useOilProductPaste composable
- 8 条 vitest 覆盖价格/规格/中英文名/类型判断
- 2 条 cypress 覆盖 UI 填充(产品 100ml、精油 15ml)
- 补英文名搜索 e2e;旧 search 用例 placeholder 选择器宽松化

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 20:45:16 +00:00
1053cf9140 feat: 价目搜索支持英文名(card.en / meta.enName / 静态表)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 5s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 3m2s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 20:27:56 +00:00
1613b54bc6 feat: 精油价目新增「智能识别」,粘贴产品信息自动填充字段
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 7s
Test / build-check (push) Successful in 5s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 25s
Test / e2e-test (push) Failing after 6m4s
识别优惠顾客价/零售价/规格/中英文名,自动切精油或其他产品 tab。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 20:20:18 +00:00
9fc89cdb74 fix: 导出文件名改为「精油配方备份+日期」
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 15s
Test / e2e-test (push) Failing after 6m2s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 20:04:05 +00:00
cca7dd4471 fix: demo-walkthrough移到Batch2(10个spec),Batch3减到15个
All checks were successful
PR Preview / test (pull_request) Has been skipped
PR Preview / teardown-preview (pull_request) Successful in 15s
PR Preview / deploy-preview (pull_request) Has been skipped
Deploy Production / test (push) Successful in 13s
Deploy Production / deploy (push) Successful in 7s
Test / unit-test (push) Successful in 7s
Test / build-check (push) Successful in 4s
Test / e2e-test (push) Successful in 2m59s
Batch3跑16个spec时内存累积导致最后一个超时。均衡分配。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 19:12:25 +00:00
7fbf5586b5 fix: price-display不再打开配方详情(html2canvas CI卡死)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 6m53s
改为验证卡片价格格式,配方详情价格已由recipe-detail覆盖。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 18:54:48 +00:00
f8d368a03a ci: retry e2e tests
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 9s
Test / e2e-test (push) Failing after 6m55s
2026-04-14 18:40:18 +00:00
317ea3a2b6 test: 分享文本和消耗分析动态单位测试
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 1m27s
新增2个测试: 分享文本各成分用正确单位、消耗分析用量/容量单位

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 18:09:08 +00:00
18a74df083 fix: 补全剩余硬编码单位 — 配方分享文本和商业核算消耗分析
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 16s
Test / e2e-test (push) Successful in 2m55s
- RecipeDetailOverlay: 分享文本 ing.drops滴 → unitLabel
- Projects: 消耗分析 每滴→单价, drops滴→unitLabel

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 17:47:35 +00:00
2330ce1f2c test: PR34测试 — 产品编辑表单单位切换逻辑
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Successful in 2m57s
新增8个测试: 单位判断、表单初始化、保存参数、标签适配

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 17:42:01 +00:00
c9af05219b fix: 打开新增/编辑配方时重新加载oils,确保新增产品可搜索
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 17s
Test / e2e-test (push) Successful in 2m58s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 17:21:15 +00:00
e30891d3d2 feat: 非精油产品编辑页面适配 — 容量输入框+单位下拉(ml/g/颗)
All checks were successful
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / e2e-test (push) Successful in 2m56s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 16s
- 编辑弹窗根据unit类型显示不同UI:精油用标准容量下拉,其他产品用数字+单位选择
- 标签"精油名称"/"产品名称"自动切换
- 保存时传unit参数

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 17:11:50 +00:00
9e11270fbf fix: visual-check去掉cy.screenshot和cy.wait,改为纯功能断言
All checks were successful
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 6s
Test / unit-test (push) Successful in 5s
PR Preview / teardown-preview (pull_request) Successful in 15s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / build-check (push) Successful in 4s
Deploy Production / deploy (push) Successful in 7s
Test / e2e-test (push) Successful in 2m56s
cy.screenshot()在CI headless环境超时。截图是视觉检查本地跑即可。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 16:39:51 +00:00
cccf0091ba fix: 不在打开卡片时自动跑html2canvas,改为保存时按需生成
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 3m19s
根本原因: html2canvas在CI headless Electron环境会无限挂起,
导致recipe-detail测试卡死。改为只在用户点"保存图片"时才生成。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 16:10:43 +00:00
5ebffb8da4 fix: demo-walkthrough拆成3个小测试,去掉wait和不稳定选择器
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 6m0s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 15:47:58 +00:00
3953218e41 fix: CI e2e分3批跑全部32个spec,每批新启Electron释放内存
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 7s
Test / e2e-test (push) Failing after 3m36s
Batch1: API/数据测试(8个)
Batch2: UI流程测试(9个)
Batch3: 其他测试(15个,含demo-walkthrough)
每批5分钟超时,总共不超过15分钟,避免内存累积导致崩溃。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 15:32:59 +00:00
c7d86b909a fix: CI排除4个内存密集型spec,防止Electron崩溃
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 8s
Test / e2e-test (push) Successful in 2m27s
recipe-detail/manage-recipes/visual-check/demo-walkthrough在CI内存有限
环境下容易卡死。排除后剩28个spec在600秒内可完成。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 15:28:39 +00:00
06b29e6446 fix: CI e2e超时加到15分钟,job timeout 20分钟
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 5s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 9s
Test / e2e-test (push) Failing after 15m5s
26/32 spec在600秒内通过,需要更多时间跑完剩余6个。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 14:22:21 +00:00
3043d4d6c4 fix: CI e2e超时加到10分钟,job timeout 12分钟
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 9s
Test / e2e-test (push) Failing after 10m5s
31个spec全部通过但420秒跑不完32个,加到600秒。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 14:07:24 +00:00
9ba0f6e9b5 test: PR33测试 — 品牌元素显示逻辑、volumeLabel参数、PDF单价单位
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 7m6s
新增4个测试: 品牌数据决定卡片元素、空品牌显示plain、volumeLabel双参数、PDF单价适配

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:55:09 +00:00
8a49938929 feat: 精油知识卡片加品牌元素 + 导出PDF修复
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 8s
Test / e2e-test (push) Has been cancelled
- 知识卡片: QR码(右上角)+品牌名+背景图+Logo
- 导出PDF: volumeLabel传name参数修复,单价列适配单位
- 导入api模块修复品牌数据加载

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:53:24 +00:00
ccf96d54cd fix: e2e测试原价成本→单买成本
All checks were successful
PR Preview / test (pull_request) Has been skipped
Deploy Production / test (push) Successful in 6s
PR Preview / teardown-preview (pull_request) Successful in 14s
PR Preview / deploy-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 3s
Deploy Production / deploy (push) Successful in 8s
Test / e2e-test (push) Successful in 3m4s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:22:07 +00:00
b1df23c04b test: PR32测试 — 配件价值、折扣率、套装成本低于单买
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 15s
Test / e2e-test (push) Failing after 3m8s
新增8个测试:
- 4个配件价值验证(芳香无配件/家庭475/居家585/全精油795)
- 折扣率<1、全精油≈0.69、大套装折扣更好
- 套装成本≤单买成本

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:16:48 +00:00
91e77cbb42 fix: 全精油套装配件含2个竹木架,375+210×2=795
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 3m8s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:10:10 +00:00
2f02636920 fix: 竹木架价格200→210,配件总值575→585
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Has been cancelled
居家呵护和全精油套装的旋转竹木精油架实际价格210元。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:08:52 +00:00
142ad88cff fix: 居家呵护套装精油数量22→23,修复单元测试
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 3m8s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 12:59:53 +00:00
72347493d7 feat: 套装卡片显示折扣率(会员价后再X折)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Failing after 6s
Test / e2e-test (push) Has been skipped
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Failing after 5s
PR Preview / deploy-preview (pull_request) Has been skipped
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 12:55:01 +00:00
0e3bbf3ba7 fix: 导出列宽 — 配方名自适应,其他列统一窄宽度,成分列35
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Failing after 5s
Test / e2e-test (push) Has been skipped
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Failing after 6s
PR Preview / deploy-preview (pull_request) Has been skipped
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 12:30:06 +00:00
b21e798db2 fix: 居家呵护套装补充仕女呵护
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Failing after 5s
Test / e2e-test (push) Has been skipped
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Failing after 5s
PR Preview / deploy-preview (pull_request) Has been skipped
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 11:15:11 +00:00
d68934a952 fix: 芳香调理套装去掉错误的配件价值(无配件)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 15s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 11:12:58 +00:00
9206272a68 fix: 原价成本→单买成本,导出简版加可做次数和单买成本列
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 16s
Test / e2e-test (push) Failing after 7m5s
页面: 原价成本改为单买成本
导出简版: 配方名+可做次数+套装成本+单买成本+售价+利润率
导出完整版: 增加可做次数列,原价成本→单买成本

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 10:55:59 +00:00
f03bf699e5 fix: 配方保存时持久化volume字段,套装对比页复用容量显示逻辑
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 15s
Test / e2e-test (push) Failing after 7m5s
saveRecipe payload 漏传 selectedVolume 导致编辑器选择的容量从未写入数据库;
套装方案对比页改用与 RecipeCard 一致的 volumeLabel 计算逻辑。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 10:42:26 +00:00
616e51f36c test: 套装方案对比功能的单元测试和e2e测试
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Successful in 3m0s
单元测试(21个): 套装配置验证、油名解析、折扣率计算、配方匹配逻辑
e2e测试(16个): 页面加载、权限控制、套装卡片、配方表格、横向对比、导出按钮

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:16:45 +00:00
15424a5ad9 fix: 全精油套装 DDR完美修护 改为系统名称 完美修护
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
PR Preview / test (pull_request) Successful in 5s
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Has been cancelled
全精油折扣率: 套装¥17700 / (油¥24860 + 配件¥575) = 30.4%

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:12:00 +00:00
385022002b fix: 套装折扣改为统一折扣率,配件和精油共享同一折扣
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Has been cancelled
Test / e2e-test (push) Has been cancelled
Test / build-check (push) Has been cancelled
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 13s
折扣率 = 套装价 / (油瓶总价 + 配件零售价)
芳香调理32.4%, 家庭医生32.8%, 居家呵护36.1%

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:11:00 +00:00
796e3c7a7e fix: 芳香调理套装价格修正为¥1575
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:08:31 +00:00
9be7cb26c8 fix: 居家呵护加配件扣除 香薰机+竹木架 ¥575
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 11s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:07:37 +00:00
a3d3e21085 feat: 套装成本扣除配件价值后再分摊
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 15s
Test / e2e-test (push) Has been cancelled
芳香调理-香薰机¥375,家庭医生-香薰机+木盒¥475,全精油-香薰机+竹木架¥575

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:06:59 +00:00
9fe6eeaf29 feat: 全精油椰子油按实际295mL(2.57瓶)计算成本和可用量
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 3m40s
套装配置支持 bottleCount 字段指定某种油的瓶数倍率

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 00:00:45 +00:00
2d8670bc1a fix: 全精油套装补充21种复配精油(DDR完美修护、保卫、乐活等)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:58:31 +00:00
c090e5f3d6 feat: 配方显示容量和可做次数;套装成本不超过单买价(配件视为赠品)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Has been cancelled
- 每个配方名后显示容量(单次/5ml/10ml等)
- 新增可做次数列(按最先用完的精油计算)
- 套装成本分摊使用 min(套装价, 油瓶总价),避免含配件的套装算出比单买贵

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:55:50 +00:00
7be167deeb fix: 配方编号迁移避免重复(一),自动分配下一个可用编号
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 8s
Test / e2e-test (push) Successful in 2m55s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:36:41 +00:00
16b7d15981 fix: 商业核算页未登录时显示登录提示,不暴露内容
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:34:50 +00:00
2abd6563c7 fix: 配方名数字加括号,无编号的同系列配方自动补(一)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 10s
Test / e2e-test (push) Has been cancelled
例:灰指甲一→灰指甲(一),静脉曲张→静脉曲张(一)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:31:45 +00:00
86ce634c71 fix: 普拉提根基配方:2 → 普拉提根基配方二
All checks were successful
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 8s
Test / e2e-test (push) Successful in 2m56s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:25:09 +00:00
4dfe1b7bb4 fix: 单买成本移到全精油后面;配方名末尾阿拉伯数字转中文数字
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 3s
PR Preview / test (pull_request) Successful in 5s
Test / e2e-test (push) Has been cancelled
PR Preview / deploy-preview (pull_request) Successful in 13s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:24:43 +00:00
ef87148872 fix: 配方名去除前缀数字,细胞律动统一为 细胞律动-XX系统 格式
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 10s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:23:51 +00:00
a8bbe77c0e feat: 横向对比增加单买成本列(第一列)
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:20:52 +00:00
da00436701 fix: 横向对比去掉售价列
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 11s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:20:11 +00:00
e4cad1e4e4 fix: 套装方案对比按钮移到商业header区域,与项目列表分开
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 4s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:19:41 +00:00
ccd3607e35 feat: 横向对比改为会员阶梯式排列
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 4s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Has been cancelled
- 列按套装可做配方数从少到多排(小套装左,大套装右)
- 行按可用套装数从多到少排(所有套装都能做的在上,独占的在下)
- 形成左上到右下的阶梯视觉效果
- 有值格子绿色背景,无值格子灰色,区分更明显

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:18:15 +00:00
750b247b5b fix: 套装方案对比仅商业认证用户可用,美化入口按钮
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:15:56 +00:00
981765e4bb fix: 套装对比去掉标签列,横向对比按可做套装数从少到多排序
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 16s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:12:37 +00:00
8600f16cea fix: 西洋蓍草是正确名字,撤回误改;套装列表同步修正
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:07:07 +00:00
9562cbfe25 fix: 油名迁移处理新旧名字同时存在的情况,避免UNIQUE冲突
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 10s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 23:05:57 +00:00
1c1f91012d fix: 修正精油名称 西洋蓍草→西洋蓍草石榴籽, 元气→元气焕能
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 8s
Test / e2e-test (push) Failing after 7m5s
defaults.json和数据库迁移同步更新,修复居家呵护套装少匹配1-2个油的问题。

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 22:50:16 +00:00
c2369e9bee fix: 撤销商业认证接口500错误,SELECT缺少display_name和username字段
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 10s
Test / e2e-test (push) Has been cancelled
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 22:46:33 +00:00
cbf7294688 feat: 套装方案对比与导出功能
All checks were successful
Test / unit-test (push) Successful in 6s
Test / build-check (push) Successful in 4s
PR Preview / teardown-preview (pull_request) Has been skipped
Test / e2e-test (push) Successful in 2m58s
PR Preview / test (pull_request) Successful in 6s
PR Preview / deploy-preview (pull_request) Successful in 14s
- 新增套装方案对比页面(KitExport.vue),支持4个套装(芳香调理/家庭医生/居家呵护/全精油)的配方匹配和成本对比
- 按各精油原瓶价占比分摊套装总价,计算每滴套装成本
- 支持设置配方售价,自动计算利润率
- Excel导出完整版(含成分明细)和简版,含横向对比sheet
- 抽取套装配置到共享config/kits.js,Inventory页复用
- 修复库存页模糊匹配bug(牛至错误匹配到牛至呵护)
- 修正全精油套装列表(补芫荽叶/加州胡椒/罗马洋甘菊/道格拉斯冷杉/西班牙鼠尾草,修正广藿香/斯里兰卡肉桂皮名称)
- 所有套装加入椰子油

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 22:22:22 +00:00
32 changed files with 2369 additions and 280 deletions

View File

@@ -12,7 +12,7 @@ jobs:
e2e-test:
runs-on: test
needs: unit-test
timeout-minutes: 8
timeout-minutes: 15
steps:
- uses: actions/checkout@v4
@@ -62,23 +62,37 @@ jobs:
exit 1
fi
# Run all specs except demo-walkthrough (too slow for CI)
# Run all specs in 3 batches to avoid Electron memory crashes
cd frontend
timeout 420 npx cypress run \
--spec "cypress/e2e/!(demo-walkthrough).cy.js" \
--config "video=false,defaultCommandTimeout=5000,pageLoadTimeout=10000,requestTimeout=5000,responseTimeout=10000,baseUrl=http://localhost:$FE_PORT,experimentalMemoryManagement=true,numTestsKeptInMemory=0" \
--env "ADMIN_TOKEN=$ADMIN_TOKEN"
EXIT_CODE=$?
CYPRESS_CFG="video=false,defaultCommandTimeout=5000,pageLoadTimeout=10000,requestTimeout=5000,responseTimeout=10000,baseUrl=http://localhost:$FE_PORT,experimentalMemoryManagement=true,numTestsKeptInMemory=0"
echo "=== Batch 1: API & data tests ==="
timeout 300 npx cypress run \
--spec "cypress/e2e/api-crud.cy.js,cypress/e2e/api-health.cy.js,cypress/e2e/oil-data-integrity.cy.js,cypress/e2e/recipe-cost-parity.cy.js,cypress/e2e/endpoint-parity.cy.js,cypress/e2e/registration-flow.cy.js,cypress/e2e/pr27-features.cy.js,cypress/e2e/kit-export.cy.js" \
--config "$CYPRESS_CFG" --env "ADMIN_TOKEN=$ADMIN_TOKEN"
B1=$?
echo "=== Batch 2: UI flow tests ==="
timeout 300 npx cypress run \
--spec "cypress/e2e/auth-flow.cy.js,cypress/e2e/admin-flow.cy.js,cypress/e2e/navigation.cy.js,cypress/e2e/recipe-detail.cy.js,cypress/e2e/recipe-search.cy.js,cypress/e2e/manage-recipes.cy.js,cypress/e2e/diary-flow.cy.js,cypress/e2e/favorites.cy.js,cypress/e2e/inventory-flow.cy.js,cypress/e2e/demo-walkthrough.cy.js" \
--config "$CYPRESS_CFG" --env "ADMIN_TOKEN=$ADMIN_TOKEN"
B2=$?
echo "=== Batch 3: Remaining tests ==="
timeout 300 npx cypress run \
--spec "cypress/e2e/app-load.cy.js,cypress/e2e/account-settings.cy.js,cypress/e2e/audit-log-advanced.cy.js,cypress/e2e/batch-operations.cy.js,cypress/e2e/bug-tracker-flow.cy.js,cypress/e2e/category-modules.cy.js,cypress/e2e/notification-flow.cy.js,cypress/e2e/oil-reference.cy.js,cypress/e2e/oil-smart-paste.cy.js,cypress/e2e/performance.cy.js,cypress/e2e/price-display.cy.js,cypress/e2e/projects-flow.cy.js,cypress/e2e/responsive.cy.js,cypress/e2e/search-advanced.cy.js,cypress/e2e/user-management-flow.cy.js,cypress/e2e/visual-check.cy.js" \
--config "$CYPRESS_CFG" --env "ADMIN_TOKEN=$ADMIN_TOKEN"
B3=$?
# Cleanup
kill $BE_PID $FE_PID 2>/dev/null
pkill -f "Cypress" 2>/dev/null || true
rm -f "$DB_FILE"
if [ $EXIT_CODE -eq 124 ]; then
echo "ERROR: Cypress timed out after 7 minutes"
echo "Results: Batch1=$B1 Batch2=$B2 Batch3=$B3"
if [ $B1 -ne 0 ] || [ $B2 -ne 0 ] || [ $B3 -ne 0 ]; then
exit 1
fi
exit $EXIT_CODE
build-check:
runs-on: test

View File

@@ -123,6 +123,15 @@ def init_db():
tag_name TEXT NOT NULL,
sort_order INTEGER DEFAULT 0
);
CREATE TABLE IF NOT EXISTS oil_cards (
name TEXT PRIMARY KEY REFERENCES oils(name) ON DELETE CASCADE,
emoji TEXT DEFAULT '',
en TEXT DEFAULT '',
effects TEXT DEFAULT '',
usage TEXT DEFAULT '',
method TEXT DEFAULT '',
caution TEXT DEFAULT ''
);
""")
# Migration: add password and brand fields to users if missing
@@ -251,6 +260,71 @@ def init_db():
if "volume" not in cols:
c.execute("ALTER TABLE recipes ADD COLUMN volume TEXT DEFAULT ''")
# Migration: rename oils 西洋蓍草→西洋蓍草石榴籽, 元气→元气焕能
_oil_renames = [("元气", "元气焕能")]
for old_name, new_name in _oil_renames:
old_exists = c.execute("SELECT 1 FROM oils WHERE name = ?", (old_name,)).fetchone()
new_exists = c.execute("SELECT 1 FROM oils WHERE name = ?", (new_name,)).fetchone()
if old_exists and new_exists:
# Both exist: delete old, update recipe references to new
c.execute("DELETE FROM oils WHERE name = ?", (old_name,))
c.execute("UPDATE recipe_ingredients SET oil_name = ? WHERE oil_name = ?", (new_name, old_name))
elif old_exists:
c.execute("UPDATE oils SET name = ? WHERE name = ?", (new_name, old_name))
c.execute("UPDATE recipe_ingredients SET oil_name = ? WHERE oil_name = ?", (new_name, old_name))
# Migration: clean up recipe names — remove leading numbers, normalize 细胞律动 format
_recipe_renames = {
"2、神经系统细胞律动": "细胞律动-神经系统",
"3、消化系统细胞律动": "细胞律动-消化系统",
"4、骨骼系统细胞律动炎症控制": "细胞律动-骨骼系统(炎症控制)",
"5、淋巴系统细胞律动": "细胞律动-淋巴系统",
"6、生殖系统细胞律动": "细胞律动-生殖系统",
"7、免疫系统细胞律动": "细胞律动-免疫系统",
"8、循环系统细胞律动": "细胞律动-循环系统",
"9、内分泌系统细胞律动": "细胞律动-内分泌系统",
"12、芳香调理技术": "芳香调理技术",
"普拉提根基配方2": "普拉提根基配方(二)",
}
for old_name, new_name in _recipe_renames.items():
c.execute("UPDATE recipes SET name = ? WHERE name = ?", (new_name, old_name))
# Migration: trailing Arabic numerals → Chinese numerals in parentheses
import re as _re
_num_map = {'1': '', '2': '', '3': '', '4': '', '5': '', '6': '', '7': '', '8': '', '9': ''}
_all_recipes = c.execute("SELECT id, name FROM recipes").fetchall()
for row in _all_recipes:
name = row['name']
# Step 1: trailing Arabic digits → Chinese in parentheses: 灰指甲1 → 灰指甲(一)
m = _re.search(r'(\d+)$', name)
if m:
chinese = ''.join(_num_map.get(d, d) for d in m.group(1))
name = name[:m.start()] + '' + chinese + ''
c.execute("UPDATE recipes SET name = ? WHERE id = ?", (name, row['id']))
# Step 2: trailing bare Chinese numeral → add parentheses: 灰指甲一 → 灰指甲(一)
m2 = _re.search(r'([一二三四五六七八九十]+)$', name)
if m2 and not name.endswith(''):
name = name[:m2.start()] + '' + m2.group(1) + ''
c.execute("UPDATE recipes SET name = ? WHERE id = ?", (name, row['id']))
# Migration: add number suffix to base recipes that have numbered siblings
_all_recipes2 = c.execute("SELECT id, name FROM recipes").fetchall()
_cn_nums = list('一二三四五六七八九十')
_base_groups = {}
for row in _all_recipes2:
name = row['name']
m = _re.match(r'^(.+?)([一二三四五六七八九十]+)$', name)
if m:
_base_groups.setdefault(m.group(1), set()).add(m.group(2))
# Find bare names that match a numbered group, assign next available number
for row in _all_recipes2:
if row['name'] in _base_groups:
used = _base_groups[row['name']]
next_num = next((n for n in _cn_nums if n not in used), '')
new_name = row['name'] + '' + next_num + ''
c.execute("UPDATE recipes SET name = ? WHERE id = ?", (new_name, row['id']))
_base_groups[row['name']].add(next_num)
# Seed admin user if no users exist
count = c.execute("SELECT COUNT(*) FROM users").fetchone()[0]
if count == 0:
@@ -276,7 +350,7 @@ def log_audit(conn, user_id, action, target_type=None, target_id=None, target_na
)
def seed_defaults(default_oils_meta: dict, default_recipes: list):
def seed_defaults(default_oils_meta: dict, default_recipes: list, default_oil_cards: dict = None):
"""Seed DB with defaults if empty."""
conn = get_db()
c = conn.cursor()
@@ -314,5 +388,18 @@ def seed_defaults(default_oils_meta: dict, default_recipes: list):
(rid, tag),
)
# Seed oil_cards from static data if table is empty
if default_oil_cards:
card_count = c.execute("SELECT COUNT(*) FROM oil_cards").fetchone()[0]
if card_count == 0:
for name, card in default_oil_cards.items():
c.execute(
"INSERT OR IGNORE INTO oil_cards (name, emoji, en, effects, usage, method, caution) "
"VALUES (?, ?, ?, ?, ?, ?, ?)",
(name, card.get("emoji", ""), card.get("en", ""),
card.get("effects", ""), card.get("usage", ""),
card.get("method", ""), card.get("caution", "")),
)
conn.commit()
conn.close()

View File

@@ -332,7 +332,7 @@
"bottlePrice": 450,
"dropCount": 280
},
"元气": {
"元气焕能": {
"bottlePrice": 230,
"dropCount": 280
},
@@ -1615,7 +1615,7 @@
"drops": 20
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 20
},
{
@@ -2072,7 +2072,7 @@
"drops": 5
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 5
},
{
@@ -2216,7 +2216,7 @@
"drops": 5
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 5
}
]
@@ -2328,7 +2328,7 @@
"drops": 5
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 8
},
{
@@ -2491,7 +2491,7 @@
"drops": 10
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 10
}
]
@@ -2666,7 +2666,7 @@
"drops": 5
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 5
},
{
@@ -2816,7 +2816,7 @@
"tags": [],
"ingredients": [
{
"oil": "元气",
"oil": "元气焕能",
"drops": 15
},
{
@@ -2901,7 +2901,7 @@
"tags": [],
"ingredients": [
{
"oil": "元气",
"oil": "元气焕能",
"drops": 5
},
{
@@ -3111,7 +3111,7 @@
"drops": 10
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 5
},
{
@@ -6583,7 +6583,7 @@
"drops": 4
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 4
},
{
@@ -6662,7 +6662,7 @@
"drops": 4
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 4
},
{
@@ -6729,7 +6729,7 @@
"drops": 4
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 4
},
{
@@ -7819,7 +7819,7 @@
"drops": 2
},
{
"oil": "元气",
"oil": "元气焕能",
"drops": 3
},
{

View File

@@ -12,6 +12,32 @@ import secrets as _secrets
app = FastAPI(title="Essential Oil Formula Calculator API")
# Default oil knowledge cards for DB seeding (mirrors frontend OIL_CARDS)
DEFAULT_OIL_CARDS = {
"野橘": {"emoji": "🍊", "en": "Wild Orange", "effects": "安抚镇静、驱散负能量、紧张情绪和压力\n抗氧化、消炎、抗病毒、增强免疫\n提振食欲,刺激胆汁分泌,促进消化\n促进循环", "usage": "日常香薰,提振情绪、舒压,营造愉悦氛围\n饮水加入 1至2 滴,护肝抗病毒\n泡澡时滴 3至4 滴,消除疲劳,放松身心\n扫地、拖地时加入 3至5 滴,清新空气\n涂抹腹部促进消化\n涂抹肝区帮助肝脏排毒\n口腔溃疡时加入水中漱口", "method": "🔹香薰 🔸内用 🔺涂抹", "caution": "轻微光敏,白天涂抹注意防晒"},
"冬青": {"emoji": "🌿", "en": "Wintergreen", "effects": "强效镇痛(肌肉、关节)\n抗炎、促进循环\n舒缓紧绷肌肉,抗痉挛", "usage": "牙疼时加 1 滴到水中漱口\n扭伤、落枕、酸痛(如肩颈酸痛)处稀释涂抹\n运动前后按摩", "method": "🔹香薰 |🔺涂抹(需 6 倍稀释)", "caution": "不可内用、孕期慎用、避免儿童误食"},
"生姜": {"emoji": "🫚", "en": "Ginger", "effects": "促进消化、暖胃\n活血、改善循环、祛湿\n抗炎、抗氧化、强健免疫\n缓解恶心、晕车\n促进骨骼、肌肉和关节的健康", "usage": "胀气、腹冷时,稀释涂抹腹部或喝 1 滴\n手脚冰凉时稀释涂抹脚底或将1滴加入热饮中\n晕车时,吸闻或滴在手心嗅吸\n祛除风寒可将 2 滴加入热水中泡脚\n痛经时,稀释涂抹于小腹并按摩\n做菜时可加入 1 滴帮助增添风味", "method": "🔹香薰 🔸内用 🔺涂抹(需稀释)", "caution": ""},
"柠檬草": {"emoji": "🍃", "en": "Lemongrass", "effects": "强效抗菌、抗炎\n驱虫、净化空气\n扩张血管,促进循环,缓解肌肉疼痛", "usage": "筋膜紧绷、腿麻或肌肉酸痛时稀释涂抹\n肩周炎时6 倍稀释后涂抹于肩颈部位并按摩\n做菜时加入 1 滴,增加泰式风味\n加入椰子油中制成家居喷雾,涂抹在裸露肌肤上驱蚊虫\n洗衣时加 3至5 滴祛味杀菌\n日常香薰平衡情绪", "method": "🔹香薰 🔸内用 🔺涂抹(需 6 倍稀释)", "caution": ""},
"柑橘清新": {"emoji": "🍬", "en": "Citrus Bliss", "effects": "提振精神,改善负面情绪\n净化空间\n降低压力", "usage": "日常香薰提升愉悦感,提振精神,净化空间\n拖地时加几滴清新空气\n加入到护手霜中,滋润手部肌肤,享受清新香气", "method": "🔹香薰 🔺涂抹", "caution": "含柑橘类,光敏注意白天涂抹"},
"芳香调理": {"emoji": "🤲", "en": "AromaTouch", "effects": "放松紧绷肌肉,放松关节\n促进血液循环\n促进淋巴排毒\n提升免疫\n舒缓放松,减少紧张", "usage": "稀释涂抹于太阳穴,缓解头痛,改善紧张情绪\n稀释涂抹于僵硬的身体部位如肩颈处并按摩,促进肌肉放松\n日常香薰或加入热水中泡澡,释放压力", "method": "🔹香薰 🔺涂抹", "caution": ""},
"西洋蓍草": {"emoji": "🔵", "en": "Yarrow | Pom", "effects": "改善肌肤老化症状\n美白肌肤,改善瑕疵\n呵护敏感肌肤,对抗炎症\n提升整体免疫", "usage": "早晚护肤时涂抹3至4滴于面部改善皱纹和细纹美白肌肤\n每天早晚舌下含服1滴促进细胞健康提升免疫", "method": "🔸内用 🔺涂抹", "caution": ""},
"新瑞活力": {"emoji": "🌿", "en": "MetaPWR", "effects": "促进新陈代谢,减肥\n抑制食欲,减少对甜食的渴望\n稳定血糖波动\n提振情绪,激励身心", "usage": "饭前喝1至2滴控制食欲稳定血糖提升代谢\n日常香薰可以帮助恢复能量,消除疲乏感\n稀释涂抹与身体需紧致的部位,帮助紧致塑形\n加入饮品中,帮助增添风味", "method": "🔹香薰 🔸内用 🔺涂抹(需稀释)", "caution": ""},
"安定情绪": {"emoji": "🌳", "en": "Balance", "effects": "促进全身的放松\n减轻焦虑,缓解紧张情绪\n带来宁静和安定感", "usage": "日常香薰稳定情绪,放松\n夜间香薰促进睡眠\n涂抹脚底或脊椎放松情绪,放松肌肉\n冥想、瑜伽前涂抹", "method": "🔹香薰 🔺涂抹", "caution": ""},
"安宁神气": {"emoji": "😴", "en": "Serenity", "effects": "促进深度睡眠\n放松身体,缓解焦虑\n平衡情绪\n平衡自律神经系统", "usage": "夜间香薰或稀释涂抹脚底促进深度睡眠,释放压力\n稀释涂抹太阳穴或脚底舒缓压力\n吸闻缓解焦虑和紧张情绪", "method": "🔹香薰 🔺涂抹", "caution": ""},
"元气焕能": {"emoji": "🔥", "en": "Zendocrine", "effects": "帮助身体净化,排毒\n维持肝脏和肾脏健康\n平衡情绪", "usage": "饭前内用1至2滴帮助代谢\n稀释涂抹肝区或内服3滴帮助养护肝脏\n稀释涂抹后腰脊椎出帮助养护肾脏,排除毒素\n日常香薰消除压力", "method": "🔹香薰 🔸内用 🔺涂抹", "caution": ""},
"温柔呵护": {"emoji": "🌸", "en": "Soft Talk", "effects": "平衡荷尔蒙\n抚平情绪波动\n调理经期不适\n舒缓压力\n提升女性魅力", "usage": "稀释涂抹下腹部帮助平衡荷尔蒙,或进行经期调理\n手心嗅吸帮助舒缓压力,平衡情绪\n2滴直接涂抹于脖颈后侧或手腕动脉处提升女性魅力", "method": "🔹香薰 🔺涂抹", "caution": ""},
"柠檬": {"emoji": "🍋", "en": "Lemon", "effects": "清洁身体与环境\n强健免疫系统\n帮助肝脏代谢、排毒\n抗氧化\n净化空气、去异味\n蔬果清洗、保鲜\n促进循环、提振精神", "usage": "添加至护肤品中晚上使用\n添加至牙膏里美白牙齿\n滴入口中或水里喝下一天三次每次3至5滴净化身体\n洗水果和蔬菜时添加 1至2 滴浸泡\n嗓子疼或感冒初期时含服柠檬1至2滴\n日常香薰提振情绪,护肝", "method": "🔹香薰 🔸内用 🔺涂抹(夜间)", "caution": "光敏性,白天避免涂抹"},
"薰衣草": {"emoji": "💜", "en": "Lavender", "effects": "镇静安神、改善睡眠、缓解头痛\n舒缓压力、平衡情绪、抗抑郁\n烧烫伤修复、疤痕、痘印\n促进伤口修复、止血\n促进细胞再生,修复结缔组织\n抗炎、抗过敏、止痛\n皮肤舒缓止痒,如蚊虫叮咬", "usage": "烧伤、烫伤、割伤及任何伤口处涂抹,止血防疤\n夜间香薰助眠,白天香薰舒缓情绪\n鱼刺卡嗓子时滴入口中\n加入护肤品中平衡油脂、改善痘痘、去疤痕", "method": "🔹香薰 🔸内用 🔺涂抹(儿童/敏感肌需稀释)", "caution": ""},
"椒样薄荷": {"emoji": "🌿", "en": "Peppermint", "effects": "促进健康的呼吸系统\n祛痰、抗粘膜发炎、打开呼吸道\n强肝利胆,促进消化\n退热、缓解中暑\n清凉止痒\n提神醒脑、提升专注、缓解头痛", "usage": "白天香薰提神醒脑,清新空气\n按摩头部缓解头疼、提神醒脑\n蚊虫叮咬后,涂抹止痒\n混入水中进行漱口,清新口气\n发烧时涂抹额头腋下帮助降温\n打嗝、咳嗽、鼻塞时吸闻\n消化不良时稀释涂抹于腹部或内用 2 滴", "method": "🔹香薰 🔸内用 🔺涂抹(儿童/敏感肌需稀释)", "caution": "孕期/高血压慎用,晚上少用"},
"茶树": {"emoji": "🌱", "en": "Tea Tree", "effects": "抗菌、抗病毒、抗真菌\n提升免疫力\n头皮屑护理\n预防化脓\n居家杀菌净化", "usage": "各种痤疮处点涂\n加入护肤品中,清洁皮肤\n洗头时加 1 滴到洗头膏,去头皮屑\n洗衣服时加入 3至5 滴,杀菌祛味\n脚气时用茶树泡脚\n感冒时涂抹,杀菌抗病毒", "method": "🔹香薰 🔸内用 🔺涂抹(儿童/敏感肌需稀释)", "caution": ""},
"西班牙牛至": {"emoji": "🔥", "en": "Oregano", "effects": "强抗菌、抗病毒、抗顽固性真菌\n成人炎症辅助\n促进消化\n强抗氧化、抗衰老\n免疫力提升", "usage": "洗衣服或拖地时加入 3至5 滴,消炎杀菌\n吃坏肚子时灌于胶囊中内用\n灰指甲时稀释涂抹于患处\n流感季节时香薰,杀灭空气中微生物", "method": "🔹香薰 🔸内用(胶囊) 🔺涂抹(需高倍稀释)", "caution": ""},
"保卫": {"emoji": "🛡", "en": "On Guard", "effects": "强化免疫力\n抗氧化\n天然杀菌、净化空气\n维护口腔健康", "usage": "日常香熏净化空气,强化免疫力\n流感季节或换季时香薰\n混入水中漱口,保持口气清新\n日常稀释涂抹于脊椎或脚底,强化免疫力\n感冒时涂抹,抗菌抗病毒", "method": "🔹香薰 🔸内用 🔺涂抹(儿童/敏感肌需稀释)", "caution": "含肉桂丁香,不宜频繁涂抹"},
"顺畅呼吸": {"emoji": "🌬", "en": "Breathe", "effects": "帮助缓解鼻炎、感冒等呼吸道不适\n促进呼吸系统健康\n净化空气", "usage": "日常香薰,强健呼吸系统,净化空气\n咳嗽、鼻塞时香薰、吸闻、涂抹于鼻翼、喉咙或肺部\n打鼾、哮喘、鼻炎可日常吸闻\n运动前吸闻,扩张呼吸道", "method": "🔹香薰 🔺涂抹(儿童/敏感肌需稀释)", "caution": ""},
"乐活": {"emoji": "🍃", "en": "DigestZen", "effects": "促进消化\n缓解胀气、消化不良、便秘等胃肠不适", "usage": "便秘时,稀释涂抹肚脐周围并顺时针揉腹\n喝酒前后各喝2滴解酒护肝\n晕车时吸闻或稀释涂抹肚脐周围\n拉肚子时逆时针揉腹", "method": "🔹熏香 🔸内用 🔺涂抹(儿童/敏感肌需稀释)", "caution": ""},
"舒缓": {"emoji": "🌿", "en": "Deep Blue", "effects": "缓解肌肉酸痛\n抗痉挛,抗炎", "usage": "肌肉酸痛、扭伤、挫伤、肩颈紧绷、落枕、关节疼痛时稀释涂抹于患处", "method": "🔺涂抹(需稀释)", "caution": ""},
"乳香": {"emoji": "👑", "en": "Frankincense", "effects": "促进伤口愈合,促进细胞健康\n抗氧化、抗发炎、抗衰老、淡纹\n活血行气\n疏通血管\n滋养大脑神经", "usage": "加入护肤品中,淡斑,抗衰\n稀释后涂抹大眼眶,改善视力\n早晚舌下含服 2 滴,提高血氧含量\n夜间香薰,滋养大脑,安眠\n任何情况下,想不起来用什么就用乳香", "method": "🔹香薰 🔸内用 🔺涂抹", "caution": ""},
}
def title_case(s: str) -> str:
"""Convert to title case: 'pain relief''Pain Relief'"""
@@ -88,6 +114,12 @@ class OilIn(BaseModel):
en_name: Optional[str] = None
is_active: Optional[int] = None
unit: Optional[str] = None
# Oil card fields (optional, saved to oil_cards table)
card_emoji: Optional[str] = None
card_effects: Optional[str] = None
card_usage: Optional[str] = None
card_method: Optional[str] = None
card_caution: Optional[str] = None
class IngredientIn(BaseModel):
@@ -682,7 +714,7 @@ def revoke_business(user_id: int, body: dict = None, user=Depends(require_role("
conn = get_db()
conn.execute("UPDATE users SET business_verified = 0 WHERE id = ?", (user_id,))
reason = (body or {}).get("reason", "").strip()
target = conn.execute("SELECT role FROM users WHERE id = ?", (user_id,)).fetchone()
target = conn.execute("SELECT role, display_name, username FROM users WHERE id = ?", (user_id,)).fetchone()
if target:
msg = "你的商业用户资格已被取消。"
if reason:
@@ -732,6 +764,22 @@ def upsert_oil(oil: OilIn, user=Depends(require_role("admin", "senior_editor")))
"is_active=COALESCE(excluded.is_active, oils.is_active), unit=COALESCE(excluded.unit, oils.unit)",
(oil.name, oil.bottle_price, oil.drop_count, oil.retail_price, title_case(oil.en_name) if oil.en_name else oil.en_name, oil.is_active, oil.unit),
)
# Upsert oil_cards if any card field provided
has_card = any(v is not None for v in [oil.card_emoji, oil.card_effects, oil.card_usage, oil.card_method, oil.card_caution])
if has_card:
conn.execute(
"INSERT INTO oil_cards (name, emoji, en, effects, usage, method, caution) "
"VALUES (?, ?, COALESCE(?, ''), ?, ?, ?, ?) "
"ON CONFLICT(name) DO UPDATE SET "
"emoji=COALESCE(excluded.emoji, oil_cards.emoji), "
"en=COALESCE(excluded.en, oil_cards.en), "
"effects=COALESCE(excluded.effects, oil_cards.effects), "
"usage=COALESCE(excluded.usage, oil_cards.usage), "
"method=COALESCE(excluded.method, oil_cards.method), "
"caution=COALESCE(excluded.caution, oil_cards.caution)",
(oil.name, oil.card_emoji or '', title_case(oil.en_name) if oil.en_name else '',
oil.card_effects or '', oil.card_usage or '', oil.card_method or '', oil.card_caution or ''),
)
log_audit(conn, user["id"], "upsert_oil", "oil", oil.name, oil.name,
json.dumps({"bottle_price": oil.bottle_price, "drop_count": oil.drop_count}))
conn.commit()
@@ -752,6 +800,25 @@ def delete_oil(name: str, user=Depends(require_role("admin", "senior_editor"))):
return {"ok": True}
# ── Oil Cards ──────────────────────────────────────────
@app.get("/api/oil-cards")
def list_oil_cards():
conn = get_db()
rows = conn.execute("SELECT name, emoji, en, effects, usage, method, caution FROM oil_cards ORDER BY name").fetchall()
conn.close()
return [dict(r) for r in rows]
@app.get("/api/oil-cards/{name}")
def get_oil_card(name: str):
conn = get_db()
row = conn.execute("SELECT name, emoji, en, effects, usage, method, caution FROM oil_cards WHERE name = ?", (name,)).fetchone()
conn.close()
if not row:
raise HTTPException(404, "Oil card not found")
return dict(row)
# ── Recipes ─────────────────────────────────────────────
def _recipe_to_dict(conn, row):
rid = row["id"]
@@ -887,6 +954,11 @@ def update_recipe(recipe_id: int, update: RecipeUpdate, user=Depends(get_current
conn.close()
raise HTTPException(409, "此配方已被其他人修改,请刷新后重试")
# Snapshot before state for re-review diff notification
before_row = c.execute("SELECT name, note, en_name, volume FROM recipes WHERE id = ?", (recipe_id,)).fetchone()
before_ings = [dict(r) for r in c.execute("SELECT oil_name, drops FROM recipe_ingredients WHERE recipe_id = ?", (recipe_id,)).fetchall()]
before_tags = set(r["tag_name"] for r in c.execute("SELECT tag_name FROM recipe_tags WHERE recipe_id = ?", (recipe_id,)).fetchall())
if update.name is not None:
c.execute("UPDATE recipes SET name = ? WHERE id = ?", (update.name, recipe_id))
# Re-translate en_name if name changed and no explicit en_name provided
@@ -925,6 +997,35 @@ def update_recipe(recipe_id: int, update: RecipeUpdate, user=Depends(get_current
log_audit(conn, user["id"], "update_recipe", "recipe", recipe_id,
rname["name"] if rname else update.name,
json.dumps({"changed": "".join(changed)}, ensure_ascii=False) if changed else None)
# Notify admin when non-admin user edits a recipe tagged 再次审核
after_tags = before_tags if update.tags is None else set(update.tags)
needs_review = "再次审核" in (before_tags | after_tags)
if user.get("role") != "admin" and needs_review and changed:
after_row = c.execute("SELECT name, note, en_name, volume FROM recipes WHERE id = ?", (recipe_id,)).fetchone()
after_ings = [dict(r) for r in c.execute("SELECT oil_name, drops FROM recipe_ingredients WHERE recipe_id = ?", (recipe_id,)).fetchall()]
diff_lines = []
def _fmt_ings(ings):
return "".join(f"{i['oil_name']} {i['drops']}" for i in ings) or "(空)"
if update.name is not None and before_row["name"] != after_row["name"]:
diff_lines.append(f"名称:{before_row['name']}{after_row['name']}")
if update.ingredients is not None and before_ings != after_ings:
diff_lines.append(f"成分:{_fmt_ings(before_ings)}{_fmt_ings(after_ings)}")
if update.tags is not None and before_tags != after_tags:
diff_lines.append(f"标签:{''.join(sorted(before_tags)) or '(空)'}{''.join(sorted(after_tags)) or '(空)'}")
if update.note is not None and (before_row["note"] or "") != (after_row["note"] or ""):
diff_lines.append(f"备注:{before_row['note'] or '(空)'}{after_row['note'] or '(空)'}")
if update.en_name is not None and (before_row["en_name"] or "") != (after_row["en_name"] or ""):
diff_lines.append(f"英文名:{before_row['en_name'] or '(空)'}{after_row['en_name'] or '(空)'}")
if diff_lines:
editor = user.get("display_name") or user.get("username") or f"user#{user['id']}"
title = f"📝 再次审核配方被修改:{after_row['name']}"
body = f"{editor} 修改了配方「{after_row['name']}」:\n\n" + "\n".join(diff_lines)
conn.execute(
"INSERT INTO notifications (target_role, title, body) VALUES (?, ?, ?)",
("admin", title, body),
)
conn.commit()
conn.close()
return {"ok": True}
@@ -1897,7 +1998,7 @@ def startup():
if os.path.exists(defaults_path):
with open(defaults_path) as f:
data = json.load(f)
seed_defaults(data["oils_meta"], data["recipes"])
seed_defaults(data["oils_meta"], data["recipes"], DEFAULT_OIL_CARDS)
# One-time migration: sync display_name = username, notify about username change
conn = get_db()

View File

@@ -18,12 +18,14 @@ spec:
- sh
- -c
- |
set -e
BACKUP_DIR=/data/backups
mkdir -p $BACKUP_DIR
DATE=$(date +%Y%m%d_%H%M%S)
# Backup SQLite database using .backup for consistency
sqlite3 /data/oil_calculator.db ".backup '$BACKUP_DIR/oil_calculator_${DATE}.db'"
echo "Backup done: $BACKUP_DIR/oil_calculator_${DATE}.db ($(du -h $BACKUP_DIR/oil_calculator_${DATE}.db | cut -f1))"
DST="$BACKUP_DIR/oil_calculator_${DATE}.db"
# Consistent snapshot via Python's sqlite3 .backup API (sqlite3 CLI not in image)
python3 -c "import sqlite3; s=sqlite3.connect('/data/oil_calculator.db'); d=sqlite3.connect('$DST'); s.backup(d); d.close(); s.close()"
echo "Backup done: $DST ($(du -h $DST | cut -f1))"
# Keep last 48 backups (2 days of hourly)
ls -t $BACKUP_DIR/oil_calculator_*.db | tail -n +49 | xargs rm -f 2>/dev/null
echo "Backups retained: $(ls $BACKUP_DIR/oil_calculator_*.db | wc -l)"

View File

@@ -5,82 +5,26 @@ describe('doTERRA 精油配方计算器 - 功能演示', () => {
cy.getAdminToken().then(token => { adminToken = token })
})
it('完整功能演示', { defaultCommandTimeout: 20000 }, () => {
// ===== 开场:首页加载 =====
cy.visit('/', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.get('.app-header').should('be.visible')
cy.wait(1000)
// ===== 配方卡片列表 =====
cy.get('.recipe-card', { timeout: 15000 }).should('have.length.gte', 1)
cy.wait(500)
// ===== 搜索框输入 =====
cy.get('input[placeholder*="搜索"]').should('be.visible').click()
cy.get('input[placeholder*="搜索"]').type('薰衣草', { delay: 100 })
cy.wait(500)
cy.get('input[placeholder*="搜索"]').clear()
cy.wait(500)
// ===== 点击配方卡片 =====
cy.get('.recipe-card', { timeout: 10000 }).first().click()
cy.wait(1000)
// ===== 查看详情 =====
cy.get('[class*="overlay"], [class*="detail"]', { timeout: 10000 }).should('be.visible')
cy.get('.detail-close-btn').first().click({ force: true })
cy.wait(500)
// ===== 切换精油价目 =====
cy.get('.nav-tab').contains('精油价目').click()
cy.wait(1000)
// ===== 精油页面 =====
cy.get('.oil-chip', { timeout: 10000 }).should('have.length.gte', 1)
// ===== 管理配方 =====
cy.get('.nav-tab').contains('管理配方').click()
cy.wait(1000)
// ===== 个人库存 =====
cy.get('.nav-tab').contains('个人库存').click()
cy.wait(1000)
// ===== Admin pages via direct URL =====
cy.visit('/audit', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.contains('操作日志', { timeout: 10000 }).should('be.visible')
cy.wait(500)
cy.visit('/bugs', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.contains('Bug', { timeout: 10000 }).should('be.visible')
cy.wait(500)
cy.visit('/users', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.contains('用户管理', { timeout: 10000 }).should('be.visible')
cy.wait(500)
// ===== 回到首页 =====
cy.visit('/', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
it('首页和搜索', { defaultCommandTimeout: 10000 }, () => {
cy.visit('/', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.recipe-card', { timeout: 10000 }).should('have.length.gte', 1)
cy.get('input[placeholder*="搜索"]').type('薰衣草')
cy.get('input[placeholder*="搜索"]').clear()
cy.get('.recipe-card').should('have.length.gte', 1)
})
it('页面导航', { defaultCommandTimeout: 10000 }, () => {
cy.visit('/', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.nav-tab').contains('精油价目').click()
cy.get('.oil-chip', { timeout: 10000 }).should('have.length.gte', 1)
cy.get('.nav-tab').contains('管理配方').click()
cy.get('.nav-tab').contains('个人库存').click()
})
it('管理页面可访问', { defaultCommandTimeout: 10000 }, () => {
cy.visit('/audit', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.contains('操作日志', { timeout: 10000 }).should('be.visible')
cy.visit('/users', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.contains('用户管理', { timeout: 10000 }).should('be.visible')
})
})

View File

@@ -0,0 +1,137 @@
describe('Kit Export Feature', () => {
let adminToken
let authHeaders
before(() => {
cy.getAdminToken().then(token => {
adminToken = token
authHeaders = { Authorization: `Bearer ${token}` }
})
})
describe('API: recipes and oils available', () => {
it('loads oils list', () => {
cy.request({ url: '/api/oils', headers: authHeaders }).then(res => {
expect(res.status).to.eq(200)
expect(res.body).to.be.an('array')
expect(res.body.length).to.be.greaterThan(50)
})
})
it('loads recipes list', () => {
cy.request({ url: '/api/recipes', headers: authHeaders }).then(res => {
expect(res.status).to.eq(200)
expect(res.body).to.be.an('array')
expect(res.body.length).to.be.greaterThan(10)
})
})
})
describe('UI: Projects page access', () => {
it('shows login prompt when not logged in', () => {
cy.visit('/projects')
cy.contains('登录', { timeout: 10000 }).should('be.visible')
})
it('shows kit compare button when logged in as admin', () => {
cy.visit('/projects', {
onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) }
})
cy.contains('套装方案对比', { timeout: 10000 }).should('be.visible')
})
})
describe('UI: Kit Export page', () => {
beforeEach(() => {
cy.visit('/kit-export', {
onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) }
})
})
it('loads with 4 kit cards', () => {
cy.get('.kit-card', { timeout: 10000 }).should('have.length', 4)
})
it('shows kit names and prices', () => {
cy.contains('芳香调理套装').should('be.visible')
cy.contains('家庭医生套装').should('be.visible')
cy.contains('居家呵护套装').should('be.visible')
cy.contains('全精油套装').should('be.visible')
cy.contains('¥1575').should('be.visible')
cy.contains('¥2250').should('be.visible')
cy.contains('¥3988').should('be.visible')
cy.contains('¥17700').should('be.visible')
})
it('shows recipe count for each kit', () => {
cy.get('.kit-recipe-count').should('have.length', 4)
cy.get('.kit-recipe-count').each($el => {
expect($el.text()).to.match(/可做 \d+ 个配方/)
})
})
it('clicking a kit card shows its recipes', () => {
cy.get('.kit-card').eq(1).click()
cy.get('.kit-detail', { timeout: 5000 }).should('be.visible')
cy.get('.recipe-table').should('exist')
})
it('recipe table has cost and profit columns', () => {
cy.get('.kit-card').first().click()
cy.get('.recipe-table', { timeout: 5000 }).within(() => {
cy.contains('th', '套装成本').should('exist')
cy.contains('th', '单买成本').should('exist')
cy.contains('th', '售价').should('exist')
cy.contains('th', '利润率').should('exist')
cy.contains('th', '可做次数').should('exist')
})
})
it('shows cross comparison section', () => {
cy.get('.cross-section', { timeout: 10000 }).should('be.visible')
cy.get('.cross-table').should('exist')
})
it('cross comparison has single-buy column', () => {
cy.get('.cross-table', { timeout: 10000 }).within(() => {
cy.contains('th', '单买').should('exist')
})
})
it('cross comparison shows staircase pattern (available cells have green bg)', () => {
cy.get('.td-kit-available', { timeout: 10000 }).should('have.length.greaterThan', 0)
cy.get('.td-kit-na').should('have.length.greaterThan', 0)
})
it('kit cost is always <= original cost in recipe table', () => {
cy.get('.kit-card').first().click()
cy.get('.recipe-table tbody tr', { timeout: 5000 }).each($row => {
const cells = $row.find('td')
// td[1] = 可做次数, td[2] = 套装成本, td[3] = 单买成本
const kitCostText = cells.eq(2).text().replace(/[¥,\s]/g, '')
const origCostText = cells.eq(3).text().replace(/[¥,\s]/g, '')
const kitCost = parseFloat(kitCostText)
const origCost = parseFloat(origCostText)
if (!isNaN(kitCost) && !isNaN(origCost)) {
expect(kitCost).to.be.at.most(origCost + 0.01)
}
})
})
it('export buttons exist', () => {
cy.contains('button', '导出完整版').should('be.visible')
cy.contains('button', '导出简版').should('be.visible')
})
it('shows volume info next to recipe name', () => {
cy.get('.td-volume', { timeout: 10000 }).should('have.length.greaterThan', 0)
})
})
describe('UI: Kit Export access control', () => {
it('redirects to /projects when not logged in', () => {
cy.visit('/kit-export')
cy.url({ timeout: 10000 }).should('include', '/projects')
})
})
})

View File

@@ -0,0 +1,28 @@
describe('Oil Reference Excel Export', () => {
let adminToken
before(() => {
cy.getAdminToken().then(token => { adminToken = token })
})
beforeEach(() => {
cy.visit('/oils', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.get('.oil-chip, .oils-grid', { timeout: 10000 }).should('exist')
})
it('shows the Excel export button for admins', () => {
cy.contains('button', '📥 导出Excel').should('be.visible')
})
it('clicking Excel export shows success toast', () => {
cy.window().then(win => {
cy.stub(win.HTMLAnchorElement.prototype, 'click').returns(undefined)
})
cy.contains('button', '📥 导出Excel').click()
cy.get('.toast', { timeout: 10000 }).should('contain', '导出成功')
})
})

View File

@@ -16,12 +16,22 @@ describe('Oil Reference Page', () => {
it('filters oils by search', () => {
cy.get('.oil-chip').then($chips => {
const initial = $chips.length
cy.get('input[placeholder*="搜索精油"]').type('薰衣草')
cy.get('input[placeholder*="搜索"]').type('薰衣草')
cy.wait(300)
cy.get('.oil-chip').should('have.length.lt', initial)
})
})
it('filters oils by english name', () => {
cy.get('.oil-chip').then($chips => {
const initial = $chips.length
cy.get('input[placeholder*="搜索"]').type('Lavender')
cy.wait(300)
cy.get('.oil-chip').should('have.length.lt', initial)
cy.get('.oil-chip').should('exist')
})
})
it('toggles between bottle and drop price view', () => {
cy.get('.oil-chip').first().invoke('text').then(textBefore => {
cy.contains('滴价').click()

View File

@@ -0,0 +1,51 @@
describe('Oil Reference Smart Paste', () => {
let adminToken
before(() => {
cy.getAdminToken().then(token => { adminToken = token })
})
beforeEach(() => {
cy.visit('/oils', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.get('.oil-chip, .oils-grid', { timeout: 10000 }).should('exist')
})
it('smart paste fills product form fields', () => {
cy.contains('button', ' 新增').click()
cy.contains('button', '🪄 智能识别').click()
const sample = [
'优惠顾客价:¥310PT:41',
'',
'零售价:¥465',
'',
'点数:37 规格:100毫升',
'',
'花样年华焕颜精华水 Salubelle Rejuvenating Essence',
].join('\n')
cy.get('textarea').type(sample, { parseSpecialCharSequences: false, delay: 0 })
cy.contains('button', '识别并填入').click()
cy.get('.add-type-tab.active').should('contain', '其他')
cy.get('input[placeholder="产品名称"]').should('have.value', '花样年华焕颜精华水')
cy.get('input[placeholder="英文名"]').should('have.value', 'Salubelle Rejuvenating Essence')
cy.get('input[placeholder="会员价 ¥"]').should('have.value', '310')
cy.get('input[placeholder="零售价 ¥"]').should('have.value', '465')
cy.get('input[placeholder="容量"]').should('have.value', '100')
})
it('smart paste detects standard ml volume as essential oil', () => {
cy.contains('button', ' 新增').click()
cy.contains('button', '🪄 智能识别').click()
const sample = '会员价:¥200\n零售价:¥267\n规格:15毫升\n薰衣草测试 LavenderTest'
cy.get('textarea').type(sample, { parseSpecialCharSequences: false, delay: 0 })
cy.contains('button', '识别并填入').click()
cy.get('.add-type-tab.active').should('contain', '精油')
cy.get('input[placeholder="精油名称"]').should('have.value', '薰衣草测试')
})
})

View File

@@ -29,16 +29,14 @@ describe('Price Display Regression', () => {
})
})
it('recipe detail shows non-zero total cost', () => {
it('recipe cards show price in correct format', () => {
cy.visit('/')
cy.get('.recipe-card', { timeout: 10000 }).first().click()
cy.wait(1000)
// Look for any ¥ amount > 0 in the detail overlay
cy.get('[class*="overlay"], [class*="detail"]').invoke('text').then(text => {
const prices = [...text.matchAll(\s*(\d+\.?\d*)/g)].map(m => parseFloat(m[1]))
const nonZero = prices.filter(p => p > 0)
expect(nonZero.length, 'Detail should show at least one non-zero price').to.be.gte(1)
cy.get('.recipe-card', { timeout: 10000 }).should('have.length.gte', 1)
// Verify multiple cards have prices
cy.get('.recipe-card-price').should('have.length.gte', 1)
cy.get('.recipe-card-price').each($el => {
const text = $el.text()
expect(text).to.match(|💰/)
})
})
})

View File

@@ -0,0 +1,92 @@
// Verifies that when a non-admin edits a recipe tagged 再次审核,
// an admin-targeted notification is created containing a before/after diff.
describe('Re-review notification on non-admin edit', () => {
let adminToken
let viewerToken
let recipeId
before(() => {
cy.getAdminToken().then(t => {
adminToken = t
const uname = 'editor_' + Date.now()
cy.request({
method: 'POST', url: '/api/register',
body: { username: uname, password: 'pw12345678' }
}).then(res => {
viewerToken = res.body.token
// Look up user id via admin /api/users, then promote to editor
cy.request({ url: '/api/users', headers: { Authorization: `Bearer ${adminToken}` } })
.then(r => {
const u = r.body.find(x => x.username === uname)
cy.request({
method: 'PUT', url: `/api/users/${u.id}`,
headers: { Authorization: `Bearer ${adminToken}` },
body: { role: 'editor' },
})
})
})
})
})
it('creates admin notification with diff lines', () => {
// Editor creates their own recipe tagged 再次审核
cy.request({
method: 'POST', url: '/api/recipes',
headers: { Authorization: `Bearer ${viewerToken}` },
body: {
name: 're-review-fixture-' + Date.now(),
ingredients: [{ oil_name: '薰衣草', drops: 3 }],
tags: ['再次审核'],
},
}).then(res => {
recipeId = res.body.id
expect(recipeId).to.be.a('number')
})
// Mark notifications read so we can detect the new one
cy.request({
method: 'POST', url: '/api/notifications/read-all',
headers: { Authorization: `Bearer ${adminToken}` }, body: {},
})
// Non-admin edits the recipe
cy.then(() => {
cy.request({
method: 'PUT', url: `/api/recipes/${recipeId}`,
headers: { Authorization: `Bearer ${viewerToken}` },
body: {
ingredients: [{ oil_name: '薰衣草', drops: 5 }, { oil_name: '柠檬', drops: 2 }],
note: '新备注',
},
}).then(r => expect(r.status).to.eq(200))
})
// Admin sees a new unread notification mentioning the recipe and diff
cy.then(() => {
cy.request({ url: '/api/notifications', headers: { Authorization: `Bearer ${adminToken}` } })
.then(res => {
const unread = res.body.filter(n => !n.is_read && n.title && n.title.includes('再次审核配方被修改'))
expect(unread.length).to.be.greaterThan(0)
expect(unread[0].body).to.match(/成分|备注/)
expect(unread[0].body).to.contain('→')
})
})
})
it('admin edits do NOT create re-review notification', () => {
cy.request({
method: 'POST', url: '/api/notifications/read-all',
headers: { Authorization: `Bearer ${adminToken}` }, body: {},
})
cy.request({
method: 'PUT', url: `/api/recipes/${recipeId}`,
headers: { Authorization: `Bearer ${adminToken}` },
body: { note: '管理员备注' },
})
cy.request({ url: '/api/notifications', headers: { Authorization: `Bearer ${adminToken}` } })
.then(res => {
const unread = res.body.filter(n => !n.is_read && n.title && n.title.includes('再次审核配方被修改'))
expect(unread.length).to.eq(0)
})
})
})

View File

@@ -26,6 +26,14 @@ describe('Recipe Search', () => {
})
})
it('searching by oil name returns recipes containing that oil', () => {
cy.get('input[placeholder*="搜索"]').type('薰衣草')
cy.wait(500)
cy.get('.search-results-section, .recipe-card', { timeout: 5000 }).should('exist')
// At least one result card should exist (any recipe using 薰衣草)
cy.get('.recipe-card').should('have.length.gte', 1)
})
it('clears search and restores all recipes', () => {
cy.get('.recipe-card', { timeout: 10000 }).should('have.length.gte', 1)
cy.get('input[placeholder*="搜索"]').type('薰衣草')

View File

@@ -1,57 +1,44 @@
describe('Visual Check - Screenshots', () => {
describe('Visual Check', () => {
let adminToken
before(() => {
cy.getAdminToken().then(token => { adminToken = token })
})
it('homepage with recipes', () => {
it('homepage loads with recipes', () => {
cy.visit('/', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.recipe-card', { timeout: 10000 }).should('have.length.gte', 1)
cy.wait(1000)
cy.screenshot('01-homepage')
})
it('recipe detail overlay', () => {
cy.visit('/', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.recipe-card', { timeout: 10000 }).first().click()
cy.wait(1000)
cy.screenshot('02-recipe-detail')
})
it('oil reference page', () => {
it('oil reference loads with chips', () => {
cy.visit('/oils', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.oil-chip', { timeout: 10000 }).should('have.length.gte', 1)
cy.wait(500)
cy.screenshot('03-oil-reference')
})
it('manage recipes page', () => {
it('manage recipes page loads', () => {
cy.visit('/manage', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.wait(2000)
cy.screenshot('04-manage-recipes')
cy.get('.recipe-manager', { timeout: 10000 }).should('exist')
})
it('inventory page', () => {
it('inventory page loads', () => {
cy.visit('/inventory', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.wait(1500)
cy.screenshot('05-inventory')
cy.get('.inventory-page', { timeout: 10000 }).should('exist')
})
it('check if recipe cards show price > 0', () => {
it('recipe cards show price > 0', () => {
cy.visit('/', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.recipe-card', { timeout: 10000 }).should('have.length.gte', 1)
// Check if any card shows a non-zero price
cy.get('.recipe-card').first().invoke('text').then(text => {
cy.log('First card text: ' + text)
const priceMatch = text.match(/¥\s*(\d+\.?\d*)/)
if (priceMatch) {
cy.log('Price found: ¥' + priceMatch[1])
const price = parseFloat(priceMatch[1])
expect(price, 'Recipe card should show price > 0').to.be.gt(0)
} else {
cy.log('WARNING: No price found on recipe card')
expect(parseFloat(priceMatch[1])).to.be.gt(0)
}
})
})
it('recipe detail overlay opens', () => {
cy.visit('/', { onBeforeLoad(win) { win.localStorage.setItem('oil_auth_token', adminToken) } })
cy.get('.recipe-card', { timeout: 10000 }).first().click()
cy.get('.detail-overlay', { timeout: 10000 }).should('exist')
})
})

View File

@@ -0,0 +1,330 @@
import { describe, it, expect } from 'vitest'
import prodData from './fixtures/production-data.json'
import { KITS } from '../config/kits'
const oils = prodData.oils
// ---------------------------------------------------------------------------
// Replicate kit cost calculation logic from useKitCost.js (pure functions)
// ---------------------------------------------------------------------------
function resolveOilName(kitOilName) {
if (oils[kitOilName]) return kitOilName
const match = Object.keys(oils).find(n => n.endsWith(kitOilName) && n !== kitOilName)
return match || kitOilName
}
function calcKitPerDrop(kit) {
const resolved = kit.oils.map(resolveOilName)
const bc = kit.bottleCount || {}
let totalBottlePrice = 0
const oilBottlePrices = {}
for (const name of resolved) {
const meta = oils[name]
const count = bc[name] || 1
const bp = meta ? meta.bottlePrice * count : 0
oilBottlePrices[name] = bp
totalBottlePrice += bp
}
if (totalBottlePrice === 0) return {}
const totalValue = totalBottlePrice + (kit.accessoryValue || 0)
const discountRate = Math.min(kit.price / totalValue, 1)
const perDrop = {}
for (const name of resolved) {
const meta = oils[name]
const count = bc[name] || 1
const bp = oilBottlePrices[name]
const kitCostForOil = bp * discountRate
const totalDrops = meta ? meta.dropCount * count : 1
perDrop[name] = totalDrops > 0 ? kitCostForOil / totalDrops : 0
}
return perDrop
}
function canMakeRecipe(kit, recipe) {
const resolvedSet = new Set(kit.oils.map(resolveOilName))
return recipe.ingredients.every(ing => resolvedSet.has(ing.oil))
}
function calcRecipeCostWithKit(kitPerDrop, recipe) {
return recipe.ingredients.reduce((sum, ing) => {
return sum + (kitPerDrop[ing.oil] || 0) * ing.drops
}, 0)
}
function calcOriginalCost(recipe) {
return recipe.ingredients.reduce((sum, ing) => {
const meta = oils[ing.oil]
if (!meta || !meta.dropCount) return sum
return sum + (meta.bottlePrice / meta.dropCount) * ing.drops
}, 0)
}
// ---------------------------------------------------------------------------
// Kit Configuration
// ---------------------------------------------------------------------------
describe('Kit Configuration', () => {
it('has 4 kits defined', () => {
expect(KITS).toHaveLength(4)
})
it('each kit has required fields', () => {
for (const kit of KITS) {
expect(kit).toHaveProperty('id')
expect(kit).toHaveProperty('name')
expect(kit).toHaveProperty('price')
expect(kit).toHaveProperty('oils')
expect(kit.price).toBeGreaterThan(0)
expect(kit.oils.length).toBeGreaterThan(0)
}
})
it('all kits include coconut oil', () => {
for (const kit of KITS) {
expect(kit.oils).toContain('椰子油')
}
})
it('kit ids are unique', () => {
const ids = KITS.map(k => k.id)
expect(new Set(ids).size).toBe(ids.length)
})
it('芳香调理 has 9 oils at ¥1575', () => {
const kit = KITS.find(k => k.id === 'aroma')
expect(kit.price).toBe(1575)
expect(kit.oils).toHaveLength(9)
})
it('家庭医生 has 11 oils at ¥2250', () => {
const kit = KITS.find(k => k.id === 'family')
expect(kit.price).toBe(2250)
expect(kit.oils).toHaveLength(11)
})
it('居家呵护 has 23 oils at ¥3988', () => {
const kit = KITS.find(k => k.id === 'home3988')
expect(kit.price).toBe(3988)
expect(kit.oils).toHaveLength(23)
})
it('全精油 has 80 oils at ¥17700 with bottleCount for coconut oil', () => {
const kit = KITS.find(k => k.id === 'full')
expect(kit.price).toBe(17700)
expect(kit.oils.length).toBe(80)
expect(kit.bottleCount).toBeDefined()
expect(kit.bottleCount['椰子油']).toBeCloseTo(2.57, 2)
})
})
// ---------------------------------------------------------------------------
// Oil Name Resolution
// ---------------------------------------------------------------------------
describe('Oil Name Resolution', () => {
it('resolves exact match', () => {
expect(resolveOilName('薰衣草')).toBe('薰衣草')
expect(resolveOilName('乳香')).toBe('乳香')
})
it('resolves 牛至 to 西班牙牛至 via endsWith', () => {
expect(resolveOilName('牛至')).toBe('西班牙牛至')
})
it('does NOT resolve 牛至 to 牛至呵护', () => {
expect(resolveOilName('牛至')).not.toBe('牛至呵护')
})
it('returns original name for unknown oil', () => {
expect(resolveOilName('不存在的油')).toBe('不存在的油')
})
})
// ---------------------------------------------------------------------------
// Kit Cost Calculation
// ---------------------------------------------------------------------------
describe('Kit Cost Calculation', () => {
it('discount rate is always <= 1 (kit never more expensive than buying individually)', () => {
for (const kit of KITS) {
const perDrop = calcKitPerDrop(kit)
for (const [name, ppd] of Object.entries(perDrop)) {
const meta = oils[name]
if (!meta || !meta.dropCount) continue
const originalPpd = meta.bottlePrice / meta.dropCount
expect(ppd).toBeLessThanOrEqual(originalPpd + 0.001) // tiny float tolerance
}
}
})
it('家庭医生 discount is ~32-33%', () => {
const kit = KITS.find(k => k.id === 'family')
const bc = kit.bottleCount || {}
let totalBp = 0
for (const name of kit.oils) {
const resolved = resolveOilName(name)
const meta = oils[resolved]
totalBp += meta ? meta.bottlePrice * (bc[resolved] || 1) : 0
}
const totalValue = totalBp + (kit.accessoryValue || 0)
const discount = 1 - kit.price / totalValue
expect(discount).toBeGreaterThan(0.30)
expect(discount).toBeLessThan(0.40)
})
it('kit cost for a recipe is less than original cost', () => {
const kit = KITS.find(k => k.id === 'family')
const perDrop = calcKitPerDrop(kit)
// 灰指甲: 西班牙牛至 + 椰子油
const recipe = prodData.recipes.find(r => r.name === '灰指甲')
if (recipe && canMakeRecipe(kit, recipe)) {
const kitCost = calcRecipeCostWithKit(perDrop, recipe)
const origCost = calcOriginalCost(recipe)
expect(kitCost).toBeLessThan(origCost)
expect(kitCost).toBeGreaterThan(0)
}
})
it('全精油 bottleCount 2.57 makes coconut oil cheaper per drop than without multiplier', () => {
const fullKit = KITS.find(k => k.id === 'full')
const perDrop = calcKitPerDrop(fullKit)
// With 2.57 bottles, per-drop cost should be roughly 1/2.57 of single-bottle kit cost
// at the same discount rate. Just verify it's significantly less than original per-drop.
const origPpd = oils['椰子油'].bottlePrice / oils['椰子油'].dropCount
expect(perDrop['椰子油']).toBeLessThan(origPpd)
expect(perDrop['椰子油']).toBeGreaterThan(0)
})
it('accessoryValue reduces effective oil cost', () => {
const kit = KITS.find(k => k.id === 'family')
// Without accessory: rate = price / totalBp
// With accessory: rate = price / (totalBp + accessoryValue) < previous rate
expect(kit.accessoryValue).toBeGreaterThan(0)
const bc = kit.bottleCount || {}
let totalBp = 0
for (const name of kit.oils) {
const resolved = resolveOilName(name)
const meta = oils[resolved]
totalBp += meta ? meta.bottlePrice * (bc[resolved] || 1) : 0
}
const rateWithAcc = kit.price / (totalBp + kit.accessoryValue)
const rateWithoutAcc = kit.price / totalBp
expect(rateWithAcc).toBeLessThan(rateWithoutAcc)
})
})
// ---------------------------------------------------------------------------
// Recipe Matching
// ---------------------------------------------------------------------------
describe('Recipe Matching', () => {
it('larger kits can make at least as many recipes as smaller ones', () => {
const counts = KITS.map(kit => {
const matched = prodData.recipes.filter(r => canMakeRecipe(kit, r))
return { id: kit.id, count: matched.length, oilCount: kit.oils.length }
}).sort((a, b) => a.oilCount - b.oilCount)
for (let i = 1; i < counts.length; i++) {
expect(counts[i].count).toBeGreaterThanOrEqual(counts[i - 1].count)
}
})
it('全精油 can make the most recipes', () => {
const fullKit = KITS.find(k => k.id === 'full')
const fullCount = prodData.recipes.filter(r => canMakeRecipe(fullKit, r)).length
for (const kit of KITS) {
if (kit.id === 'full') continue
const count = prodData.recipes.filter(r => canMakeRecipe(kit, r)).length
expect(fullCount).toBeGreaterThanOrEqual(count)
}
})
it('灰指甲 (牛至+椰子油) can be made by 家庭医生', () => {
const kit = KITS.find(k => k.id === 'family')
const recipe = prodData.recipes.find(r => r.name === '灰指甲')
expect(canMakeRecipe(kit, recipe)).toBe(true)
})
it('recipe requiring 永久花 cannot be made by 芳香调理', () => {
const kit = KITS.find(k => k.id === 'aroma')
const recipe = prodData.recipes.find(r => r.name === '小v脸') // has 永久花
expect(canMakeRecipe(kit, recipe)).toBe(false)
})
})
// ---------------------------------------------------------------------------
// Accessory Values — PR32
// ---------------------------------------------------------------------------
describe('Accessory Values', () => {
it('芳香调理 has no accessories', () => {
const kit = KITS.find(k => k.id === 'aroma')
expect(kit.accessoryValue).toBeUndefined()
})
it('家庭医生 accessories = 475 (香薰机375 + 木盒100)', () => {
const kit = KITS.find(k => k.id === 'family')
expect(kit.accessoryValue).toBe(475)
})
it('居家呵护 accessories = 585 (香薰机375 + 竹木架210)', () => {
const kit = KITS.find(k => k.id === 'home3988')
expect(kit.accessoryValue).toBe(585)
})
it('全精油 accessories = 795 (香薰机375 + 竹木架210×2)', () => {
const kit = KITS.find(k => k.id === 'full')
expect(kit.accessoryValue).toBe(795)
})
})
// ---------------------------------------------------------------------------
// Discount Rate Calculation — PR32
// ---------------------------------------------------------------------------
describe('Discount Rate', () => {
function calcDiscountRate(kit) {
const resolved = kit.oils.map(resolveOilName)
const bc = kit.bottleCount || {}
let totalBP = 0
for (const name of resolved) {
const meta = oils[name]
totalBP += meta ? meta.bottlePrice * (bc[name] || 1) : 0
}
const totalValue = totalBP + (kit.accessoryValue || 0)
return totalValue > 0 ? Math.min(kit.price / totalValue, 1) : 1
}
it('all kits have discount rate < 1 (套装比单买便宜)', () => {
for (const kit of KITS) {
const rate = calcDiscountRate(kit)
expect(rate).toBeLessThan(1)
expect(rate).toBeGreaterThan(0)
}
})
it('全精油 discount rate ≈ 0.69', () => {
const kit = KITS.find(k => k.id === 'full')
const rate = calcDiscountRate(kit)
expect(rate).toBeGreaterThan(0.65)
expect(rate).toBeLessThan(0.75)
})
it('larger kits have better discounts', () => {
const rates = KITS.map(k => ({ id: k.id, rate: calcDiscountRate(k), count: k.oils.length }))
rates.sort((a, b) => a.count - b.count)
// Generally larger kits should have lower discount rate (better deal)
// At minimum, the largest kit should have a lower rate than the smallest
expect(rates[rates.length - 1].rate).toBeLessThanOrEqual(rates[0].rate)
})
it('kit cost per recipe should be less than original cost', () => {
for (const kit of KITS) {
const perDrop = calcKitPerDrop(kit)
const recipes = prodData.recipes.filter(r => canMakeRecipe(kit, r))
for (const r of recipes.slice(0, 5)) {
const kitCost = calcRecipeCostWithKit(perDrop, r)
const originalCost = calcOriginalCost(r)
expect(kitCost).toBeLessThanOrEqual(originalCost + 0.01)
}
}
})
})

View File

@@ -0,0 +1,87 @@
import { describe, it, expect } from 'vitest'
// Replicates the fixed fmtCostWithRetail logic: retail shown whenever any ingredient
// has a retail price stored (even when it equals the member price).
function fmtCostWithRetail(ingredients, oilsMeta) {
const cost = ingredients.reduce((s, i) => {
const m = oilsMeta[i.oil]
return s + (m ? (m.bottlePrice / m.dropCount) * i.drops : 0)
}, 0)
const retail = ingredients.reduce((s, i) => {
const m = oilsMeta[i.oil]
if (m && m.retailPrice && m.dropCount) return s + (m.retailPrice / m.dropCount) * i.drops
return s + (m ? (m.bottlePrice / m.dropCount) * i.drops : 0)
}, 0)
const anyRetail = ingredients.some(i => {
const m = oilsMeta[i.oil]
return m && m.retailPrice && m.dropCount
})
if (anyRetail && retail > 0) {
return { cost: '¥ ' + cost.toFixed(2), retail: '¥ ' + retail.toFixed(2), hasRetail: true }
}
return { cost: '¥ ' + cost.toFixed(2), retail: null, hasRetail: false }
}
describe('fmtCostWithRetail — retail price display', () => {
it('shows retail when retail > member', () => {
const meta = { '玫瑰': { bottlePrice: 100, retailPrice: 150, dropCount: 10 } }
const r = fmtCostWithRetail([{ oil: '玫瑰', drops: 5 }], meta)
expect(r.hasRetail).toBe(true)
expect(r.retail).toBe('¥ 75.00')
})
it('still shows retail when retail === member (regression: 带玫瑰护手霜 case)', () => {
const meta = { '玫瑰护手霜': { bottlePrice: 300, retailPrice: 300, dropCount: 50 } }
const r = fmtCostWithRetail([{ oil: '玫瑰护手霜', drops: 5 }], meta)
expect(r.hasRetail).toBe(true)
expect(r.cost).toBe('¥ 30.00')
expect(r.retail).toBe('¥ 30.00')
})
it('no retail when ingredient has no retail price', () => {
const meta = { '薰衣草': { bottlePrice: 100, retailPrice: null, dropCount: 10 } }
const r = fmtCostWithRetail([{ oil: '薰衣草', drops: 5 }], meta)
expect(r.hasRetail).toBe(false)
expect(r.retail).toBeNull()
})
})
// getEnglishName priority fix — DB en_name must beat static card override.
function getEnglishName(name, oilsMeta, cards, aliases, oilEnFn) {
const meta = oilsMeta[name]
if (meta?.enName) return meta.enName
if (cards[name]?.en) return cards[name].en
if (aliases[name] && cards[aliases[name]]?.en) return cards[aliases[name]].en
const base = name.replace(/呵护$/, '')
if (base !== name && cards[base]?.en) return cards[base].en
return oilEnFn ? oilEnFn(name) : ''
}
describe('getEnglishName — DB wins over static card', () => {
const cards = {
'温柔呵护': { en: 'Soft Talk' },
'椒样薄荷': { en: 'Peppermint' },
'西班牙牛至': { en: 'Oregano' },
}
const aliases = { '仕女呵护': '温柔呵护', '薄荷呵护': '椒样薄荷', '牛至呵护': '西班牙牛至' }
it('uses DB en_name over static card en (温柔呵护 regression)', () => {
const meta = { '温柔呵护': { enName: 'Clary Calm' } }
expect(getEnglishName('温柔呵护', meta, cards, aliases)).toBe('Clary Calm')
})
it('uses DB en_name over aliased card en (仕女呵护 regression)', () => {
const meta = { '仕女呵护': { enName: 'Soft Talk Touch' } }
expect(getEnglishName('仕女呵护', meta, cards, aliases)).toBe('Soft Talk Touch')
})
it('falls back to static card when DB en_name is empty', () => {
const meta = { '温柔呵护': { enName: '' } }
expect(getEnglishName('温柔呵护', meta, cards, aliases)).toBe('Soft Talk')
})
it('alias still works as fallback', () => {
const meta = { '牛至呵护': {} }
expect(getEnglishName('牛至呵护', meta, cards, aliases)).toBe('Oregano')
})
})

View File

@@ -0,0 +1,73 @@
import { describe, it, expect } from 'vitest'
import { parseOilProductPaste } from '../composables/useOilProductPaste'
describe('parseOilProductPaste', () => {
it('returns empty shape for empty input', () => {
const r = parseOilProductPaste('')
expect(r.cn).toBe('')
expect(r.en).toBe('')
expect(r.memberPrice).toBeNull()
expect(r.retailPrice).toBeNull()
})
it('parses the 花样年华 sample as product with 100ml', () => {
const sample = `优惠顾客价:¥310PT:41
零售价:¥465
点数:37 规格:100毫升
花样年华焕颜精华水 Salubelle Rejuvenating Essence`
const r = parseOilProductPaste(sample)
expect(r.type).toBe('product')
expect(r.memberPrice).toBe(310)
expect(r.retailPrice).toBe(465)
expect(r.productAmount).toBe(100)
expect(r.productUnit).toBe('ml')
expect(r.cn).toBe('花样年华焕颜精华水')
expect(r.en).toBe('Salubelle Rejuvenating Essence')
})
it('detects essential oil when volume is standard ml', () => {
const sample = `会员价:¥200\n零售价:¥267\n规格:15毫升\n薰衣草 Lavender`
const r = parseOilProductPaste(sample)
expect(r.type).toBe('oil')
expect(r.volume).toBe('15')
expect(r.cn).toBe('薰衣草')
expect(r.en).toBe('Lavender')
})
it('handles half-width colon and dollar variant', () => {
const r = parseOilProductPaste('优惠顾客价: ¥99\n零售价: ¥150\n规格: 5ml\n柠檬 Lemon')
expect(r.memberPrice).toBe(99)
expect(r.retailPrice).toBe(150)
expect(r.type).toBe('oil')
expect(r.volume).toBe('5')
})
it('parses capsule spec as product', () => {
const r = parseOilProductPaste('优惠顾客价:¥200\n规格:60粒\n深海鱼油 Omega')
expect(r.type).toBe('product')
expect(r.productAmount).toBe(60)
expect(r.productUnit).toBe('capsule')
})
it('parses gram spec as product', () => {
const r = parseOilProductPaste('优惠顾客价:¥80\n规格:120克\n洁面乳 Face Wash')
expect(r.productUnit).toBe('g')
expect(r.productAmount).toBe(120)
})
it('non-standard ml volume falls to product', () => {
const r = parseOilProductPaste('优惠顾客价:¥310\n规格:100毫升\n精华 Essence')
expect(r.type).toBe('product')
expect(r.productAmount).toBe(100)
expect(r.productUnit).toBe('ml')
})
it('name without english part keeps cn only', () => {
const r = parseOilProductPaste('优惠顾客价:¥50\n规格:5毫升\n某国产品')
expect(r.cn).toBe('某国产品')
expect(r.en).toBe('')
})
})

View File

@@ -499,3 +499,149 @@ describe('volume field in recipe mapping — PR31', () => {
expect(labels['']).toBe('')
})
})
// ---------------------------------------------------------------------------
// PR33: Oil card branding logic
// ---------------------------------------------------------------------------
describe('oil card branding — PR33', () => {
it('brand data determines card display elements', () => {
const brand = { qr_code: 'data:image/png;base64,abc', brand_bg: 'data:image/png;base64,bg', brand_logo: null, brand_name: '测试品牌', brand_align: 'center' }
expect(!!brand.qr_code).toBe(true)
expect(!!brand.brand_bg).toBe(true)
expect(!!brand.brand_logo).toBe(false)
expect(!!brand.brand_name).toBe(true)
})
it('empty brand shows plain card', () => {
const brand = {}
expect(!!brand.qr_code).toBe(false)
expect(!!brand.brand_bg).toBe(false)
expect(!!brand.brand_logo).toBe(false)
})
it('volumeLabel with name parameter works for drops and ml', () => {
// Simulates the fix: volumeLabel(dropCount, name) needs both params
const DROPS_TO_VOLUME = { 93: '5ml', 280: '15ml' }
function volumeLabel(dropCount, name) {
if (name === '无香乳液') return dropCount + 'ml' // ml unit
return DROPS_TO_VOLUME[dropCount] || (dropCount + '滴')
}
expect(volumeLabel(280, '薰衣草')).toBe('15ml')
expect(volumeLabel(200, '无香乳液')).toBe('200ml')
expect(volumeLabel(93, '茶树')).toBe('5ml')
})
it('PDF export price unit adapts to product type', () => {
function oilPriceUnit(name) {
if (name === '无香乳液') return 'ml'
if (name === '植物空胶囊') return '颗'
return '滴'
}
expect(oilPriceUnit('薰衣草')).toBe('滴')
expect(oilPriceUnit('无香乳液')).toBe('ml')
expect(oilPriceUnit('植物空胶囊')).toBe('颗')
})
})
// ---------------------------------------------------------------------------
// PR34: Product edit UI — unit-based form switching
// ---------------------------------------------------------------------------
describe('product edit UI logic — PR34', () => {
it('drop unit shows standard volume selector', () => {
const unit = 'drop'
expect(unit === 'drop').toBe(true)
})
it('non-drop unit shows amount + unit selector', () => {
for (const u of ['ml', 'g', 'capsule']) {
expect(u !== 'drop').toBe(true)
}
})
it('edit form initializes correct unit from meta', () => {
const meta = { unit: 'g', dropCount: 80 }
const editUnit = meta.unit || 'drop'
const editProductAmount = editUnit !== 'drop' ? meta.dropCount : null
const editProductUnit = editUnit !== 'drop' ? editUnit : 'ml'
expect(editUnit).toBe('g')
expect(editProductAmount).toBe(80)
expect(editProductUnit).toBe('g')
})
it('edit form defaults to drop for oils', () => {
const meta = { unit: 'drop', dropCount: 280 }
const editUnit = meta.unit || 'drop'
expect(editUnit).toBe('drop')
})
it('edit form defaults to drop when unit is undefined', () => {
const meta = { dropCount: 280 }
const editUnit = meta.unit || 'drop'
expect(editUnit).toBe('drop')
})
it('save uses product amount and unit for non-drop', () => {
const editUnit = 'ml'
const editProductAmount = 200
const editProductUnit = 'ml'
const dropCount = 280 // from standard volume selector
const finalDropCount = editUnit !== 'drop' ? editProductAmount : dropCount
const finalUnit = editUnit !== 'drop' ? editProductUnit : null
expect(finalDropCount).toBe(200)
expect(finalUnit).toBe('ml')
})
it('save uses standard drop count for oils', () => {
const editUnit = 'drop'
const editProductAmount = null
const dropCount = 280
const finalDropCount = editUnit !== 'drop' ? editProductAmount : dropCount
const finalUnit = editUnit !== 'drop' ? 'ml' : null
expect(finalDropCount).toBe(280)
expect(finalUnit).toBeNull()
})
it('label adapts: 精油名称 for oils, 产品名称 for products', () => {
const labelForDrop = 'drop' === 'drop' ? '精油名称' : '产品名称'
const labelForMl = 'ml' === 'drop' ? '精油名称' : '产品名称'
expect(labelForDrop).toBe('精油名称')
expect(labelForMl).toBe('产品名称')
})
})
// ---------------------------------------------------------------------------
// PR34: Share text and consumption analysis use dynamic unit
// ---------------------------------------------------------------------------
describe('share text and consumption use dynamic unit — PR34', () => {
const UNIT_MAP = { drop: '滴', ml: 'ml', g: 'g', capsule: '颗' }
function unitLabel(name, unitMap) { return UNIT_MAP[unitMap[name] || 'drop'] }
it('share text uses unitLabel for each ingredient', () => {
const units = { '薰衣草': 'drop', '无香乳液': 'ml', '植物空胶囊': 'capsule' }
const ings = [
{ oil: '薰衣草', drops: 3 },
{ oil: '无香乳液', drops: 30 },
{ oil: '植物空胶囊', drops: 2 },
]
const lines = ings.map(i => `${i.oil} ${i.drops}${unitLabel(i.oil, units)}`)
expect(lines[0]).toBe('薰衣草 3滴')
expect(lines[1]).toBe('无香乳液 30ml')
expect(lines[2]).toBe('植物空胶囊 2颗')
})
it('consumption analysis uses unitLabel per oil', () => {
const units = { '薰衣草': 'drop', '活力磨砂膏': 'g' }
const data = [
{ oil: '薰衣草', drops: 15, bottleDrops: 280 },
{ oil: '活力磨砂膏', drops: 30, bottleDrops: 70 },
]
const display = data.map(c => ({
usage: `${c.drops}${unitLabel(c.oil, units)}`,
capacity: `${c.bottleDrops}${unitLabel(c.oil, units)}`,
}))
expect(display[0].usage).toBe('15滴')
expect(display[0].capacity).toBe('280滴')
expect(display[1].usage).toBe('30g')
expect(display[1].capacity).toBe('70g')
})
})

View File

@@ -0,0 +1,57 @@
import { describe, it, expect } from 'vitest'
// Mirrors the exactResults matching rule in RecipeSearch.vue
function matches(recipes, q, oilEn = (s) => '') {
const query = (q || '').trim().toLowerCase()
if (!query) return []
const isEn = /^[a-zA-Z\s]+$/.test(query)
return recipes.filter(r => {
if (r.tags && r.tags.includes('已下架')) return false
const nameMatch = r.name.toLowerCase().includes(query)
const enNameMatch = isEn && (r.en_name || '').toLowerCase().includes(query)
const oilEnMatch = isEn && (r.ingredients || []).some(ing => (oilEn(ing.oil) || '').toLowerCase().includes(query))
const oilZhMatch = query.length >= 2 && (r.ingredients || []).some(ing => ing.oil.toLowerCase().includes(query))
const tagMatch = (r.tags || []).some(t => t.toLowerCase().includes(query))
return nameMatch || enNameMatch || oilEnMatch || oilZhMatch || tagMatch
})
}
const recipes = [
{ name: '助眠晚安', tags: [], ingredients: [{ oil: '薰衣草', drops: 3 }, { oil: '乳香', drops: 2 }] },
{ name: '提神醒脑', tags: [], ingredients: [{ oil: '椒样薄荷', drops: 2 }, { oil: '柠檬', drops: 3 }] },
{ name: '肩颈舒缓', tags: ['舒缓'], ingredients: [{ oil: '西班牙牛至', drops: 1 }, { oil: '椰子油', drops: 10 }] },
{ name: '感冒护理', tags: [], ingredients: [{ oil: '牛至呵护', drops: 2 }] },
{ name: '下架配方', tags: ['已下架'], ingredients: [{ oil: '薰衣草', drops: 1 }] },
]
describe('Recipe search by oil name', () => {
it('finds recipes containing the oil (Chinese exact)', () => {
const r = matches(recipes, '薰衣草')
expect(r.map(x => x.name)).toEqual(['助眠晚安'])
})
it('finds multiple recipes for a common oil', () => {
expect(matches(recipes, '牛至').map(x => x.name).sort()).toEqual(['感冒护理', '肩颈舒缓'])
})
it('excludes 已下架 recipes', () => {
const r = matches(recipes, '薰衣草')
expect(r.some(x => x.name === '下架配方')).toBe(false)
})
it('single-char query does not match oil names (avoids noise)', () => {
const r = matches(recipes, '草')
expect(r).toEqual([])
})
it('still matches recipe name for short queries', () => {
const r = matches(recipes, '晚')
expect(r.map(x => x.name)).toEqual(['助眠晚安'])
})
it('matches english oil name when query is english', () => {
const oilEn = (o) => ({ '薰衣草': 'Lavender', '乳香': 'Frankincense' }[o] || '')
const r = matches(recipes, 'Lavender', oilEn)
expect(r.map(x => x.name)).toEqual(['助眠晚安'])
})
})

View File

@@ -66,7 +66,7 @@
<span class="ec-oil-name">{{ getCardOilName(ing.oil) }}</span>
<span class="ec-drops">{{ ing.drops }} {{ oilsStore.unitLabelPlural(ing.oil, ing.drops, cardLang) }}</span>
<span class="ec-cost">{{ oilsStore.fmtPrice(oilsStore.pricePerDrop(ing.oil) * ing.drops) }}</span>
<span v-if="cardHasAnyRetail" class="ec-retail">{{ hasRetailForOil(ing.oil) && retailPerDrop(ing.oil) > oilsStore.pricePerDrop(ing.oil) ? oilsStore.fmtPrice(retailPerDrop(ing.oil) * ing.drops) : '' }}</span>
<span v-if="cardHasAnyRetail" class="ec-retail">{{ hasRetailForOil(ing.oil) ? oilsStore.fmtPrice(retailPerDrop(ing.oil) * ing.drops) : '' }}</span>
</li>
</ul>
@@ -483,7 +483,7 @@ function copyText() {
const ings = cardIngredients.value
const lines = ings.map(ing => {
const cost = oilsStore.pricePerDrop(ing.oil) * ing.drops
return `${ing.oil} ${ing.drops} ${oilsStore.fmtPrice(cost)}`
return `${ing.oil} ${ing.drops}${oilsStore.unitLabel(ing.oil)} ${oilsStore.fmtPrice(cost)}`
})
const total = priceInfo.value.cost
const text = [
@@ -591,7 +591,7 @@ function getCardRecipeName() {
}
const cardHasAnyRetail = computed(() =>
cardIngredients.value.some(ing => hasRetailForOil(ing.oil) && retailPerDrop(ing.oil) > oilsStore.pricePerDrop(ing.oil))
cardIngredients.value.some(ing => hasRetailForOil(ing.oil))
)
const cardTitleSize = computed(() => {
@@ -829,7 +829,7 @@ onMounted(() => {
else { selectedVolume.value = 'custom'; customVolumeValue.value = Math.round(ml) }
loadBrand()
nextTick(() => generateCardImage())
// Don't auto-generate card image on mount — generate on demand when saving
})
function addIngredient() {
@@ -1010,6 +1010,7 @@ async function saveRecipe() {
note: editNote.value.trim(),
tags: editTags.value,
ingredients: allIngs,
volume: selectedVolume.value || '',
}
await recipesStore.saveRecipe(payload)
// Reload recipes so the data is fresh when re-opened
@@ -1698,8 +1699,8 @@ async function saveRecipe() {
}
.editor-drops {
width: 42px;
padding: 5px 2px;
width: 58px;
padding: 5px 4px 5px 6px;
border: 1.5px solid var(--border, #e0d4c0);
border-radius: 8px;
font-size: 13px;

View File

@@ -0,0 +1,173 @@
import { computed } from 'vue'
import { useOilsStore } from '../stores/oils'
import { useRecipesStore } from '../stores/recipes'
import { KITS } from '../config/kits'
/**
* 套装成本分摊与配方匹配
*
* 分摊逻辑:按各精油原瓶价占比分摊套装总价
* 某精油套装内成本 = (该油原瓶价 / 套装内所有油原瓶价之和) × 套装总价
* 套装内每滴成本 = 套装内成本 / 该油滴数
*/
export function useKitCost() {
const oils = useOilsStore()
const recipeStore = useRecipesStore()
// Resolve kit oil name to system oil name (handles 牛至→西班牙牛至 etc.)
function resolveOilName(kitOilName) {
if (oils.oilsMeta[kitOilName]) return kitOilName
// Try finding system oil that ends with kit name
return oils.oilNames.find(n => n.endsWith(kitOilName) && n !== kitOilName) || kitOilName
}
// Calculate per-drop costs for a kit
function calcKitPerDrop(kit) {
const resolved = kit.oils.map(name => resolveOilName(name))
const bc = kit.bottleCount || {} // e.g. { '椰子油': 2.57 }
// Sum of bottle prices for all oils in kit (accounting for multiple bottles)
let totalBottlePrice = 0
const oilBottlePrices = {}
for (const name of resolved) {
const meta = oils.oilsMeta[name]
const count = bc[name] || 1
const bp = meta ? meta.bottlePrice * count : 0
oilBottlePrices[name] = bp
totalBottlePrice += bp
}
if (totalBottlePrice === 0) return {}
// Uniform discount: kit price covers oils + accessories at the same discount rate
// discount_rate = kit_price / (oil_total + accessory_value)
// each oil's kit cost = bottle_price × discount_rate
const totalValue = totalBottlePrice + (kit.accessoryValue || 0)
const discountRate = Math.min(kit.price / totalValue, 1) // cap at 1 (no markup)
const perDrop = {}
for (const name of resolved) {
const meta = oils.oilsMeta[name]
const count = bc[name] || 1
const bp = oilBottlePrices[name]
const kitCostForOil = bp * discountRate
const totalDrops = meta ? meta.dropCount * count : 1
perDrop[name] = totalDrops > 0 ? kitCostForOil / totalDrops : 0
}
return perDrop
}
// Check if a recipe can be made with a kit
function canMakeRecipe(kit, recipe) {
const resolvedSet = new Set(kit.oils.map(name => resolveOilName(name)))
return recipe.ingredients.every(ing => resolvedSet.has(ing.oil))
}
// Calculate recipe cost using kit pricing
function calcRecipeCostWithKit(kitPerDrop, recipe) {
return recipe.ingredients.reduce((sum, ing) => {
const ppd = kitPerDrop[ing.oil] || 0
return sum + ppd * ing.drops
}, 0)
}
// Get all matching recipes for a kit
function getKitRecipes(kit) {
const perDrop = calcKitPerDrop(kit)
return recipeStore.recipes
.filter(r => canMakeRecipe(kit, r))
.map(r => ({
...r,
kitCost: calcRecipeCostWithKit(perDrop, r),
originalCost: oils.calcCost(r.ingredients),
}))
.sort((a, b) => a.name.localeCompare(b.name, 'zh'))
}
// Build full analysis for all kits, sorted by recipe count ascending (fewest recipes first)
const kitAnalysis = computed(() => {
return KITS.map(kit => {
const perDrop = calcKitPerDrop(kit)
const recipes = getKitRecipes(kit)
// Calculate discount rate for display
const resolved = kit.oils.map(name => resolveOilName(name))
const bc = kit.bottleCount || {}
let totalBP = 0
for (const name of resolved) {
const meta = oils.oilsMeta[name]
totalBP += meta ? meta.bottlePrice * (bc[name] || 1) : 0
}
const totalValue = totalBP + (kit.accessoryValue || 0)
const discountRate = totalValue > 0 ? Math.min(kit.price / totalValue, 1) : 1
return {
...kit,
perDrop,
recipes,
recipeCount: recipes.length,
discountRate,
}
}).sort((a, b) => a.recipeCount - b.recipeCount)
})
// Cross-kit comparison: membership-tier style
// Columns: kits ordered by recipe count (fewest→most, from kitAnalysis)
// Rows: recipes available to most kits at top, exclusive recipes at bottom (staircase pattern)
const crossComparison = computed(() => {
const analysis = kitAnalysis.value
// Kit order for staircase: index in sorted analysis (0 = smallest kit)
const kitOrder = analysis.map(ka => ka.id)
const allRecipeIds = new Set()
for (const ka of analysis) {
for (const r of ka.recipes) allRecipeIds.add(r._id)
}
const rows = []
for (const id of allRecipeIds) {
let recipe = null
const costs = {}
let availCount = 0
// Track which kit columns have this recipe (by index in sorted order)
let smallestKitIdx = kitOrder.length
for (let i = 0; i < analysis.length; i++) {
const ka = analysis[i]
const found = ka.recipes.find(r => r._id === id)
if (found) {
if (!recipe) recipe = found
costs[ka.id] = found.kitCost
availCount++
if (i < smallestKitIdx) smallestKitIdx = i
} else {
costs[ka.id] = null
}
}
rows.push({
id,
name: recipe.name,
tags: recipe.tags,
volume: recipe.volume,
ingredients: recipe.ingredients,
originalCost: recipe.originalCost,
costs,
availCount,
smallestKitIdx,
})
}
// Staircase sort: most available first, then by smallest kit that has it, then by name
rows.sort((a, b) => {
if (a.availCount !== b.availCount) return b.availCount - a.availCount
if (a.smallestKitIdx !== b.smallestKitIdx) return a.smallestKitIdx - b.smallestKitIdx
return a.name.localeCompare(b.name, 'zh')
})
return rows
})
return {
KITS,
resolveOilName,
calcKitPerDrop,
canMakeRecipe,
calcRecipeCostWithKit,
getKitRecipes,
kitAnalysis,
crossComparison,
}
}

View File

@@ -1,5 +1,6 @@
// Oil knowledge cards - usage guides for common essential oils
// Ported from original vanilla JS implementation
import { useOilsStore } from '../stores/oils'
export const OIL_CARDS = {
'野橘': { emoji: '🍊', en: 'Wild Orange', effects: '安抚镇静、驱散负能量、紧张情绪和压力\n抗氧化、消炎、抗病毒、增强免疫\n提振食欲刺激胆汁分泌促进消化\n促进循环', usage: '日常香薰,提振情绪、舒压,营造愉悦氛围\n饮水加入 1至2 滴,护肝抗病毒\n泡澡时滴 3至4 滴,消除疲劳,放松身心\n扫地、拖地时加入 3至5 滴,清新空气\n涂抹腹部促进消化\n涂抹肝区帮助肝脏排毒\n口腔溃疡时加入水中漱口', method: '🔹香薰 🔸内用 🔺涂抹', caution: '轻微光敏,白天涂抹注意防晒' },
@@ -33,6 +34,17 @@ export const OIL_CARD_ALIAS = {
}
export function getOilCard(name) {
// Check DB-persisted cards first (via store)
try {
const store = useOilsStore()
if (store.oilCards[name]) return store.oilCards[name]
// Check alias in store too
const aliased = OIL_CARD_ALIAS[name]
if (aliased && store.oilCards[aliased]) return store.oilCards[aliased]
} catch {
// Store may not be initialized yet, fall through to static data
}
// Fall back to static OIL_CARDS
if (OIL_CARDS[name]) return OIL_CARDS[name]
if (OIL_CARD_ALIAS[name] && OIL_CARDS[OIL_CARD_ALIAS[name]]) return OIL_CARDS[OIL_CARD_ALIAS[name]]
const base = name.replace(/呵护$/, '')

View File

@@ -0,0 +1,52 @@
const OIL_VOLUMES = new Set(['2.5', '5', '10', '15', '115'])
export function parseOilProductPaste(raw) {
const result = {
type: 'product',
cn: '',
en: '',
memberPrice: null,
retailPrice: null,
volume: null,
customDrops: null,
productAmount: null,
productUnit: null,
}
if (!raw || !raw.trim()) return result
const text = raw.replace(/[:]/g, ':').replace(/[¥¥]/g, '')
const memberMatch = text.match(/(?:优惠顾客价|会员价|批发价)\s*:?\s*(\d+(?:\.\d+)?)/)
const retailMatch = text.match(/零售价\s*:?\s*(\d+(?:\.\d+)?)/)
const specMatch = text.match(/规格\s*:?\s*(\d+(?:\.\d+)?)\s*(毫升|ml|ML|克|g|G|颗|粒|片)/)
if (memberMatch) result.memberPrice = Number(memberMatch[1])
if (retailMatch) result.retailPrice = Number(retailMatch[1])
for (const line of raw.split(/\r?\n/)) {
const s = line.trim()
if (!s) continue
if (/优惠顾客价|会员价|零售价|点数|规格|PT\s*:|批发价/i.test(s)) continue
const m = s.match(/^([^A-Za-z]+?)\s+([A-Za-z].*)$/)
if (m) { result.cn = m[1].trim(); result.en = m[2].trim() } else { result.cn = s }
break
}
if (specMatch) {
const amount = specMatch[1]
const unitRaw = specMatch[2].toLowerCase()
const isMl = unitRaw === '毫升' || unitRaw === 'ml'
if (isMl && OIL_VOLUMES.has(String(Number(amount)))) {
result.type = 'oil'
result.volume = String(Number(amount))
} else {
result.type = 'product'
result.productAmount = Number(amount)
result.productUnit = (unitRaw === '克' || unitRaw === 'g') ? 'g'
: (unitRaw === '颗' || unitRaw === '粒' || unitRaw === '片') ? 'capsule'
: 'ml'
}
}
return result
}

View File

@@ -0,0 +1,64 @@
// doTERRA 套装配置
// 价格和内容更新频率低,手动维护即可
// bottleCount: 某种油在套装中的瓶数用于成本分摊和可做次数计算默认1
// accessoryValue: 配件价值(香薰机、木盒等),从套装价中扣除后再分摊到精油
export const KITS = [
{
id: 'aroma',
name: '芳香调理套装',
price: 1575,
oils: [
'茶树', '野橘', '椒样薄荷', '薰衣草',
'芳香调理', '安定情绪', '保卫', '舒缓',
'椰子油',
],
},
{
id: 'family',
name: '家庭医生套装',
price: 2250,
accessoryValue: 475, // 香薰机375 + 木盒100
oils: [
'乳香', '茶树', '薰衣草', '柠檬', '椒样薄荷', '西班牙牛至',
'乐活', '舒缓', '保卫', '顺畅呼吸',
'椰子油',
],
},
{
id: 'home3988',
name: '居家呵护套装',
price: 3988,
accessoryValue: 585, // 香薰机375 + 旋转竹木精油架210
oils: [
'乳香', '野橘', '柠檬', '薰衣草', '椒样薄荷', '冬青', '茶树', '生姜', '柠檬草', '西班牙牛至',
'西洋蓍草', '乐活', '保卫', '新瑞活力', '舒缓', '安定情绪', '安宁神气', '顺畅呼吸', '柑橘清新', '芳香调理', '元气焕能',
'仕女呵护',
'椰子油',
],
},
{
id: 'full',
name: '全精油套装',
price: 17700,
oils: [
'侧柏', '乳香', '雪松', '芫荽', '芫荽叶', '丝柏', '圆柚', '红橘', '冬青', '没药', '扁柏', '檀香', '姜黄', '玫瑰',
'绿薄荷', '薰衣草', '永久花', '香蜂草', '迷迭香', '麦卢卡', '天竺葵', '蓝艾菊', '小茴香',
'古巴香脂', '依兰依兰', '丁香花蕾', '柠檬尤加利', '广藿香',
'罗勒', '莱姆', '生姜', '柠檬', '茶树', '野橘', '香茅', '枫香',
'芹菜籽', '岩兰草', '苦橙叶', '柠檬草', '山鸡椒', '黑云杉',
'马郁兰', '佛手柑', '黑胡椒', '小豆蔻', '尤加利', '百里香',
'椒样薄荷', '杜松浆果', '加州胡椒', '罗马洋甘菊', '道格拉斯冷杉', '西班牙鼠尾草',
'快乐鼠尾草', '西伯利亚冷杉',
'西班牙牛至', '斯里兰卡肉桂皮',
// 复配精油
'完美修护', '西洋蓍草', '花样年华焕肤油', '元气焕能', '舒缓',
'保卫', '乐释', '乐活', '愈创木', '椰风香草', '清醇薄荷',
'柑橘绚烂', '新瑞活力', '安宁神气', '芳香调理', '安定情绪',
'柑橘清新', '顺畅呼吸', '净化清新', '赋活呼吸', '天然防护',
'椰子油',
],
accessoryValue: 795, // 香薰机375 + 旋转竹木精油架210×2
// 115mL×1 + 30mL×6 = 295mL, 标准瓶115mL, 约2.57瓶
bottleCount: { '椰子油': 2.57 },
},
]

View File

@@ -29,6 +29,12 @@ const routes = [
component: () => import('../views/Projects.vue'),
meta: { requiresAuth: true },
},
{
path: '/kit-export',
name: 'KitExport',
component: () => import('../views/KitExport.vue'),
meta: { requiresAuth: true },
},
{
path: '/mydiary',
name: 'MyDiary',

View File

@@ -16,6 +16,7 @@ export const VOLUME_DROPS = {
export const useOilsStore = defineStore('oils', () => {
const oils = ref({})
const oilsMeta = ref({})
const oilCards = ref({})
// Getters
const oilNames = computed(() =>
@@ -50,7 +51,11 @@ export const useOilsStore = defineStore('oils', () => {
const cost = calcCost(ingredients)
const retail = calcRetailCost(ingredients)
const costStr = fmtPrice(cost)
if (retail > cost) {
const anyRetail = ingredients.some(i => {
const m = oilsMeta.value[i.oil]
return m && m.retailPrice && m.dropCount
})
if (anyRetail && retail > 0) {
return { cost: costStr, retail: fmtPrice(retail), hasRetail: true }
}
return { cost: costStr, retail: null, hasRetail: false }
@@ -75,9 +80,28 @@ export const useOilsStore = defineStore('oils', () => {
}
oils.value = newOils
oilsMeta.value = newMeta
// Also fetch oil cards from DB
try {
const cards = await api.get('/api/oil-cards')
const newCards = {}
for (const card of cards) {
newCards[card.name] = {
emoji: card.emoji || '',
en: card.en || '',
effects: card.effects || '',
usage: card.usage || '',
method: card.method || '',
caution: card.caution || '',
}
}
oilCards.value = newCards
} catch {
// oil_cards table may not exist yet on older backends
}
}
async function saveOil(name, bottlePrice, dropCount, retailPrice, enName = null, unit = null) {
async function saveOil(name, bottlePrice, dropCount, retailPrice, enName = null, unit = null, card = null) {
const payload = {
name,
bottle_price: bottlePrice,
@@ -86,6 +110,13 @@ export const useOilsStore = defineStore('oils', () => {
en_name: enName,
}
if (unit) payload.unit = unit
if (card) {
payload.card_emoji = card.emoji ?? null
payload.card_effects = card.effects ?? null
payload.card_usage = card.usage ?? null
payload.card_method = card.method ?? null
payload.card_caution = card.caution ?? null
}
await api.post('/api/oils', payload)
await loadOils()
}
@@ -134,6 +165,7 @@ export const useOilsStore = defineStore('oils', () => {
return {
oils,
oilsMeta,
oilCards,
oilNames,
pricePerDrop,
calcCost,

View File

@@ -101,6 +101,7 @@ import { useOilsStore } from '../stores/oils'
import { useRecipesStore } from '../stores/recipes'
import { useUiStore } from '../stores/ui'
import { api } from '../composables/useApi'
import { KITS as KIT_LIST } from '../config/kits'
const auth = useAuthStore()
const oils = useOilsStore()
@@ -120,30 +121,17 @@ const searchResults = computed(() => {
return oils.oilNames.filter(n => n.toLowerCase().includes(q)).slice(0, 15)
})
// Kit definitions
const KITS = {
family: ['乳香', '茶树', '薰衣草', '柠檬', '椒样薄荷', '保卫', '牛至', '乐活', '顺畅呼吸', '舒缓'],
home3988: ['乳香', '野橘', '柠檬', '薰衣草', '椒样薄荷', '冬青', '茶树', '生姜', '柠檬草', '西班牙牛至',
'西洋蓍草石榴籽', '乐活', '保卫', '新瑞活力', '舒缓', '安定情绪', '安宁神气', '顺畅呼吸', '柑橘清新', '芳香调理', '元气焕能'],
aroma: ['薰衣草', '舒缓', '安定情绪', '芳香调理', '野橘', '椒样薄荷', '保卫', '茶树'],
full: ['侧柏', '乳香', '雪松', '芫荽', '丝柏', '圆柚', '红橘', '冬青', '没药', '扁柏', '檀香', '姜黄', '玫瑰',
'绿薄荷', '薰衣草', '永久花', '香蜂草', '迷迭香', '麦卢卡', '天竺葵', '蓝艾菊', '小茴香',
'古巴香脂', '依兰依兰', '丁香花蕾', '柠檬尤加利', '藿香', '西班牙牛至尾草',
'罗勒', '莱姆', '生姜', '柠檬', '茶树', '野橘', '香茅', '枫香',
'芹菜籽', '岩兰草', '苦橙叶', '柠檬草', '山鸡椒', '黑云杉',
'马郁兰', '佛手柑', '黑胡椒', '小豆蔻', '尤加利', '百里香',
'椒样薄荷', '杜松浆果', '加州白鼠尾草',
'快乐鼠尾草', '西伯利亚冷杉',
'西班牙牛至', '斯里兰卡肉桂']
}
// Kit definitions from shared config
const KITS = Object.fromEntries(KIT_LIST.map(k => [k.id, k.oils]))
function addKit(kitName) {
const kit = KITS[kitName]
if (!kit) return
let added = 0
for (const name of kit) {
// Match existing oil names (fuzzy)
const match = oils.oilNames.find(n => n === name) || oils.oilNames.find(n => n.includes(name) || name.includes(n))
// Match existing oil names: exact first, then oil name ending with kit name (西班牙牛至 matches 牛至, but 牛至呵护 does not)
const match = oils.oilNames.find(n => n === name)
|| oils.oilNames.find(n => n.endsWith(name) && n !== name)
if (match && !ownedOils.value.includes(match)) {
ownedOils.value.push(match)
added++

View File

@@ -0,0 +1,507 @@
<template>
<div class="kit-export-page">
<div class="toolbar-sticky">
<div class="toolbar-inner">
<button class="btn-back" @click="$router.push('/projects')">&larr; 返回</button>
<h3 class="page-title">套装方案对比</h3>
<div class="toolbar-actions">
<button class="btn-outline btn-sm" @click="exportExcel('full')">导出完整版</button>
<button class="btn-outline btn-sm" @click="exportExcel('simple')">导出简版</button>
</div>
</div>
</div>
<!-- Kit Summary Cards -->
<div class="kit-cards">
<div
v-for="ka in kitAnalysis"
:key="ka.id"
class="kit-card"
:class="{ active: activeKit === ka.id }"
@click="activeKit = ka.id"
>
<div class="kit-name">{{ ka.name }}</div>
<div class="kit-price">¥{{ ka.price }}</div>
<div class="kit-discount">会员价后再{{ (ka.discountRate * 10).toFixed(1) }}</div>
<div class="kit-stats">
<span>{{ ka.oils.length }} 种精油</span>
<span class="kit-recipe-count">可做 {{ ka.recipeCount }} 个配方</span>
</div>
</div>
</div>
<!-- Active Kit Detail -->
<div v-if="activeKitData" class="kit-detail">
<div class="detail-header">
<h4>{{ activeKitData.name }} 可做配方 ({{ activeKitData.recipeCount }})</h4>
</div>
<div v-if="activeKitData.recipes.length === 0" class="empty-hint">该套装暂无完全匹配的配方</div>
<table v-else class="recipe-table">
<thead>
<tr>
<th class="th-name">配方名</th>
<th class="th-times">可做次数</th>
<th class="th-cost">套装成本</th>
<th class="th-cost">单买成本</th>
<th class="th-price">售价</th>
<th class="th-profit">利润率</th>
</tr>
</thead>
<tbody>
<tr v-for="r in activeKitData.recipes" :key="r._id">
<td class="td-name">{{ r.name }} <span v-if="volumeLabel(r)" class="td-volume">{{ volumeLabel(r) }}</span></td>
<td class="td-times">{{ calcMaxTimes(r) }}</td>
<td class="td-cost">{{ fmtPrice(r.kitCost) }}</td>
<td class="td-cost original">{{ fmtPrice(r.originalCost) }}</td>
<td class="td-price">
<div class="price-input-wrap">
<span>¥</span>
<input
type="number"
:value="getSellingPrice(r._id)"
@change="setSellingPrice(r._id, $event.target.value)"
class="selling-input"
/>
</div>
</td>
<td class="td-profit" :class="{ negative: calcMargin(r.kitCost, getSellingPrice(r._id)) < 0 }">
{{ calcMargin(r.kitCost, getSellingPrice(r._id)).toFixed(1) }}%
</td>
</tr>
</tbody>
</table>
</div>
<!-- Cross Comparison -->
<div class="cross-section">
<h4>横向对比 ({{ crossComparison.length }} 个配方)</h4>
<div class="cross-scroll">
<table class="cross-table">
<thead>
<tr>
<th class="th-name">配方名</th>
<th v-for="ka in kitAnalysis" :key="ka.id" class="th-kit">{{ ka.name }}</th>
<th class="th-kit">单买</th>
</tr>
</thead>
<tbody>
<tr v-for="row in crossComparison" :key="row.id">
<td class="td-name">{{ row.name }} <span v-if="volumeLabel(row)" class="td-volume">{{ volumeLabel(row) }}</span></td>
<td v-for="ka in kitAnalysis" :key="ka.id" :class="row.costs[ka.id] != null ? 'td-kit-available' : 'td-kit-na'">
<template v-if="row.costs[ka.id] != null">{{ fmtPrice(row.costs[ka.id]) }}</template>
<template v-else><span class="na"></span></template>
</td>
<td class="td-cost original">{{ fmtPrice(row.originalCost) }}</td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
</template>
<script setup>
import { ref, computed, onMounted, watch } from 'vue'
import { useRouter } from 'vue-router'
import { useAuthStore } from '../stores/auth'
import { useOilsStore } from '../stores/oils'
import { useRecipesStore } from '../stores/recipes'
import { useUiStore } from '../stores/ui'
import { useKitCost } from '../composables/useKitCost'
const router = useRouter()
const auth = useAuthStore()
const oils = useOilsStore()
const recipeStore = useRecipesStore()
const ui = useUiStore()
const { KITS, kitAnalysis, crossComparison } = useKitCost()
const activeKit = ref(KITS[0].id)
const sellingPrices = ref({})
const activeKitData = computed(() => kitAnalysis.value.find(k => k.id === activeKit.value))
function volumeLabel(recipe) {
const vol = recipe.volume
if (vol) {
if (vol === 'single') return '单次'
if (vol === 'custom') return ''
if (/^\d+$/.test(vol)) return `${vol}ml`
return vol
}
const ings = recipe.ingredients || []
const coco = ings.find(i => i.oil === '椰子油')
if (coco && coco.drops) {
const totalDrops = ings.reduce((s, i) => s + (i.drops || 0), 0)
const ml = totalDrops / 18.6
if (ml <= 2) return '单次'
return `${Math.round(ml)}ml`
}
let totalMl = 0
let hasProduct = false
for (const ing of ings) {
if (!oils.isPortionUnit(ing.oil)) continue
hasProduct = true
totalMl += ing.drops || 0
}
if (hasProduct && totalMl > 0) return `${Math.round(totalMl)}ml`
return ''
}
onMounted(async () => {
if (!auth.isBusiness && !auth.isAdmin) {
router.replace('/projects')
return
}
if (!oils.oilNames.length) await oils.loadOils()
if (!recipeStore.recipes.length) await recipeStore.loadRecipes()
loadSellingPrices()
})
// Persist selling prices to localStorage
const STORAGE_KEY = 'kit-export-selling-prices'
function loadSellingPrices() {
try {
const stored = localStorage.getItem(STORAGE_KEY)
if (stored) sellingPrices.value = JSON.parse(stored)
} catch { /* ignore */ }
}
function saveSellingPrices() {
localStorage.setItem(STORAGE_KEY, JSON.stringify(sellingPrices.value))
}
// Calculate how many times a recipe can be made with the kit (limited by the oil that runs out first)
function calcMaxTimes(recipe) {
const ings = (recipe.ingredients || []).filter(i => i.oil && i.oil !== '椰子油' && i.drops > 0)
if (!ings.length) return '—'
let minTimes = Infinity
for (const ing of ings) {
const meta = oils.oilsMeta[ing.oil]
if (!meta || !meta.dropCount) return '—'
const times = Math.floor(meta.dropCount / ing.drops)
if (times < minTimes) minTimes = times
}
return minTimes === Infinity ? '—' : minTimes + '次'
}
function getSellingPrice(recipeId) {
return sellingPrices.value[recipeId] ?? 0
}
function setSellingPrice(recipeId, val) {
sellingPrices.value[recipeId] = Number(val) || 0
saveSellingPrices()
}
function fmtPrice(n) {
return '¥' + (n || 0).toFixed(2)
}
function calcMargin(cost, price) {
if (!price || price <= 0) return 0
return ((price - cost) / price) * 100
}
// Excel export
async function exportExcel(mode) {
const ExcelJS = (await import('exceljs')).default || await import('exceljs')
const wb = new ExcelJS.Workbook()
const headerFill = { type: 'pattern', pattern: 'solid', fgColor: { argb: 'FF4A9D7E' } }
const headerFont = { bold: true, color: { argb: 'FFFFFFFF' }, size: 11 }
const priceFont = { color: { argb: 'FF4A9D7E' } }
const naFont = { color: { argb: 'FFBBBBBB' } }
function applyHeaderStyle(row) {
row.eachCell(cell => {
cell.fill = headerFill
cell.font = headerFont
cell.alignment = { horizontal: 'center', vertical: 'middle' }
})
row.height = 24
}
function autoCols(ws, ingredientCol = -1) {
ws.columns.forEach((col, i) => {
if (i === 0) {
// First column (配方名): fit longest content
let max = 8
col.eachCell({ includeEmpty: true }, cell => {
const len = cell.value ? String(cell.value).length * 1.8 : 0
if (len > max) max = len
})
col.width = Math.min(max + 2, 30)
} else if (i === ingredientCol) {
// Ingredient column: wider
col.width = 35
} else {
// All other columns: uniform narrow width
col.width = 10
}
})
}
// Per-kit sheets
for (const ka of kitAnalysis.value) {
const ws = wb.addWorksheet(ka.name)
// Kit info header
ws.mergeCells(mode === 'full' ? 'A1:H1' : 'A1:F1')
const titleCell = ws.getCell('A1')
titleCell.value = `${ka.name} — ¥${ka.price}${ka.oils.length}种精油 — 可做${ka.recipeCount}个配方`
titleCell.font = { bold: true, size: 13 }
titleCell.alignment = { horizontal: 'center' }
if (mode === 'full') {
// Full version: recipe name, tags, ingredients, times, kit cost, original cost, selling price, margin
const headers = ['配方名', '标签', '精油成分', '可做次数', '套装成本', '单买成本', '售价', '利润率']
const headerRow = ws.addRow(headers)
applyHeaderStyle(headerRow)
for (const r of ka.recipes) {
const price = getSellingPrice(r._id)
const margin = calcMargin(r.kitCost, price)
const ingredientStr = r.ingredients.map(i => {
const unit = oils.unitLabel(i.oil)
return `${i.oil} ${i.drops}${unit}`
}).join('、')
const vol = volumeLabel(r)
ws.addRow([
vol ? `${r.name}${vol}` : r.name,
(r.tags || []).join('/'),
ingredientStr,
calcMaxTimes(r),
Number(r.kitCost.toFixed(2)),
Number(r.originalCost.toFixed(2)),
price || '',
price ? `${margin.toFixed(1)}%` : '',
])
}
} else {
// Simple version: recipe name, times, kit cost, original cost, selling price, margin
const headers = ['配方名', '可做次数', '套装成本', '单买成本', '售价', '利润率']
const headerRow = ws.addRow(headers)
applyHeaderStyle(headerRow)
for (const r of ka.recipes) {
const price = getSellingPrice(r._id)
const margin = calcMargin(r.kitCost, price)
const vol = volumeLabel(r)
ws.addRow([
vol ? `${r.name}${vol}` : r.name,
calcMaxTimes(r),
Number(r.kitCost.toFixed(2)),
Number(r.originalCost.toFixed(2)),
price || '',
price ? `${margin.toFixed(1)}%` : '',
])
}
}
autoCols(ws, mode === 'full' ? 2 : -1)
// Style cost columns
ws.eachRow((row, rowNum) => {
if (rowNum <= 2) return
row.eachCell((cell, colNum) => {
cell.alignment = { horizontal: 'center', vertical: 'middle' }
// ingredient column left-aligned
if (mode === 'full' && colNum === 3) {
cell.alignment = { horizontal: 'left', vertical: 'middle', wrapText: true }
}
})
})
}
// Cross comparison sheet
const csWs = wb.addWorksheet('横向对比')
const csHeaders = ['配方名']
if (mode === 'full') csHeaders.push('标签')
for (const ka of kitAnalysis.value) csHeaders.push(ka.name)
csHeaders.push('售价')
if (mode === 'full') csHeaders.push('精油成分')
const csHeaderRow = csWs.addRow(csHeaders)
applyHeaderStyle(csHeaderRow)
for (const row of crossComparison.value) {
const price = getSellingPrice(row.id)
const vol = volumeLabel(row)
const vals = [vol ? `${row.name}${vol}` : row.name]
if (mode === 'full') vals.push((row.tags || []).join('/'))
for (const ka of kitAnalysis.value) {
const cost = row.costs[ka.id]
vals.push(cost != null ? Number(cost.toFixed(2)) : '—')
}
vals.push(price || '')
if (mode === 'full') {
vals.push(row.ingredients.map(i => `${i.oil} ${i.drops}${oils.unitLabel(i.oil)}`).join('、'))
}
const dataRow = csWs.addRow(vals)
// Grey out "—" cells
dataRow.eachCell((cell) => {
if (cell.value === '—') cell.font = naFont
cell.alignment = { horizontal: 'center', vertical: 'middle' }
})
}
// Cross sheet: ingredient is last column in full mode
const csIngCol = mode === 'full' ? csHeaders.length - 1 : -1
autoCols(csWs, csIngCol)
// Download
const buf = await wb.xlsx.writeBuffer()
const blob = new Blob([buf], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' })
const url = URL.createObjectURL(blob)
const a = document.createElement('a')
const today = new Date().toISOString().slice(0, 10)
a.href = url
a.download = `套装方案对比_${mode === 'full' ? '完整版' : '简版'}_${today}.xlsx`
a.click()
URL.revokeObjectURL(url)
ui.showToast('导出成功')
}
</script>
<style scoped>
.kit-export-page {
padding: 0 12px 24px;
}
.toolbar-sticky {
position: sticky;
top: 0;
z-index: 20;
background: linear-gradient(135deg, #f0faf5 0%, #e8f0e8 100%);
margin: 0 -12px;
padding: 0 12px;
border-bottom: 1.5px solid #d4e8d4;
}
.toolbar-inner {
display: flex;
align-items: center;
gap: 10px;
padding: 12px 0;
}
.toolbar-actions {
margin-left: auto;
display: flex;
gap: 6px;
}
.page-title {
margin: 0;
font-size: 16px;
color: #3e3a44;
}
/* Kit Cards */
.kit-cards {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
gap: 10px;
margin: 16px 0;
}
.kit-card {
padding: 14px;
background: #fff;
border: 1.5px solid #e5e4e7;
border-radius: 12px;
cursor: pointer;
transition: all 0.15s;
text-align: center;
}
.kit-card:hover { border-color: #7ec6a4; }
.kit-card.active {
border-color: #4a9d7e;
background: #f0faf5;
box-shadow: 0 2px 8px rgba(74, 157, 126, 0.15);
}
.kit-name { font-weight: 600; font-size: 14px; color: #3e3a44; margin-bottom: 4px; }
.kit-price { font-size: 18px; font-weight: 700; color: #4a9d7e; margin-bottom: 2px; }
.kit-discount { font-size: 11px; color: #e65100; margin-bottom: 6px; }
.kit-stats { font-size: 12px; color: #6b6375; display: flex; flex-direction: column; gap: 2px; }
.kit-recipe-count { color: #4a9d7e; font-weight: 500; }
/* Detail Table */
.kit-detail {
margin-bottom: 20px;
padding: 14px;
background: #f8f7f5;
border-radius: 12px;
border: 1.5px solid #e5e4e7;
}
.detail-header {
margin-bottom: 12px;
}
.detail-header h4 { margin: 0; font-size: 14px; color: #3e3a44; }
.recipe-table, .cross-table {
width: 100%; border-collapse: collapse; font-size: 13px;
}
.recipe-table th, .cross-table th {
text-align: center; padding: 8px 6px; font-size: 12px; font-weight: 600;
color: #999; border-bottom: 2px solid #e5e4e7; white-space: nowrap;
}
.recipe-table td, .cross-table td {
padding: 8px 6px; border-bottom: 1px solid #f0f0f0; text-align: center;
}
.th-name { text-align: left !important; }
.td-name { text-align: left !important; font-weight: 500; color: #3e3a44; }
.td-tags { font-size: 11px; color: #b0aab5; }
.td-cost { color: #4a9d7e; font-weight: 500; }
.td-cost.original { color: #999; font-weight: 400; }
.td-profit { font-weight: 600; color: #4a9d7e; }
.td-profit.negative { color: #ef5350; }
.td-volume { font-size: 10px; color: #b0aab5; margin-left: 4px; }
.td-times { color: #6b6375; font-size: 12px; }
.td-kit-available { font-weight: 500; color: #4a9d7e; background: #f0faf5; }
.td-kit-na { background: #fafafa; }
.na { color: #ddd; }
.price-input-wrap {
display: inline-flex; align-items: center; gap: 2px; font-size: 13px; color: #3e3a44;
}
.selling-input {
width: 55px; text-align: right; padding: 3px 4px; border: 1px solid #d4cfc7;
border-radius: 6px; font-size: 12px; font-family: inherit; outline: none;
}
.selling-input:focus { border-color: #7ec6a4; }
/* Cross Comparison */
.cross-section {
margin-bottom: 20px;
padding: 14px;
background: #f8f7f5;
border-radius: 12px;
border: 1.5px solid #e5e4e7;
}
.cross-section h4 { margin: 0 0 12px; font-size: 14px; color: #3e3a44; }
.cross-scroll { overflow-x: auto; }
/* Buttons */
.btn-back {
border: none; background: #f0eeeb; padding: 8px 14px; border-radius: 8px;
cursor: pointer; font-family: inherit; font-size: 13px; color: #6b6375;
}
.btn-back:hover { background: #e5e4e7; }
.btn-outline {
background: #fff; color: #6b6375; border: 1.5px solid #d4cfc7; border-radius: 10px;
padding: 9px 20px; font-size: 13px; cursor: pointer; font-family: inherit;
}
.btn-outline:hover { background: #f8f7f5; }
.btn-sm { padding: 6px 14px; font-size: 12px; border-radius: 8px; }
.empty-hint {
text-align: center; color: #b0aab5; font-size: 13px; padding: 24px 0;
}
@media (max-width: 600px) {
.kit-cards { grid-template-columns: repeat(2, 1fr); }
.toolbar-inner { flex-wrap: wrap; }
.toolbar-actions { width: 100%; justify-content: flex-end; }
}
</style>

View File

@@ -91,7 +91,7 @@
<!-- Search + View Toggle + Add + PDF -->
<div style="display:flex;gap:6px;align-items:center;margin-bottom:12px;flex-wrap:nowrap">
<div class="search-box" style="flex:1;min-width:140px;margin-bottom:0">
<input class="search-input" v-model="searchQuery" placeholder="搜索精油名称…" style="width:100%" />
<input class="search-input" v-model="searchQuery" placeholder="搜索中文或英文名…" style="width:100%" />
</div>
<div style="display:flex;border:1.5px solid var(--border);border-radius:10px;overflow:hidden;flex-shrink:0">
<button @click="viewMode = 'bottle'" :style="viewMode === 'bottle' ? 'background:var(--sage);color:white' : 'background:white;color:var(--text-mid)'" style="border:none;border-radius:0;font-size:12px;padding:6px 12px;cursor:pointer">每瓶价</button>
@@ -99,10 +99,10 @@
</div>
<!-- Desktop: text buttons -->
<button v-if="auth.canManage" class="toolbar-btn-text" @click="showAddForm = !showAddForm">{{ showAddForm ? '收起' : ' 新增' }}</button>
<button v-if="auth.isAdmin" class="toolbar-btn-text" @click="exportPDF">📥 导出PDF</button>
<button v-if="auth.isAdmin" class="toolbar-btn-text" @click="exportExcel">📥 导出Excel</button>
<!-- Mobile: emoji-only buttons -->
<button v-if="auth.canManage" class="toolbar-btn-icon" @click="showAddForm = !showAddForm" title="新增精油"></button>
<button v-if="auth.isAdmin" class="toolbar-btn-icon" @click="exportPDF" title="导出PDF">📄</button>
<button v-if="auth.isAdmin" class="toolbar-btn-icon" @click="exportExcel" title="导出Excel">📄</button>
</div>
<!-- Add Oil Form (toggleable) -->
@@ -110,6 +110,14 @@
<div class="add-type-tabs">
<button class="add-type-tab" :class="{ active: addType === 'oil' }" @click="addType = 'oil'">精油</button>
<button class="add-type-tab" :class="{ active: addType === 'product' }" @click="addType = 'product'">其他</button>
<button class="add-type-tab" :class="{ active: showSmartPaste }" @click="showSmartPaste = !showSmartPaste" style="margin-left:auto">🪄 智能识别</button>
</div>
<div v-if="showSmartPaste" class="form-row" style="flex-direction:column;align-items:stretch;gap:6px">
<textarea v-model="smartPasteText" rows="4" class="form-input-sm" placeholder="粘贴产品信息,例如:&#10;优惠顾客价:¥310&#10;零售价:¥465&#10;规格:100毫升&#10;花样年华焕颜精华水 Salubelle Rejuvenating Essence" style="width:100%;resize:vertical;font-family:inherit"></textarea>
<div style="display:flex;gap:8px">
<button class="btn btn-primary btn-sm" @click="runSmartPaste" :disabled="!smartPasteText.trim()">识别并填入</button>
<button class="btn btn-sm" @click="smartPasteText = ''">清空</button>
</div>
</div>
<!-- 新增精油 -->
<div v-if="addType === 'oil'" class="form-row">
@@ -181,9 +189,16 @@
<!-- Oil Knowledge Card Modal -->
<div v-if="activeCard && activeCardName" class="modal-overlay" @mousedown.self="closeOilModal">
<div class="oil-card-modal">
<div class="oil-card-header">
<div class="oil-card-header-content">
<div class="oil-card-modal" style="position:relative;overflow:hidden">
<!-- Brand background -->
<div v-if="brand.brand_bg" style="position:absolute;inset:0;width:100%;height:100%;background-size:cover;background-position:center;opacity:0.08;pointer-events:none;z-index:0" :style="{ backgroundImage: `url('${brand.brand_bg}')` }"></div>
<!-- QR code -->
<div v-if="brand.qr_code" style="position:absolute;top:16px;right:16px;display:flex;flex-direction:column;gap:3px;z-index:3" :style="{ alignItems: (brand.brand_align === 'left' ? 'flex-start' : brand.brand_align === 'right' ? 'flex-end' : 'center') }">
<img :src="brand.qr_code" crossorigin="anonymous" style="width:48px;height:48px;object-fit:cover;border-radius:6px;box-shadow:0 2px 6px rgba(0,0,0,0.1)" />
<div v-if="brand.brand_name" style="font-size:7px;color:rgba(255,255,255,0.8);line-height:1.3;max-width:60px;white-space:pre-line;text-align:center">{{ brand.brand_name }}</div>
</div>
<div class="oil-card-header" style="position:relative;z-index:1">
<div class="oil-card-header-content" :style="brand.qr_code ? 'padding-right:70px' : ''">
<span class="oil-card-emoji">{{ activeCard.emoji }}</span>
<div>
<h2 class="oil-card-title">{{ activeCardName }}</h2>
@@ -196,7 +211,7 @@
</div>
</div>
</div>
<button class="btn-close btn-close-light" @click="closeOilModal"></button>
<button class="btn-close btn-close-light" @click="closeOilModal" style="z-index:4"></button>
</div>
<!-- Method badges -->
<div class="oil-card-methods">
@@ -227,6 +242,10 @@
<h4 class="oil-card-caution-title"> 注意事项</h4>
<p>{{ activeCard.caution }}</p>
</div>
<!-- Logo -->
<div v-if="brand.brand_logo" style="padding-top:8px">
<img :src="brand.brand_logo" crossorigin="anonymous" style="height:24px;opacity:0.7" />
</div>
<div style="text-align:center;padding-top:12px">
<button class="btn btn-outline btn-sm" @click="saveCardImage(activeCardName)">💾 保存图片</button>
</div>
@@ -316,28 +335,45 @@
</div>
<div class="modal-body">
<div class="form-group">
<label>精油名称</label>
<input v-model="editOilDisplayName" class="form-input" type="text" placeholder="精油名称" />
<label>{{ editUnit === 'drop' ? '精油名称' : '产品名称' }}</label>
<input v-model="editOilDisplayName" class="form-input" type="text" :placeholder="editUnit === 'drop' ? '精油名称' : '产品名称'" />
</div>
<div class="form-group">
<label>英文名</label>
<input v-model="editOilEnName" class="form-input" type="text" placeholder="English name" />
</div>
<div class="form-group">
<label>容量</label>
<select v-model="editVolume" class="form-select">
<option value="2.5">2.5ml (46)</option>
<option value="5">5ml (93)</option>
<option value="10">10ml (186)</option>
<option value="15">15ml (280)</option>
<option value="115">115ml (2146)</option>
<option value="custom">自定义</option>
</select>
</div>
<div class="form-group" v-if="editVolume === 'custom'">
<label>自定义滴数</label>
<input v-model.number="editDropCount" class="form-input" type="number" />
</div>
<!-- 精油容量 -->
<template v-if="editUnit === 'drop'">
<div class="form-group">
<label>容量</label>
<select v-model="editVolume" class="form-select">
<option value="2.5">2.5ml (46)</option>
<option value="5">5ml (93)</option>
<option value="10">10ml (186)</option>
<option value="15">15ml (280)</option>
<option value="115">115ml (2146)</option>
<option value="custom">自定义</option>
</select>
</div>
<div class="form-group" v-if="editVolume === 'custom'">
<label>自定义滴数</label>
<input v-model.number="editDropCount" class="form-input" type="number" />
</div>
</template>
<!-- 其他产品容量 -->
<template v-else>
<div class="form-group">
<label>容量</label>
<div style="display:flex;gap:6px;align-items:center">
<input v-model.number="editProductAmount" class="form-input" type="number" min="1" style="flex:1" />
<select v-model="editProductUnit" class="form-select" style="width:70px">
<option value="ml">ml</option>
<option value="g">g</option>
<option value="capsule"></option>
</select>
</div>
</div>
</template>
<div class="form-group">
<label>会员价 (¥)</label>
<input v-model.number="editBottlePrice" class="form-input" type="number" />
@@ -406,14 +442,22 @@ import { useAuthStore } from '../stores/auth'
import { useUiStore } from '../stores/ui'
import { useRecipesStore } from '../stores/recipes'
import { oilEn } from '../composables/useOilTranslation'
import { getOilCard, setOilCard } from '../composables/useOilCards'
import { getOilCard } from '../composables/useOilCards'
import { showConfirm } from '../composables/useDialog'
import { api } from '../composables/useApi'
import { parseOilProductPaste } from '../composables/useOilProductPaste'
const auth = useAuthStore()
const oils = useOilsStore()
const recipeStore = useRecipesStore()
const ui = useUiStore()
// Brand data for card
const brand = ref({})
async function loadBrand() {
try { brand.value = await api.get('/api/brand') } catch {}
}
// Modal states
const showDilution = ref(false)
const showContra = ref(false)
@@ -433,6 +477,28 @@ const activeCard = ref(null)
// Add oil form
const addType = ref('oil')
const showSmartPaste = ref(false)
const smartPasteText = ref('')
function runSmartPaste() {
const raw = smartPasteText.value || ''
if (!raw.trim()) return
const parsed = parseOilProductPaste(raw)
if (parsed.memberPrice != null) newBottlePrice.value = parsed.memberPrice
if (parsed.retailPrice != null) newRetailPrice.value = parsed.retailPrice
if (parsed.cn) newOilName.value = parsed.cn
if (parsed.en) newOilEnName.value = parsed.en
addType.value = parsed.type
if (parsed.type === 'oil') {
if (parsed.volume) newVolume.value = parsed.volume
newCustomDrops.value = null
} else {
if (parsed.productAmount != null) newProductAmount.value = parsed.productAmount
if (parsed.productUnit) newProductUnit.value = parsed.productUnit
}
ui.showToast('已识别并填入,请检查后点添加')
}
const newOilName = ref('')
const newOilEnName = ref('')
const newBottlePrice = ref(null)
@@ -450,6 +516,9 @@ const editVolume = ref('5')
const editDropCount = ref(0)
const editRetailPrice = ref(null)
const editOilEnName = ref('')
const editUnit = ref('drop')
const editProductAmount = ref(null)
const editProductUnit = ref('ml')
const editCardEmoji = ref('')
const editCardEffects = ref('')
const editCardUsage = ref('')
@@ -576,8 +645,14 @@ const filteredOilNames = computed(() => {
if (!searchQuery.value.trim()) return oils.oilNames
const q = searchQuery.value.trim().toLowerCase()
return oils.oilNames.filter(n => {
const en = getEnglishName(n).toLowerCase()
return n.toLowerCase().includes(q) || en.includes(q)
if (n.toLowerCase().includes(q)) return true
const card = getOilCard(n)
if (card?.en && card.en.toLowerCase().includes(q)) return true
const meta = oils.oilsMeta[n]
if (meta?.enName && meta.enName.toLowerCase().includes(q)) return true
const fallback = oilEn(n)
if (fallback && fallback.toLowerCase().includes(q)) return true
return false
})
})
@@ -595,12 +670,12 @@ function getMeta(name) {
}
function getEnglishName(name) {
// 1. Oil card has priority
const card = getOilCard(name)
if (card && card.en) return card.en
// 2. Stored en_name in meta
// 1. User-edited en_name in DB wins — prevents saves being masked by static cards
const meta = oils.oilsMeta[name]
if (meta?.enName) return meta.enName
// 2. Oil card fallback
const card = getOilCard(name)
if (card && card.en) return card.en
// 3. Static translation map
return oilEn(name)
}
@@ -637,12 +712,9 @@ async function openOilDetail(name) {
activeCardName.value = name
activeCard.value = card
selectedOilName.value = null
// Pre-generate card image for instant save
loadBrand()
// Generate image on demand when saving, not on open
oilCardImageUrl.value = null
await nextTick()
await new Promise(r => setTimeout(r, 300))
const el = document.querySelector('.oil-card-modal')
if (el) await generateImageFromRef({ value: el }, oilCardImageUrl)
} else {
activeCard.value = null
activeCardName.value = null
@@ -712,6 +784,11 @@ function editOil(name) {
editDropCount.value = dc
editRetailPrice.value = meta?.retailPrice || null
editOilEnName.value = meta?.enName || getEnglishName(name) || ''
editUnit.value = meta?.unit || 'drop'
if (editUnit.value !== 'drop') {
editProductAmount.value = dc
editProductUnit.value = editUnit.value
}
// Load knowledge card if exists
const card = getOilCard(name)
editCardEmoji.value = card?.emoji || ''
@@ -737,25 +814,26 @@ async function saveEditOil() {
if (newName && newName !== oldName) {
await oils.deleteOil(oldName)
}
const finalDropCount = editUnit.value !== 'drop' ? editProductAmount.value : dropCount
const finalUnit = editUnit.value !== 'drop' ? editProductUnit.value : null
// Build card payload if any card content provided
const hasCard = editCardEffects.value.trim() || editCardUsage.value.trim()
const cardPayload = hasCard ? {
emoji: editCardEmoji.value || '🌿',
effects: editCardEffects.value.trim(),
usage: editCardUsage.value.trim(),
method: editCardMethod.value.trim(),
caution: editCardCaution.value.trim(),
} : null
await oils.saveOil(
newName || oldName,
editBottlePrice.value,
dropCount,
finalDropCount,
editRetailPrice.value,
editOilEnName.value.trim() || null
editOilEnName.value.trim() || null,
finalUnit,
cardPayload
)
// Save knowledge card if any content provided
const finalName = newName || oldName
if (editCardEffects.value.trim() || editCardUsage.value.trim()) {
setOilCard(finalName, {
emoji: editCardEmoji.value || '🌿',
en: editOilEnName.value.trim() || '',
effects: editCardEffects.value.trim(),
usage: editCardUsage.value.trim(),
method: editCardMethod.value.trim(),
caution: editCardCaution.value.trim(),
})
}
cardVersion.value++ // trigger re-render for card badges
ui.showToast('已更新')
editingOilName.value = null
@@ -811,65 +889,43 @@ async function removeOil(name) {
}
}
// PDF Export
function exportPDF() {
// Excel Export
async function exportExcel() {
const XLSX = (await import('xlsx')).default || await import('xlsx')
const today = new Date()
const dateStr = today.getFullYear() + String(today.getMonth()+1).padStart(2,'0') + String(today.getDate()).padStart(2,'0')
const title = '精油价目表' + dateStr
const dateStr = today.getFullYear() + '-' + String(today.getMonth()+1).padStart(2,'0') + '-' + String(today.getDate()).padStart(2,'0')
const sortedNames = [...oils.oilNames].sort((a, b) => a.localeCompare(b, 'zh'))
let rows = ''
const rows = []
for (const name of sortedNames) {
const meta = getMeta(name)
if (!meta) continue
const en = getEnglishName(name)
const bp = meta.bottlePrice != null ? '¥' + meta.bottlePrice.toFixed(0) : '--'
const rp = meta.retailPrice != null ? '¥' + meta.retailPrice.toFixed(0) : '--'
const vol = volumeLabel(meta.dropCount)
const ppd = oils.pricePerDrop(name) ? '¥' + oils.pricePerDrop(name).toFixed(2) : '--'
rows += `<tr>
<td>${name}</td>
<td>${en}</td>
<td>${bp}</td>
<td>${rp}</td>
<td>${vol}</td>
<td>${ppd}</td>
</tr>`
const en = meta.enName || getEnglishName(name)
const vol = volumeLabel(meta.dropCount, name)
const unit = oilPriceUnit(name)
const ppdNum = oils.pricePerDrop(name)
const card = getOilCard(name)
rows.push({
'精油': name,
'英文名': en,
'会员价': meta.bottlePrice != null ? Number(meta.bottlePrice.toFixed(2)) : '',
'零售价': meta.retailPrice != null ? Number(meta.retailPrice.toFixed(2)) : '',
'容量': vol,
'单价': ppdNum ? `¥${ppdNum.toFixed(2)}/${unit}` : '',
'功效': card?.effects || '',
'使用方法': card?.usage || '',
'使用方式': card?.method || '',
'注意事项': card?.caution || '',
'状态': meta.isActive === false ? '下架' : '在售',
})
}
const html = `<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>${title}</title>
<style>
body { font-family: 'PingFang SC','Hiragino Sans GB','Microsoft YaHei',sans-serif; padding: 20px; font-size: 11px; color: #333; }
h1 { font-size: 18px; text-align: center; margin-bottom: 16px; }
table { width: 100%; border-collapse: collapse; }
th { background: #7a9e7e; color: white; padding: 6px 8px; text-align: center; font-size: 11px; font-weight: 600; }
td { padding: 5px 8px; border-bottom: 1px solid #e0e0e0; text-align: center; font-size: 11px; }
td:first-child, th:first-child { text-align: left; font-weight: 500; }
td:nth-child(2), th:nth-child(2) { text-align: left; }
tr:nth-child(even) { background: #f9f9f9; }
tr:hover { background: #e8f5e9; }
@media print { body { padding: 10px; } h1 { font-size: 16px; } }
</style>
</head>
<body>
<h1>doTERRA 精油价目表 ${dateStr}</h1>
<table>
<thead>
<tr><th>精油</th><th>英文名</th><th>会员价</th><th>零售价</th><th>容量</th><th>单价/滴</th></tr>
</thead>
<tbody>${rows}</tbody>
</table>
<p style="text-align:center;font-size:10px;color:#aaa;margin-top:12px">共 ${sortedNames.length} 种精油 · doTERRA 配方计算器导出</p>
</body>
</html>`
const w = window.open('', '_blank')
w.document.write(html)
w.document.close()
w.document.title = title
setTimeout(() => w.print(), 500)
const ws = XLSX.utils.json_to_sheet(rows)
ws['!cols'] = [{ wch: 16 }, { wch: 28 }, { wch: 10 }, { wch: 10 }, { wch: 12 }, { wch: 16 }, { wch: 40 }, { wch: 40 }, { wch: 20 }, { wch: 24 }, { wch: 8 }]
const wb = XLSX.utils.book_new()
XLSX.utils.book_append_sheet(wb, ws, '精油价目表')
XLSX.writeFile(wb, `精油价目表${dateStr}.xlsx`)
ui.showToast('导出成功')
}
// ──── Save image logic (identical to RecipeDetailOverlay) ────

View File

@@ -1,9 +1,17 @@
<template>
<div class="projects-page">
<!-- Login prompt -->
<div v-if="!auth.isLoggedIn" class="login-prompt">
<div class="commercial-icon">💼</div>
<p>登录后可使用商业核算功能</p>
<button class="btn-primary" @click="ui.openLogin()">登录 / 注册</button>
</div>
<template v-else>
<!-- Header -->
<div class="commercial-header">
<div class="commercial-icon">💼</div>
<div class="commercial-desc">商业用户专属功能包含项目核算成本分析等工具</div>
<button class="btn-kit-compare" @click="handleKitExport">📦 套装方案对比</button>
</div>
<!-- Project List -->
@@ -73,7 +81,7 @@
<tr>
<th>精油</th>
<th>用量</th>
<th>每滴</th>
<th>单价</th>
<th>小计</th>
<th></th>
</tr>
@@ -125,8 +133,8 @@
<tbody>
<tr v-for="c in consumptionData" :key="c.oil" :class="{ 'limit-oil': c.isLimit }">
<td>{{ c.oil }}</td>
<td>{{ c.drops }}</td>
<td>{{ c.bottleDrops }}</td>
<td>{{ c.drops }}{{ oils.unitLabel(c.oil) }}</td>
<td>{{ c.bottleDrops }}{{ oils.unitLabel(c.oil) }}</td>
<td>{{ c.sessions }}</td>
<td></td>
</tr>
@@ -227,6 +235,7 @@
</div>
</div>
</div>
</template>
</div>
</template>
@@ -296,6 +305,14 @@ function selectDemoProject() {
}
}
function handleKitExport() {
if (!auth.isBusiness && !auth.isAdmin) {
showCertPrompt()
return
}
router.push('/kit-export')
}
function handleCreateProject() {
if (!auth.isBusiness && !auth.isAdmin) {
showCertPrompt()
@@ -866,6 +883,25 @@ function formatDate(d) {
font-weight: 500;
}
.btn-kit-compare {
display: inline-block;
margin-top: 12px;
background: linear-gradient(135deg, #f0e6d3 0%, #e8d5b8 100%);
color: #7a6540;
border: 1.5px solid #d4c4a0;
border-radius: 10px;
padding: 8px 20px;
font-size: 13px;
cursor: pointer;
font-family: inherit;
font-weight: 500;
transition: all 0.15s;
}
.btn-kit-compare:hover {
background: linear-gradient(135deg, #e8d5b8 0%, #d4c4a0 100%);
box-shadow: 0 2px 6px rgba(122, 101, 64, 0.15);
}
/* Buttons */
.btn-primary {
background: linear-gradient(135deg, #7ec6a4 0%, #4a9d7e 100%);
@@ -923,6 +959,14 @@ function formatDate(d) {
color: #6b6375;
}
.login-prompt {
text-align: center;
padding: 60px 20px;
color: #6b6375;
}
.login-prompt .commercial-icon { font-size: 48px; margin-bottom: 12px; }
.login-prompt p { margin-bottom: 16px; font-size: 15px; }
.empty-hint {
text-align: center;
color: #b0aab5;

View File

@@ -44,7 +44,7 @@
<!-- Action buttons -->
<div class="action-bar">
<button class="action-chip" @click="showAddOverlay = true">新增</button>
<button class="action-chip" @click="oils.loadOils(); showAddOverlay = true">新增</button>
<button class="action-chip" :class="{ active: isAllSelected }" @click="toggleSelectAll">
全选<span v-if="totalSelected > 0" class="chip-count">{{ totalSelected }}</span>
</button>
@@ -808,6 +808,7 @@ function editRecipe(recipe) {
}
formNote.value = recipe.note || ''
formTags.value = [...(recipe.tags || [])]
oils.loadOils()
showAddOverlay.value = true
}
@@ -1698,7 +1699,7 @@ async function exportExcel() {
}
const today = new Date().toISOString().slice(0, 10)
XLSX.writeFile(wb, `精油配方${today}.xlsx`)
XLSX.writeFile(wb, `精油配方备份${today}.xlsx`)
ui.showToast('导出成功')
}
@@ -2113,7 +2114,7 @@ watch(() => recipeStore.recipes, () => {
.editor-table th { text-align: center; padding: 6px 4px; color: #999; font-weight: 500; font-size: 12px; border-bottom: 1px solid #eee; }
.editor-table th:first-child { text-align: left; }
.editor-table td { padding: 6px 4px; border-bottom: 1px solid #f5f5f5; }
.editor-drops { width: 42px; padding: 5px 2px; border: 1.5px solid #d4cfc7; border-radius: 8px; font-size: 13px; text-align: center; outline: none; font-family: inherit; }
.editor-drops { width: 58px; padding: 5px 4px 5px 6px; border: 1.5px solid #d4cfc7; border-radius: 8px; font-size: 13px; text-align: center; outline: none; font-family: inherit; }
.editor-drops:focus { border-color: #7ec6a4; }
.drops-with-unit { display: flex; align-items: center; gap: 2px; }
.unit-hint { font-size: 11px; color: #b0aab5; white-space: nowrap; }

View File

@@ -312,7 +312,7 @@ function expandQuery(q) {
return terms
}
// Search results: exact matches (query in recipe name or tags, NOT oil names to avoid noise like 西班牙牛至)
// Search results: matches in recipe name, tags, oil names (zh + en)
const exactResults = computed(() => {
if (!searchQuery.value.trim()) return []
const q = searchQuery.value.trim().toLowerCase()
@@ -322,9 +322,10 @@ const exactResults = computed(() => {
const nameMatch = r.name.toLowerCase().includes(q)
const enNameMatch = isEn && (r.en_name || '').toLowerCase().includes(q)
const oilEnMatch = isEn && r.ingredients.some(ing => (oilEn(ing.oil) || '').toLowerCase().includes(q))
const oilZhMatch = q.length >= 2 && r.ingredients.some(ing => ing.oil.toLowerCase().includes(q))
const visibleTags = auth.canEdit ? (r.tags || []) : (r.tags || []).filter(t => !EDITOR_ONLY_TAGS.includes(t))
const tagMatch = visibleTags.some(t => t.toLowerCase().includes(q))
return nameMatch || enNameMatch || oilEnMatch || tagMatch
return nameMatch || enNameMatch || oilEnMatch || oilZhMatch || tagMatch
}).sort((a, b) => a.name.localeCompare(b.name, 'zh'))
})