Compare commits
4 Commits
7599599910
...
fix/save-t
| Author | SHA1 | Date | |
|---|---|---|---|
| b07b97bf1e | |||
| 2ab192c3ba | |||
| 70413971e3 | |||
| 7ba1e28370 |
@@ -119,7 +119,7 @@ async function submit() {
|
|||||||
position: fixed;
|
position: fixed;
|
||||||
inset: 0;
|
inset: 0;
|
||||||
background: rgba(0, 0, 0, 0.35);
|
background: rgba(0, 0, 0, 0.35);
|
||||||
z-index: 6000;
|
z-index: 5000;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|||||||
@@ -36,23 +36,6 @@
|
|||||||
>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 -->
|
||||||
@@ -143,11 +126,6 @@
|
|||||||
class="action-btn"
|
class="action-btn"
|
||||||
@click="showTranslationEditor = true"
|
@click="showTranslationEditor = true"
|
||||||
>✏️ 修改翻译</button>
|
>✏️ 修改翻译</button>
|
||||||
<button
|
|
||||||
v-if="showBrandHint"
|
|
||||||
class="action-btn action-btn-qr"
|
|
||||||
@click="goUploadQr"
|
|
||||||
>📲 上传我的二维码</button>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Translation editor (inline) -->
|
<!-- Translation editor (inline) -->
|
||||||
@@ -364,7 +342,6 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, onMounted, nextTick, watch } from 'vue'
|
import { ref, computed, onMounted, nextTick, watch } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
|
||||||
import html2canvas from 'html2canvas'
|
import html2canvas from 'html2canvas'
|
||||||
import { useOilsStore, DROPS_PER_ML, VOLUME_DROPS } from '../stores/oils'
|
import { useOilsStore, DROPS_PER_ML, VOLUME_DROPS } from '../stores/oils'
|
||||||
import { useRecipesStore } from '../stores/recipes'
|
import { useRecipesStore } from '../stores/recipes'
|
||||||
@@ -373,6 +350,7 @@ 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 { oilEn, recipeNameEn } from '../composables/useOilTranslation'
|
import { oilEn, recipeNameEn } from '../composables/useOilTranslation'
|
||||||
|
import { showPrompt } from '../composables/useDialog'
|
||||||
// TagPicker replaced with inline tag editing
|
// TagPicker replaced with inline tag editing
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
@@ -386,14 +364,12 @@ const recipesStore = useRecipesStore()
|
|||||||
const authStore = useAuthStore()
|
const authStore = useAuthStore()
|
||||||
const ui = useUiStore()
|
const ui = useUiStore()
|
||||||
const diaryStore = useDiaryStore()
|
const diaryStore = useDiaryStore()
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
// ---- View state ----
|
// ---- View state ----
|
||||||
const viewMode = ref('card')
|
const viewMode = ref('card')
|
||||||
const cardRef = ref(null)
|
const cardRef = ref(null)
|
||||||
const cardImageUrl = ref(null)
|
const cardImageUrl = ref(null)
|
||||||
const cardLang = ref('zh')
|
const cardLang = ref('zh')
|
||||||
const selectedCardVolume = ref('单次')
|
|
||||||
const showTranslationEditor = ref(false)
|
const showTranslationEditor = ref(false)
|
||||||
const customRecipeNameEn = ref('')
|
const customRecipeNameEn = ref('')
|
||||||
const customOilNameEn = ref({})
|
const customOilNameEn = ref({})
|
||||||
@@ -478,22 +454,6 @@ async function loadBrand() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Whether to show the brand/QR upload hint (show to all users who haven't set up brand assets)
|
|
||||||
const showBrandHint = computed(() =>
|
|
||||||
!!brand.value && !brand.value.qr_code && !brand.value.brand_bg
|
|
||||||
)
|
|
||||||
|
|
||||||
function goUploadQr() {
|
|
||||||
if (!authStore.isLoggedIn) {
|
|
||||||
ui.openLogin()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (recipe.value._id) {
|
|
||||||
localStorage.setItem('oil_return_recipe_id', recipe.value._id)
|
|
||||||
}
|
|
||||||
router.push('/mydiary')
|
|
||||||
}
|
|
||||||
|
|
||||||
// ---- Card image generation ----
|
// ---- Card image generation ----
|
||||||
async function generateCardImage() {
|
async function generateCardImage() {
|
||||||
if (!cardRef.value || generatingImage.value) return
|
if (!cardRef.value || generatingImage.value) return
|
||||||
@@ -617,16 +577,16 @@ async function saveToDiary() {
|
|||||||
ui.openLogin()
|
ui.openLogin()
|
||||||
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 recipesStore.saveRecipe({
|
||||||
name,
|
name,
|
||||||
source_recipe_id: recipe.value._id || null,
|
|
||||||
ingredients: recipe.value.ingredients.map(i => ({ oil: i.oil, drops: i.drops })),
|
|
||||||
note: recipe.value.note || '',
|
note: recipe.value.note || '',
|
||||||
|
ingredients: recipe.value.ingredients.map(i => ({ oil_name: i.oil, drops: i.drops })),
|
||||||
|
tags: recipe.value.tags || [],
|
||||||
})
|
})
|
||||||
ui.showToast('已保存到「我的配方日记」')
|
ui.showToast('已保存到「我的配方」')
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
ui.showToast('保存失败: ' + (e?.message || '未知错误'))
|
ui.showToast('保存失败: ' + (e?.message || '未知错误'))
|
||||||
}
|
}
|
||||||
@@ -842,7 +802,6 @@ async function saveRecipe() {
|
|||||||
ingredients: ingredients.map(i => ({ oil_name: i.oil, drops: i.drops })),
|
ingredients: ingredients.map(i => ({ oil_name: i.oil, drops: i.drops })),
|
||||||
}
|
}
|
||||||
await recipesStore.saveRecipe(payload)
|
await recipesStore.saveRecipe(payload)
|
||||||
// Reload recipes so the data is fresh when re-opened
|
|
||||||
await recipesStore.loadRecipes()
|
await recipesStore.loadRecipes()
|
||||||
ui.showToast('保存成功')
|
ui.showToast('保存成功')
|
||||||
emit('close')
|
emit('close')
|
||||||
@@ -1188,42 +1147,6 @@ async function saveRecipe() {
|
|||||||
background: var(--sage-mist, #eef4ee);
|
background: var(--sage-mist, #eef4ee);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Brand upload hint banner */
|
|
||||||
.brand-upload-hint {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
background: linear-gradient(135deg, #eef4ee, #f5f0e8);
|
|
||||||
border: 1.5px dashed var(--sage-light, #c8ddc9);
|
|
||||||
border-radius: 10px;
|
|
||||||
padding: 10px 16px;
|
|
||||||
margin-bottom: 14px;
|
|
||||||
font-size: 13px;
|
|
||||||
color: var(--sage-dark, #5a7d5e);
|
|
||||||
font-weight: 500;
|
|
||||||
animation: hint-pop 0.3s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.hint-icon {
|
|
||||||
font-size: 18px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@keyframes hint-pop {
|
|
||||||
from { opacity: 0; transform: translateY(-6px); }
|
|
||||||
to { opacity: 1; transform: translateY(0); }
|
|
||||||
}
|
|
||||||
|
|
||||||
.action-btn-qr {
|
|
||||||
background: linear-gradient(135deg, #c8ddc9, var(--sage, #7a9e7e));
|
|
||||||
color: #fff;
|
|
||||||
border-color: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.action-btn-qr:hover {
|
|
||||||
opacity: 0.88;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Card bottom actions */
|
/* Card bottom actions */
|
||||||
.card-bottom-actions {
|
.card-bottom-actions {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -123,16 +123,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
|
||||||
<label>我的二维码图片</label>
|
|
||||||
<div class="upload-area" @click="triggerUpload('qr')">
|
|
||||||
<img v-if="brandQrImage" :src="brandQrImage" class="upload-preview qr-upload-preview" />
|
|
||||||
<span v-else class="upload-hint">📲 点击上传二维码图片</span>
|
|
||||||
</div>
|
|
||||||
<input ref="qrInput" type="file" accept="image/*" style="display:none" @change="handleUpload('qr', $event)" />
|
|
||||||
<div class="field-hint">上传后将显示在配方卡片右下角</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label>品牌Logo</label>
|
<label>品牌Logo</label>
|
||||||
<div class="upload-area" @click="triggerUpload('logo')">
|
<div class="upload-area" @click="triggerUpload('logo')">
|
||||||
@@ -212,7 +202,6 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, onMounted, watch } from 'vue'
|
import { ref, computed, onMounted, watch } from 'vue'
|
||||||
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'
|
||||||
import { useDiaryStore } from '../stores/diary'
|
import { useDiaryStore } from '../stores/diary'
|
||||||
@@ -225,7 +214,6 @@ const auth = useAuthStore()
|
|||||||
const oils = useOilsStore()
|
const oils = useOilsStore()
|
||||||
const diaryStore = useDiaryStore()
|
const diaryStore = useDiaryStore()
|
||||||
const ui = useUiStore()
|
const ui = useUiStore()
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
const activeTab = ref('brand')
|
const activeTab = ref('brand')
|
||||||
const pasteText = ref('')
|
const pasteText = ref('')
|
||||||
@@ -236,12 +224,10 @@ const newEntryText = ref('')
|
|||||||
// Brand settings
|
// Brand settings
|
||||||
const brandName = ref('')
|
const brandName = ref('')
|
||||||
const brandQrUrl = ref('')
|
const brandQrUrl = ref('')
|
||||||
const brandQrImage = ref('')
|
|
||||||
const brandLogo = ref('')
|
const brandLogo = ref('')
|
||||||
const brandBg = ref('')
|
const brandBg = ref('')
|
||||||
const logoInput = ref(null)
|
const logoInput = ref(null)
|
||||||
const bgInput = ref(null)
|
const bgInput = ref(null)
|
||||||
const qrInput = ref(null)
|
|
||||||
|
|
||||||
// Account settings
|
// Account settings
|
||||||
const displayName = ref('')
|
const displayName = ref('')
|
||||||
@@ -359,9 +345,8 @@ async function loadBrandSettings() {
|
|||||||
const data = await res.json()
|
const data = await res.json()
|
||||||
brandName.value = data.brand_name || ''
|
brandName.value = data.brand_name || ''
|
||||||
brandQrUrl.value = data.qr_url || ''
|
brandQrUrl.value = data.qr_url || ''
|
||||||
brandQrImage.value = data.qr_code || ''
|
brandLogo.value = data.logo_url || ''
|
||||||
brandLogo.value = data.brand_logo || ''
|
brandBg.value = data.bg_url || ''
|
||||||
brandBg.value = data.brand_bg || ''
|
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
// no brand settings yet
|
// no brand settings yet
|
||||||
@@ -384,46 +369,26 @@ async function saveBrandSettings() {
|
|||||||
|
|
||||||
function triggerUpload(type) {
|
function triggerUpload(type) {
|
||||||
if (type === 'logo') logoInput.value?.click()
|
if (type === 'logo') logoInput.value?.click()
|
||||||
else if (type === 'bg') bgInput.value?.click()
|
else bgInput.value?.click()
|
||||||
else if (type === 'qr') qrInput.value?.click()
|
|
||||||
}
|
|
||||||
|
|
||||||
function readFileAsBase64(file) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
const reader = new FileReader()
|
|
||||||
reader.onload = e => resolve(e.target.result)
|
|
||||||
reader.onerror = reject
|
|
||||||
reader.readAsDataURL(file)
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async function handleUpload(type, event) {
|
async function handleUpload(type, event) {
|
||||||
const file = event.target.files[0]
|
const file = event.target.files[0]
|
||||||
if (!file) return
|
if (!file) return
|
||||||
|
const formData = new FormData()
|
||||||
|
formData.append('file', file)
|
||||||
|
formData.append('type', type)
|
||||||
try {
|
try {
|
||||||
const base64 = await readFileAsBase64(file)
|
const token = localStorage.getItem('oil_auth_token') || ''
|
||||||
const fieldMap = { logo: 'brand_logo', bg: 'brand_bg', qr: 'qr_code' }
|
const res = await fetch('/api/brand-upload', {
|
||||||
const field = fieldMap[type]
|
method: 'POST',
|
||||||
if (!field) return
|
headers: token ? { Authorization: 'Bearer ' + token } : {},
|
||||||
const res = await api('/api/brand', {
|
body: formData,
|
||||||
method: 'PUT',
|
|
||||||
body: JSON.stringify({ [field]: base64 }),
|
|
||||||
})
|
})
|
||||||
if (res.ok) {
|
if (res.ok) {
|
||||||
if (type === 'logo') brandLogo.value = base64
|
const data = await res.json()
|
||||||
else if (type === 'bg') brandBg.value = base64
|
if (type === 'logo') brandLogo.value = data.url
|
||||||
else if (type === 'qr') {
|
else brandBg.value = data.url
|
||||||
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 {
|
||||||
@@ -818,18 +783,6 @@ async function applyBusiness() {
|
|||||||
max-height: 100px;
|
max-height: 100px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.qr-upload-preview {
|
|
||||||
max-width: 120px;
|
|
||||||
max-height: 120px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.field-hint {
|
|
||||||
font-size: 12px;
|
|
||||||
color: #9b94a3;
|
|
||||||
margin-top: 4px;
|
|
||||||
padding-left: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.upload-hint {
|
.upload-hint {
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
color: #b0aab5;
|
color: #b0aab5;
|
||||||
|
|||||||
@@ -123,8 +123,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, onMounted, nextTick, watch } from 'vue'
|
import { ref, computed, onMounted, nextTick } from 'vue'
|
||||||
import { useRoute, useRouter } from 'vue-router'
|
|
||||||
import { useAuthStore } from '../stores/auth'
|
import { useAuthStore } from '../stores/auth'
|
||||||
import { useOilsStore } from '../stores/oils'
|
import { useOilsStore } from '../stores/oils'
|
||||||
import { useRecipesStore } from '../stores/recipes'
|
import { useRecipesStore } from '../stores/recipes'
|
||||||
@@ -137,8 +136,6 @@ const auth = useAuthStore()
|
|||||||
const oils = useOilsStore()
|
const oils = useOilsStore()
|
||||||
const recipeStore = useRecipesStore()
|
const recipeStore = useRecipesStore()
|
||||||
const ui = useUiStore()
|
const ui = useUiStore()
|
||||||
const route = useRoute()
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
const searchQuery = ref('')
|
const searchQuery = ref('')
|
||||||
const selectedCategory = ref(null)
|
const selectedCategory = ref(null)
|
||||||
@@ -157,27 +154,6 @@ onMounted(async () => {
|
|||||||
} catch {
|
} catch {
|
||||||
// category modules are optional
|
// category modules are optional
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return to a recipe card after QR upload redirect
|
|
||||||
const openRecipeId = route.query.openRecipe
|
|
||||||
if (openRecipeId) {
|
|
||||||
router.replace({ path: '/', query: {} })
|
|
||||||
const tryOpen = () => {
|
|
||||||
const idx = recipeStore.recipes.findIndex(r => r._id === openRecipeId)
|
|
||||||
if (idx >= 0) {
|
|
||||||
openDetail(idx)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if (!tryOpen()) {
|
|
||||||
// Recipes might not be loaded yet, watch until available
|
|
||||||
const stop = watch(
|
|
||||||
() => recipeStore.recipes.length,
|
|
||||||
() => { if (tryOpen()) stop() },
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
|
|
||||||
function selectCategory(cat) {
|
function selectCategory(cat) {
|
||||||
|
|||||||
Reference in New Issue
Block a user