19 Commits

Author SHA1 Message Date
42aefaab17 fix: 保存配方卡片所有英文翻译(含精油名称)
Some checks failed
Test / unit-test (push) Successful in 7s
Test / build-check (push) Successful in 3s
Test / e2e-test (push) Failing after 1m19s
PR Preview / test (pull_request) Has been skipped
PR Preview / teardown-preview (pull_request) Successful in 14s
PR Preview / deploy-preview (pull_request) Has been skipped
之前保存翻译时只保存了配方名的 en_name,精油的自定义英文名
(customOilNameEn)没有持久化,刷新后丢失。

- backend: recipes 表新增 en_oils(JSON 字符串)列
- RecipeUpdate 模型增加 en_oils 字段,update_recipe 写入 DB
- 所有 SELECT 语句补上 en_oils,_recipe_to_dict 返回该字段
- frontend: applyTranslation() 同时提交 en_oils
- onMounted 优先从 r.en_oils 还原精油译名,再 fallback 静态表
- recipes store 映射 en_oils 字段

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-08 19:57:52 +00:00
dcf516f2de fix: 移除所有权限身份显示,QR上传布局还原为initial commit样式
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) Failing after 1m19s
2026-04-08 19:47:24 +00:00
c8de1ad229 fix: 退出登录后跳转到配方查询页面
将 handleLogout 里的 window.location.reload() 改为 router.push('/'),
确保在任何需要登录的页面退出后都能回到配方查询页面。

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-08 19:47:24 +00:00
533cd2a0bd fix: 上传图片后不再自动跳转,由用户手动点击返回配方卡片
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 3s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 1m21s
Co-Authored-By: YoYo <yoyo@euphon.net>
2026-04-08 19:41:04 +00:00
de74ffe638 fix: 配方卡片品牌样式与 initial commit 保持一致
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 15s
Test / e2e-test (push) Failing after 1m16s
- 背景图改用 div+background-image,opacity 从 0.06 恢复为 0.12
- 二维码位置改回 top:36px/right:24px,尺寸 54×54,加圆角和阴影
- 二维码下方新增品牌名称小字(7px)
- Logo 位置改回 bottom:60px,尺寸 height:60px,opacity 0.2

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-08 07:42:47 +00:00
4761253d73 fix: 存为我的改用自定义 showPrompt,与主分支保持一致
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 4s
PR Preview / deploy-preview (pull_request) Successful in 15s
Test / e2e-test (push) Failing after 1m18s
将 saveToDiary 中的原生 prompt() 替换为项目内置的
showPrompt(),使对话框样式与主分支及全站风格保持一致。

Co-Authored-By: YoYo <yoyo@euphon.net>
2026-04-07 23:09:29 +00:00
65239abc53 QR upload: replace inline hint with dialog, add back button in brand page
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 4s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 1m21s
- Replace the inline brand-upload-hint bar in RecipeDetailOverlay with a
  confirm dialog (「去上传」/「取消」) that pops up when the card opens
- Extend useDialog/CustomDialog to support custom ok/cancel button text
- Add a 「← 返回配方卡片」banner in MyDiary brand tab when navigated
  from a recipe card, allowing return without uploading

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-07 22:55:39 +00:00
19f4ab8abe fix: 顶部导航栏sticky修正、右上角显示角色身份、QR上传预览布局优化
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 13s
Test / e2e-test (push) Has been cancelled
- 预览环境下nav-tabs top偏移36px,避免被orange banner遮挡
- 登录用户名旁显示角色身份(查看者/编辑/管理员等)
- 账户页角色改用绿色渐变徽章样式,对齐UserMenu风格
- QR图片上传区改为120x120居中方形,预览样式与二维码链接预览一致

Closes #9 (part: items 1-3, accumulated to PR #5)

Co-Authored-By: YoYo <yoyo@euphon.net>
2026-04-07 22:54:45 +00:00
96504ed1d7 Resume pending action after login (favorite, diary, QR upload)
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 13s
Test / e2e-test (push) Failing after 1m4s
Add pendingAction callback to UI store. When user clicks favorite,
save-to-diary, or upload QR while not logged in, the action is stored
and automatically executed after successful login/register instead of
reloading the page.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-07 22:39:59 +00:00
5eba04a1fa Fix toast rendering: display toast.msg instead of object
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 3s
PR Preview / test (pull_request) Successful in 4s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 1m7s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-07 22:35:41 +00:00
4d5e3c46e7 Shorten copy toast to just '已复制'
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 4s
PR Preview / deploy-preview (pull_request) Successful in 13s
Test / e2e-test (push) Failing after 1m4s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-07 22:34:07 +00:00
3bbe437616 Remove volume selector from recipe card view
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 3s
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-07 22:33:16 +00:00
edc053ae0e Raise toast z-index to 9000 so it shows above all overlays
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 3s
PR Preview / test (pull_request) Successful in 4s
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-07 22:32:42 +00:00
b9681141af Fix LoginModal z-index to 6000 so it appears above recipe overlay
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 4s
PR Preview / deploy-preview (pull_request) Successful in 14s
Test / e2e-test (push) Failing after 1m4s
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-07 22:28:17 +00:00
dee4b1649a 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 4s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Has been cancelled
- VOLUME_DROPS 新增「单次」(null,不缩放)
- 新增 scaleIngredients() 按比例缩放成分
- 卡片预览区加入容量切换按钮,切换后实时更新滴数与成本
- priceInfo 与 coconutDrops 均基于缩放后数据

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-07 22:27:21 +00:00
955512d344 feat: 未登录用户也显示二维码上传提示,点击时引导登录/注册
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
PR Preview / test (pull_request) Successful in 4s
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 1m8s
2026-04-07 22:25:10 +00:00
81ec5987b3 feat: 配方卡片加入上传个人二维码功能
- RecipeDetailOverlay: 未上传二维码/背景图时,卡片上方显示提示横幅,下方出现「上传我的二维码」按钮,点击跳转到 MyDiary 品牌设置页并记录来源配方
- MyDiary: 新增二维码图片上传区域(直接上传图片文件,存为 base64 → PUT /api/brand qr_code 字段);上传成功后若有待返回配方则自动跳回配方卡片;修复 loadBrandSettings 字段名与后端不一致的问题
- RecipeSearch: 支持 ?openRecipe= 查询参数,页面挂载时自动打开指定配方卡片,实现从 MyDiary 上传后无缝返回

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-07 22:25:10 +00:00
70413971e3 Merge pull request 'dev' (#2) from dev into main
Some checks failed
Deploy Production / test (push) Successful in 5s
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 1m7s
Reviewed-on: #2
2026-04-07 22:12:00 +00:00
fam
7ba1e28370 Merge pull request 'Refactor: 前端重构为 Vue 3 + Vite + Pinia + Cypress E2E' (#1) from dev into main
Reviewed-on: #1
2026-04-06 19:22:19 +00:00
13 changed files with 207 additions and 135 deletions

View File

@@ -240,6 +240,8 @@ def init_db():
c.execute("ALTER TABLE recipes ADD COLUMN updated_by INTEGER") c.execute("ALTER TABLE recipes ADD COLUMN updated_by INTEGER")
if "en_name" not in cols: if "en_name" not in cols:
c.execute("ALTER TABLE recipes ADD COLUMN en_name TEXT DEFAULT ''") c.execute("ALTER TABLE recipes ADD COLUMN en_name TEXT DEFAULT ''")
if "en_oils" not in cols:
c.execute("ALTER TABLE recipes ADD COLUMN en_oils TEXT DEFAULT '{}'")
# Seed admin user if no users exist # Seed admin user if no users exist
count = c.execute("SELECT COUNT(*) FROM users").fetchone()[0] count = c.execute("SELECT COUNT(*) FROM users").fetchone()[0]

View File

@@ -96,6 +96,7 @@ class RecipeIn(BaseModel):
class RecipeUpdate(BaseModel): class RecipeUpdate(BaseModel):
name: Optional[str] = None name: Optional[str] = None
en_name: Optional[str] = None en_name: Optional[str] = None
en_oils: Optional[str] = None
note: Optional[str] = None note: Optional[str] = None
ingredients: Optional[list[IngredientIn]] = None ingredients: Optional[list[IngredientIn]] = None
tags: Optional[list[str]] = None tags: Optional[list[str]] = None
@@ -309,7 +310,7 @@ def symptom_search(body: dict, user=Depends(get_current_user)):
conn = get_db() conn = get_db()
# Search in recipe names # Search in recipe names
rows = conn.execute( rows = conn.execute(
"SELECT id, name, note, owner_id, version, en_name FROM recipes ORDER BY id" "SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes ORDER BY id"
).fetchall() ).fetchall()
exact = [] exact = []
related = [] related = []
@@ -698,6 +699,7 @@ def _recipe_to_dict(conn, row):
"id": rid, "id": rid,
"name": row["name"], "name": row["name"],
"en_name": row["en_name"] if "en_name" in row.keys() else "", "en_name": row["en_name"] if "en_name" in row.keys() else "",
"en_oils": row["en_oils"] if "en_oils" in row.keys() else "{}",
"note": row["note"], "note": row["note"],
"owner_id": row["owner_id"], "owner_id": row["owner_id"],
"owner_name": (owner["display_name"] or owner["username"]) if owner else None, "owner_name": (owner["display_name"] or owner["username"]) if owner else None,
@@ -712,19 +714,19 @@ def list_recipes(user=Depends(get_current_user)):
conn = get_db() conn = get_db()
# Admin sees all; others see admin-owned (adopted) + their own # Admin sees all; others see admin-owned (adopted) + their own
if user["role"] == "admin": if user["role"] == "admin":
rows = conn.execute("SELECT id, name, note, owner_id, version, en_name FROM recipes ORDER BY id").fetchall() rows = conn.execute("SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes ORDER BY id").fetchall()
else: else:
admin = conn.execute("SELECT id FROM users WHERE role = 'admin' LIMIT 1").fetchone() admin = conn.execute("SELECT id FROM users WHERE role = 'admin' LIMIT 1").fetchone()
admin_id = admin["id"] if admin else 1 admin_id = admin["id"] if admin else 1
user_id = user.get("id") user_id = user.get("id")
if user_id: if user_id:
rows = conn.execute( rows = conn.execute(
"SELECT id, name, note, owner_id, version, en_name FROM recipes WHERE owner_id = ? OR owner_id = ? ORDER BY id", "SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes WHERE owner_id = ? OR owner_id = ? ORDER BY id",
(admin_id, user_id) (admin_id, user_id)
).fetchall() ).fetchall()
else: else:
rows = conn.execute( rows = conn.execute(
"SELECT id, name, note, owner_id, version, en_name FROM recipes WHERE owner_id = ? ORDER BY id", "SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes WHERE owner_id = ? ORDER BY id",
(admin_id,) (admin_id,)
).fetchall() ).fetchall()
result = [_recipe_to_dict(conn, r) for r in rows] result = [_recipe_to_dict(conn, r) for r in rows]
@@ -735,7 +737,7 @@ def list_recipes(user=Depends(get_current_user)):
@app.get("/api/recipes/{recipe_id}") @app.get("/api/recipes/{recipe_id}")
def get_recipe(recipe_id: int): def get_recipe(recipe_id: int):
conn = get_db() conn = get_db()
row = conn.execute("SELECT id, name, note, owner_id, version, en_name FROM recipes WHERE id = ?", (recipe_id,)).fetchone() row = conn.execute("SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes WHERE id = ?", (recipe_id,)).fetchone()
if not row: if not row:
conn.close() conn.close()
raise HTTPException(404, "Recipe not found") raise HTTPException(404, "Recipe not found")
@@ -808,6 +810,8 @@ def update_recipe(recipe_id: int, update: RecipeUpdate, user=Depends(get_current
c.execute("UPDATE recipes SET note = ? WHERE id = ?", (update.note, recipe_id)) c.execute("UPDATE recipes SET note = ? WHERE id = ?", (update.note, recipe_id))
if update.en_name is not None: if update.en_name is not None:
c.execute("UPDATE recipes SET en_name = ? WHERE id = ?", (update.en_name, recipe_id)) c.execute("UPDATE recipes SET en_name = ? WHERE id = ?", (update.en_name, recipe_id))
if update.en_oils is not None:
c.execute("UPDATE recipes SET en_oils = ? WHERE id = ?", (update.en_oils, recipe_id))
if update.ingredients is not None: if update.ingredients is not None:
c.execute("DELETE FROM recipe_ingredients WHERE recipe_id = ?", (recipe_id,)) c.execute("DELETE FROM recipe_ingredients WHERE recipe_id = ?", (recipe_id,))
for ing in update.ingredients: for ing in update.ingredients:
@@ -837,7 +841,7 @@ def delete_recipe(recipe_id: int, user=Depends(get_current_user)):
conn = get_db() conn = get_db()
row = _check_recipe_permission(conn, recipe_id, user) row = _check_recipe_permission(conn, recipe_id, user)
# Save full snapshot for undo # Save full snapshot for undo
full = conn.execute("SELECT id, name, note, owner_id, version, en_name FROM recipes WHERE id = ?", (recipe_id,)).fetchone() full = conn.execute("SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes WHERE id = ?", (recipe_id,)).fetchone()
snapshot = _recipe_to_dict(conn, full) snapshot = _recipe_to_dict(conn, full)
log_audit(conn, user["id"], "delete_recipe", "recipe", recipe_id, row["name"], log_audit(conn, user["id"], "delete_recipe", "recipe", recipe_id, row["name"],
json.dumps(snapshot, ensure_ascii=False)) json.dumps(snapshot, ensure_ascii=False))
@@ -1340,7 +1344,7 @@ def recipes_by_inventory(user=Depends(get_current_user)):
if not inv: if not inv:
conn.close() conn.close()
return [] return []
rows = conn.execute("SELECT id, name, note, owner_id, version, en_name FROM recipes ORDER BY id").fetchall() rows = conn.execute("SELECT id, name, note, owner_id, version, en_name, en_oils FROM recipes ORDER BY id").fetchall()
result = [] result = []
for r in rows: for r in rows:
recipe = _recipe_to_dict(conn, r) recipe = _recipe_to_dict(conn, r)

View File

@@ -15,7 +15,8 @@
@click="toggleUserMenu" @click="toggleUserMenu"
> >
<template v-if="auth.isLoggedIn"> <template v-if="auth.isLoggedIn">
👤 {{ auth.user.display_name || auth.user.username }} 👤 {{ auth.user.display_name || auth.user.username }}
</template> </template>
<template v-else> <template v-else>
<span style="background:rgba(255,255,255,0.2);padding:4px 12px;border-radius:12px">登录</span> <span style="background:rgba(255,255,255,0.2);padding:4px 12px;border-radius:12px">登录</span>
@@ -41,7 +42,7 @@
<UserMenu v-if="showUserMenu" @close="showUserMenu = false" /> <UserMenu v-if="showUserMenu" @close="showUserMenu = false" />
<!-- Nav tabs --> <!-- Nav tabs -->
<div class="nav-tabs"> <div class="nav-tabs" :style="isPreview ? { top: '36px' } : {}">
<div class="nav-tab" :class="{ active: ui.currentSection === 'search' }" @click="goSection('search')">🔍 配方查询</div> <div class="nav-tab" :class="{ active: ui.currentSection === 'search' }" @click="goSection('search')">🔍 配方查询</div>
<div class="nav-tab" :class="{ active: ui.currentSection === 'manage' }" @click="requireLogin('manage')">📋 管理配方</div> <div class="nav-tab" :class="{ active: ui.currentSection === 'manage' }" @click="requireLogin('manage')">📋 管理配方</div>
<div class="nav-tab" :class="{ active: ui.currentSection === 'inventory' }" @click="requireLogin('inventory')">📦 个人库存</div> <div class="nav-tab" :class="{ active: ui.currentSection === 'inventory' }" @click="requireLogin('inventory')">📦 个人库存</div>
@@ -64,7 +65,7 @@
<CustomDialog /> <CustomDialog />
<!-- Toast messages --> <!-- Toast messages -->
<div v-for="(toast, i) in ui.toasts" :key="i" class="toast">{{ toast }}</div> <div v-for="toast in ui.toasts" :key="toast.id" class="toast">{{ toast.msg }}</div>
</template> </template>
<script setup> <script setup>

View File

@@ -445,7 +445,7 @@ body {
.toast { .toast {
position: fixed; bottom: 80px; left: 50%; transform: translateX(-50%); position: fixed; bottom: 80px; left: 50%; transform: translateX(-50%);
background: rgba(0,0,0,0.8); color: white; padding: 10px 24px; background: rgba(0,0,0,0.8); color: white; padding: 10px 24px;
border-radius: 20px; font-size: 14px; z-index: 999; border-radius: 20px; font-size: 14px; z-index: 9000;
pointer-events: none; transition: opacity 0.3s; pointer-events: none; transition: opacity 0.3s;
} }

View File

@@ -11,8 +11,8 @@
ref="promptInput" ref="promptInput"
/> />
<div class="dialog-btn-row"> <div class="dialog-btn-row">
<button v-if="dialogState.type !== 'alert'" class="dialog-btn-outline" @click="cancel">取消</button> <button v-if="dialogState.type !== 'alert'" class="dialog-btn-outline" @click="cancel">{{ dialogState.cancelText || '取消' }}</button>
<button class="dialog-btn-primary" @click="ok">确定</button> <button class="dialog-btn-primary" @click="ok">{{ dialogState.okText || '确定' }}</button>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -104,8 +104,11 @@ async function submit() {
ui.showToast('注册成功') ui.showToast('注册成功')
} }
emit('close') emit('close')
// Reload page data after auth change if (ui.pendingAction) {
ui.runPendingAction()
} else {
window.location.reload() window.location.reload()
}
} catch (e) { } catch (e) {
errorMsg.value = e?.message || (mode.value === 'login' ? '登录失败,请检查用户名和密码' : '注册失败,请重试') errorMsg.value = e?.message || (mode.value === 'login' ? '登录失败,请检查用户名和密码' : '注册失败,请重试')
} finally { } finally {

View File

@@ -36,38 +36,22 @@
>English</button> >English</button>
</div> </div>
<!-- Volume selector -->
<div class="volume-controls card-volume-controls">
<button
v-for="(drops, ml) in VOLUME_DROPS"
:key="ml"
class="volume-btn"
:class="{ active: selectedCardVolume === ml }"
@click="onCardVolumeChange(ml)"
>{{ ml === '单次' ? '单次' : ml + 'ml' }}</button>
</div>
<!-- QR / brand upload hint -->
<div v-if="showBrandHint" class="brand-upload-hint">
<span class="hint-icon"></span>
<span>上传你的专属二维码生成属于自己的配方卡片</span>
</div>
<!-- Card image (rendered by html2canvas) --> <!-- Card image (rendered by html2canvas) -->
<div v-show="!cardImageUrl" ref="cardRef" class="export-card"> <div v-show="!cardImageUrl" ref="cardRef" class="export-card">
<!-- Brand overlay layers --> <!-- Brand overlay layers -->
<img <div
v-if="brand.brand_bg" v-if="brand.brand_bg"
:src="brand.brand_bg"
class="card-brand-bg" class="card-brand-bg"
crossorigin="anonymous" :style="{ backgroundImage: `url('${brand.brand_bg}')` }"
/> />
<div v-if="brand.qr_code" class="card-qr-wrapper">
<img <img
v-if="brand.qr_code"
:src="brand.qr_code" :src="brand.qr_code"
class="card-qr" class="card-qr"
crossorigin="anonymous" crossorigin="anonymous"
/> />
<div v-if="brand.brand_name" class="card-qr-name">{{ brand.brand_name }}</div>
</div>
<img <img
v-if="brand.brand_logo" v-if="brand.brand_logo"
:src="brand.brand_logo" :src="brand.brand_logo"
@@ -372,6 +356,7 @@ import { useAuthStore } from '../stores/auth'
import { useUiStore } from '../stores/ui' import { useUiStore } from '../stores/ui'
import { useDiaryStore } from '../stores/diary' import { useDiaryStore } from '../stores/diary'
import { api } from '../composables/useApi' import { api } from '../composables/useApi'
import { showConfirm, showPrompt } from '../composables/useDialog'
import { oilEn, recipeNameEn } from '../composables/useOilTranslation' import { oilEn, recipeNameEn } from '../composables/useOilTranslation'
// TagPicker replaced with inline tag editing // TagPicker replaced with inline tag editing
@@ -412,14 +397,30 @@ const canEditThisRecipe = computed(() => {
const isFav = computed(() => recipesStore.isFavorite(recipe.value)) const isFav = computed(() => recipesStore.isFavorite(recipe.value))
// Card ingredients: exclude coconut oil // Scale ingredients proportionally to target volume; '单次' = no scaling
const cardIngredients = computed(() => function scaleIngredients(ingredients, volume) {
recipe.value.ingredients.filter(ing => ing.oil !== '椰子油') const targetDrops = VOLUME_DROPS[volume]
if (!targetDrops) return ingredients // 单次:不缩放
const totalDrops = ingredients.reduce((sum, ing) => sum + (ing.drops || 0), 0)
if (totalDrops === 0) return ingredients
return ingredients.map(ing => ({
...ing,
drops: Math.round(ing.drops * targetDrops / totalDrops),
}))
}
// Card ingredients: scaled to selected volume, coconut oil excluded from display
const scaledCardIngredients = computed(() =>
scaleIngredients(recipe.value.ingredients, selectedCardVolume.value)
) )
// Coconut oil drops const cardIngredients = computed(() =>
scaledCardIngredients.value.filter(ing => ing.oil !== '椰子油')
)
// Coconut oil drops (from scaled set)
const coconutDrops = computed(() => { const coconutDrops = computed(() => {
const coco = recipe.value.ingredients.find(ing => ing.oil === '椰子油') const coco = scaledCardIngredients.value.find(ing => ing.oil === '椰子油')
return coco ? coco.drops : 0 return coco ? coco.drops : 0
}) })
@@ -445,7 +446,7 @@ const dilutionDesc = computed(() => {
: `该配方适用于单次用量(共${totalDrops}滴),其中纯精油 ${totalEoDrops.value} 滴,椰子油 ${coconutDrops.value} 滴,稀释比例为 1:${ratio}` : `该配方适用于单次用量(共${totalDrops}滴),其中纯精油 ${totalEoDrops.value} 滴,椰子油 ${coconutDrops.value} 滴,稀释比例为 1:${ratio}`
}) })
const priceInfo = computed(() => oilsStore.fmtCostWithRetail(recipe.value.ingredients)) const priceInfo = computed(() => oilsStore.fmtCostWithRetail(scaledCardIngredients.value))
// Today string // Today string
const todayStr = computed(() => { const todayStr = computed(() => {
@@ -476,6 +477,14 @@ async function loadBrand() {
} catch { } catch {
brand.value = {} brand.value = {}
} }
// Show upload prompt if user hasn't set up brand assets yet
if (showBrandHint.value) {
const ok = await showConfirm(
'上传你的专属二维码,让配方卡片更专业 ✨',
{ okText: '去上传', cancelText: '取消' }
)
if (ok) goUploadQr()
}
} }
// Whether to show the brand/QR upload hint (show to all users who haven't set up brand assets) // Whether to show the brand/QR upload hint (show to all users who haven't set up brand assets)
@@ -485,7 +494,7 @@ const showBrandHint = computed(() =>
function goUploadQr() { function goUploadQr() {
if (!authStore.isLoggedIn) { if (!authStore.isLoggedIn) {
ui.openLogin() ui.openLogin(() => goUploadQr())
return return
} }
if (recipe.value._id) { if (recipe.value._id) {
@@ -524,6 +533,12 @@ function switchLang(lang) {
nextTick(() => generateCardImage()) nextTick(() => generateCardImage())
} }
function onCardVolumeChange(ml) {
selectedCardVolume.value = ml
cardImageUrl.value = null
nextTick(() => generateCardImage())
}
async function saveImage() { async function saveImage() {
if (!cardImageUrl.value) { if (!cardImageUrl.value) {
await generateCardImage() await generateCardImage()
@@ -553,7 +568,7 @@ function copyText() {
].filter(Boolean).join('\n') ].filter(Boolean).join('\n')
navigator.clipboard.writeText(text).then(() => { navigator.clipboard.writeText(text).then(() => {
ui.showToast('已复制到剪贴板') ui.showToast('已复制')
}).catch(() => { }).catch(() => {
ui.showToast('复制失败') ui.showToast('复制失败')
}) })
@@ -561,11 +576,12 @@ function copyText() {
async function applyTranslation() { async function applyTranslation() {
showTranslationEditor.value = false showTranslationEditor.value = false
// Persist en_name to backend // Persist en_name and en_oils to backend
if (recipe.value._id && customRecipeNameEn.value) { if (recipe.value._id) {
try { try {
await api.put(`/api/recipes/${recipe.value._id}`, { await api.put(`/api/recipes/${recipe.value._id}`, {
en_name: customRecipeNameEn.value, en_name: customRecipeNameEn.value,
en_oils: JSON.stringify(customOilNameEn.value),
version: recipe.value._version, version: recipe.value._version,
}) })
ui.showToast('翻译已保存') ui.showToast('翻译已保存')
@@ -595,7 +611,7 @@ function getCardRecipeName() {
// ---- Favorite ---- // ---- Favorite ----
async function handleToggleFavorite() { async function handleToggleFavorite() {
if (!authStore.isLoggedIn) { if (!authStore.isLoggedIn) {
ui.openLogin() ui.openLogin(() => handleToggleFavorite())
return return
} }
if (!recipe.value._id) { if (!recipe.value._id) {
@@ -614,10 +630,10 @@ async function handleToggleFavorite() {
// ---- Save to diary ---- // ---- Save to diary ----
async function saveToDiary() { async function saveToDiary() {
if (!authStore.isLoggedIn) { if (!authStore.isLoggedIn) {
ui.openLogin() ui.openLogin(() => saveToDiary())
return return
} }
const name = prompt('保存为我的配方,名称:', recipe.value.name) const name = await showPrompt('保存为我的配方,名称:', recipe.value.name)
if (!name) return if (!name) return
try { try {
await api.post('/api/diary', { await api.post('/api/diary', {
@@ -693,9 +709,10 @@ onMounted(() => {
editIngredients.value = (r.ingredients || []).map(i => ({ oil: i.oil, drops: i.drops })) editIngredients.value = (r.ingredients || []).map(i => ({ oil: i.oil, drops: i.drops }))
// Init translation defaults // Init translation defaults
customRecipeNameEn.value = r.en_name || recipeNameEn(r.name) customRecipeNameEn.value = r.en_name || recipeNameEn(r.name)
const savedOilMap = r.en_oils ? (() => { try { return JSON.parse(r.en_oils) } catch { return {} } })() : {}
const enMap = {} const enMap = {}
;(r.ingredients || []).forEach(ing => { ;(r.ingredients || []).forEach(ing => {
enMap[ing.oil] = oilEn(ing.oil) || ing.oil enMap[ing.oil] = savedOilMap[ing.oil] || oilEn(ing.oil) || ing.oil
}) })
customOilNameEn.value = enMap customOilNameEn.value = enMap
@@ -983,32 +1000,52 @@ async function saveRecipe() {
inset: 0; inset: 0;
width: 100%; width: 100%;
height: 100%; height: 100%;
object-fit: cover; background-size: cover;
opacity: 0.06; background-position: center;
background-repeat: no-repeat;
opacity: 0.12;
z-index: 0; z-index: 0;
pointer-events: none; pointer-events: none;
} }
.card-qr { .card-qr-wrapper {
position: absolute; position: absolute;
top: 16px; top: 36px;
right: 16px; right: 24px;
width: 64px; display: flex;
height: 64px; flex-direction: column;
object-fit: contain; align-items: center;
z-index: 3; gap: 3px;
opacity: 0.85; z-index: 2;
}
.card-qr {
width: 54px;
height: 54px;
object-fit: cover;
border-radius: 6px;
box-shadow: 0 2px 6px rgba(0, 0, 0, 0.1);
}
.card-qr-name {
font-size: 7px;
color: var(--text-light, #8a7a6a);
text-align: center;
line-height: 1.3;
max-width: 68px;
white-space: pre-line;
} }
.card-logo { .card-logo {
position: absolute; position: absolute;
bottom: 16px; bottom: 60px;
left: 50%; left: 50%;
transform: translateX(-50%); transform: translateX(-50%);
height: 28px; height: 60px;
object-fit: contain; object-fit: contain;
z-index: 3; z-index: 1;
opacity: 0.3; opacity: 0.2;
pointer-events: none;
} }
.card-brand-text { .card-brand-text {
@@ -1481,6 +1518,10 @@ async function saveRecipe() {
margin-bottom: 10px; margin-bottom: 10px;
} }
.card-volume-controls {
margin: 8px 0 12px;
}
.volume-btn { .volume-btn {
padding: 7px 16px; padding: 7px 16px;
border: 1.5px solid var(--border, #e0d4c0); border: 1.5px solid var(--border, #e0d4c0);

View File

@@ -2,9 +2,6 @@
<div class="usermenu-overlay" @click.self="$emit('close')"> <div class="usermenu-overlay" @click.self="$emit('close')">
<div class="usermenu-card"> <div class="usermenu-card">
<div class="usermenu-name">{{ auth.user.display_name || auth.user.username }}</div> <div class="usermenu-name">{{ auth.user.display_name || auth.user.username }}</div>
<div class="usermenu-role">
<span class="role-badge">{{ roleLabel }}</span>
</div>
<div class="usermenu-actions"> <div class="usermenu-actions">
<button class="usermenu-btn" @click="goMyDiary"> <button class="usermenu-btn" @click="goMyDiary">
@@ -71,16 +68,6 @@ const bugContent = ref('')
const unreadCount = computed(() => notifications.value.filter(n => !n.is_read).length) const unreadCount = computed(() => notifications.value.filter(n => !n.is_read).length)
const roleLabel = computed(() => {
const map = {
admin: '管理员',
senior_editor: '高级编辑',
editor: '编辑',
viewer: '查看者',
}
return map[auth.user.role] || auth.user.role
})
function formatTime(d) { function formatTime(d) {
if (!d) return '' if (!d) return ''
return new Date(d + 'Z').toLocaleString('zh-CN', { month: 'short', day: 'numeric', hour: '2-digit', minute: '2-digit' }) return new Date(d + 'Z').toLocaleString('zh-CN', { month: 'short', day: 'numeric', hour: '2-digit', minute: '2-digit' })
@@ -136,7 +123,7 @@ function handleLogout() {
auth.logout() auth.logout()
ui.showToast('已退出登录') ui.showToast('已退出登录')
emit('close') emit('close')
window.location.reload() router.push('/')
} }
onMounted(loadNotifications) onMounted(loadNotifications)
@@ -162,13 +149,7 @@ onMounted(loadNotifications)
z-index: 4001; z-index: 4001;
} }
.usermenu-name { font-size: 16px; font-weight: 600; color: #3e3a44; margin-bottom: 4px; } .usermenu-name { font-size: 16px; font-weight: 600; color: #3e3a44; margin-bottom: 14px; }
.usermenu-role { margin-bottom: 14px; }
.role-badge {
display: inline-block; font-size: 11px; padding: 2px 10px;
border-radius: 8px; background: linear-gradient(135deg, #e8f5e9, #c8e6c9);
color: #4a9d7e; font-weight: 500;
}
.usermenu-actions { display: flex; flex-direction: column; gap: 4px; } .usermenu-actions { display: flex; flex-direction: column; gap: 4px; }
.usermenu-btn { .usermenu-btn {

View File

@@ -5,6 +5,8 @@ export const dialogState = reactive({
type: 'alert', // 'alert', 'confirm', 'prompt' type: 'alert', // 'alert', 'confirm', 'prompt'
message: '', message: '',
defaultValue: '', defaultValue: '',
okText: '',
cancelText: '',
resolve: null resolve: null
}) })
@@ -13,15 +15,19 @@ export function showAlert(msg) {
dialogState.visible = true dialogState.visible = true
dialogState.type = 'alert' dialogState.type = 'alert'
dialogState.message = msg dialogState.message = msg
dialogState.okText = ''
dialogState.cancelText = ''
dialogState.resolve = resolve dialogState.resolve = resolve
}) })
} }
export function showConfirm(msg) { export function showConfirm(msg, opts = {}) {
return new Promise(resolve => { return new Promise(resolve => {
dialogState.visible = true dialogState.visible = true
dialogState.type = 'confirm' dialogState.type = 'confirm'
dialogState.message = msg dialogState.message = msg
dialogState.okText = opts.okText || ''
dialogState.cancelText = opts.cancelText || ''
dialogState.resolve = resolve dialogState.resolve = resolve
}) })
} }

View File

@@ -5,6 +5,7 @@ import { api } from '../composables/useApi'
export const DROPS_PER_ML = 18.6 export const DROPS_PER_ML = 18.6
export const VOLUME_DROPS = { export const VOLUME_DROPS = {
'单次': null,
'2.5': 46, '2.5': 46,
'5': 93, '5': 93,
'10': 186, '10': 186,

View File

@@ -17,6 +17,7 @@ export const useRecipesStore = defineStore('recipes', () => {
_version: r._version ?? r.version ?? 1, _version: r._version ?? r.version ?? 1,
name: r.name, name: r.name,
en_name: r.en_name ?? '', en_name: r.en_name ?? '',
en_oils: r.en_oils ?? '{}',
note: r.note ?? '', note: r.note ?? '',
tags: r.tags ?? [], tags: r.tags ?? [],
ingredients: (r.ingredients ?? []).map((ing) => ({ ingredients: (r.ingredients ?? []).map((ing) => ({

View File

@@ -5,6 +5,7 @@ export const useUiStore = defineStore('ui', () => {
const currentSection = ref('search') const currentSection = ref('search')
const showLoginModal = ref(false) const showLoginModal = ref(false)
const toasts = ref([]) const toasts = ref([])
const pendingAction = ref(null)
let toastId = 0 let toastId = 0
@@ -20,7 +21,10 @@ export const useUiStore = defineStore('ui', () => {
}, duration) }, duration)
} }
function openLogin() { function openLogin(afterLogin) {
if (afterLogin) {
pendingAction.value = afterLogin
}
showLoginModal.value = true showLoginModal.value = true
} }
@@ -28,13 +32,23 @@ export const useUiStore = defineStore('ui', () => {
showLoginModal.value = false showLoginModal.value = false
} }
function runPendingAction() {
if (pendingAction.value) {
const action = pendingAction.value
pendingAction.value = null
action()
}
}
return { return {
currentSection, currentSection,
showLoginModal, showLoginModal,
toasts, toasts,
pendingAction,
showSection, showSection,
showToast, showToast,
openLogin, openLogin,
closeLogin, closeLogin,
runPendingAction,
} }
}) })

View File

@@ -107,6 +107,11 @@
<!-- Brand Tab --> <!-- Brand Tab -->
<div v-if="activeTab === 'brand'" class="tab-content"> <div v-if="activeTab === 'brand'" class="tab-content">
<!-- Back to recipe card (when navigated from a recipe) -->
<div v-if="returnRecipeId" class="return-banner">
<span>📋 上传完成后可返回配方卡片</span>
<button class="btn-return" @click="goBackToRecipe"> 返回配方卡片</button>
</div>
<div class="section-card"> <div class="section-card">
<h4>🏷 品牌设置</h4> <h4>🏷 品牌设置</h4>
@@ -169,10 +174,6 @@
<div class="form-static">{{ auth.user.username }}</div> <div class="form-static">{{ auth.user.username }}</div>
</div> </div>
<div class="form-group">
<label>角色</label>
<div class="form-static role-badge">{{ roleLabel }}</div>
</div>
</div> </div>
<div class="section-card"> <div class="section-card">
@@ -211,7 +212,7 @@
</template> </template>
<script setup> <script setup>
import { ref, computed, onMounted, watch } from 'vue' import { ref, onMounted, watch } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { useAuthStore } from '../stores/auth' import { useAuthStore } from '../stores/auth'
import { useOilsStore } from '../stores/oils' import { useOilsStore } from '../stores/oils'
@@ -230,6 +231,7 @@ const router = useRouter()
const activeTab = ref('brand') const activeTab = ref('brand')
const pasteText = ref('') const pasteText = ref('')
const selectedDiaryId = ref(null) const selectedDiaryId = ref(null)
const returnRecipeId = ref(null)
const selectedDiary = ref(null) const selectedDiary = ref(null)
const newEntryText = ref('') const newEntryText = ref('')
@@ -250,22 +252,20 @@ const newPassword = ref('')
const confirmPassword = ref('') const confirmPassword = ref('')
const businessReason = ref('') const businessReason = ref('')
const roleLabel = computed(() => {
const roles = {
admin: '管理员',
senior_editor: '高级编辑',
editor: '编辑',
viewer: '查看者',
}
return roles[auth.user.role] || auth.user.role
})
onMounted(async () => { onMounted(async () => {
await diaryStore.loadDiary() await diaryStore.loadDiary()
displayName.value = auth.user.display_name || '' displayName.value = auth.user.display_name || ''
await loadBrandSettings() await loadBrandSettings()
returnRecipeId.value = localStorage.getItem('oil_return_recipe_id') || null
}) })
function goBackToRecipe() {
if (returnRecipeId.value) {
localStorage.removeItem('oil_return_recipe_id')
router.push('/?openRecipe=' + encodeURIComponent(returnRecipeId.value))
}
}
function selectDiary(d) { function selectDiary(d) {
const id = d._id || d.id const id = d._id || d.id
selectedDiaryId.value = id selectedDiaryId.value = id
@@ -412,18 +412,7 @@ async function handleUpload(type, event) {
if (res.ok) { if (res.ok) {
if (type === 'logo') brandLogo.value = base64 if (type === 'logo') brandLogo.value = base64
else if (type === 'bg') brandBg.value = base64 else if (type === 'bg') brandBg.value = base64
else if (type === 'qr') { else if (type === 'qr') brandQrImage.value = base64
brandQrImage.value = base64
// If user came here from a recipe card, navigate back
const returnRecipeId = localStorage.getItem('oil_return_recipe_id')
if (returnRecipeId) {
localStorage.removeItem('oil_return_recipe_id')
ui.showToast('二维码已上传,返回配方卡片 ✨')
await new Promise(r => setTimeout(r, 800))
router.push('/?openRecipe=' + encodeURIComponent(returnRecipeId))
return
}
}
ui.showToast('上传成功') ui.showToast('上传成功')
} }
} catch { } catch {
@@ -753,6 +742,39 @@ async function applyBusiness() {
border-color: #7ec6a4; border-color: #7ec6a4;
} }
/* Return banner */
.return-banner {
display: flex;
align-items: center;
justify-content: space-between;
background: #f0faf5;
border: 1.5px solid #7ec6a4;
border-radius: 12px;
padding: 10px 14px;
margin-bottom: 14px;
font-size: 13px;
color: #3e7d5a;
gap: 10px;
}
.btn-return {
background: linear-gradient(135deg, #7ec6a4 0%, #4a9d7e 100%);
color: #fff;
border: none;
border-radius: 8px;
padding: 6px 14px;
font-size: 12px;
cursor: pointer;
font-family: inherit;
font-weight: 500;
white-space: nowrap;
flex-shrink: 0;
}
.btn-return:hover {
opacity: 0.9;
}
/* Brand */ /* Brand */
.form-group { .form-group {
margin-bottom: 14px; margin-bottom: 14px;
@@ -774,22 +796,6 @@ async function applyBusiness() {
color: #6b6375; color: #6b6375;
} }
.role-badge {
display: inline-block;
}
.qr-preview {
margin-top: 10px;
text-align: center;
}
.qr-img {
width: 120px;
height: 120px;
border-radius: 8px;
border: 1.5px solid #e5e4e7;
}
.upload-area { .upload-area {
width: 100%; width: 100%;
min-height: 80px; min-height: 80px;
@@ -807,6 +813,18 @@ async function applyBusiness() {
border-color: #7ec6a4; border-color: #7ec6a4;
} }
.qr-preview {
margin-top: 10px;
text-align: center;
}
.qr-img {
width: 120px;
height: 120px;
border-radius: 8px;
border: 1.5px solid #e5e4e7;
}
.upload-preview { .upload-preview {
max-width: 80px; max-width: 80px;
max-height: 80px; max-height: 80px;