diff --git a/frontend/src/__tests__/pr27Features.test.js b/frontend/src/__tests__/pr27Features.test.js index aa41519..4aa015e 100644 --- a/frontend/src/__tests__/pr27Features.test.js +++ b/frontend/src/__tests__/pr27Features.test.js @@ -1,5 +1,6 @@ import { describe, it, expect } from 'vitest' import { recipeNameEn, oilEn } from '../composables/useOilTranslation' +import { matchesPinyinInitials, getPinyinInitials } from '../composables/usePinyinMatch' // --------------------------------------------------------------------------- // EDITOR_ONLY_TAGS includes '已下架' @@ -283,3 +284,95 @@ describe('one-time username change guard', () => { expect(!!user.username_changed).toBe(false) }) }) + +// --------------------------------------------------------------------------- +// Pinyin matching — PR29 extended coverage +// --------------------------------------------------------------------------- +describe('pinyin matching — extended oil names', () => { + it('matches mlk → 麦卢卡', () => { + expect(matchesPinyinInitials('麦卢卡', 'mlk')).toBe(true) + }) + + it('matches tx → 檀香', () => { + expect(matchesPinyinInitials('檀香', 'tx')).toBe(true) + }) + + it('matches xm → 香茅', () => { + expect(matchesPinyinInitials('香茅', 'xm')).toBe(true) + }) + + it('matches gbxz → 古巴香脂', () => { + expect(matchesPinyinInitials('古巴香脂', 'gbxz')).toBe(true) + }) + + it('matches my → 没药', () => { + expect(matchesPinyinInitials('没药', 'my')).toBe(true) + }) + + it('matches xhx → 小茴香', () => { + expect(matchesPinyinInitials('小茴香', 'xhx')).toBe(true) + }) + + it('matches jybh → 椒样薄荷', () => { + expect(matchesPinyinInitials('椒样薄荷', 'jybh')).toBe(true) + }) + + it('matches xbynz → 西班牙牛至', () => { + expect(matchesPinyinInitials('西班牙牛至', 'xbynz')).toBe(true) + }) + + it('matches sc → 顺畅呼吸 prefix', () => { + expect(matchesPinyinInitials('顺畅呼吸', 'sc')).toBe(true) + }) + + it('does not match wrong initials', () => { + expect(matchesPinyinInitials('麦卢卡', 'abc')).toBe(false) + }) + + it('getPinyinInitials returns correct string', () => { + expect(getPinyinInitials('麦卢卡')).toBe('mlk') + expect(getPinyinInitials('檀香')).toBe('tx') + expect(getPinyinInitials('没药')).toBe('my') + }) +}) + +// --------------------------------------------------------------------------- +// Viewer tag visibility — PR29 +// --------------------------------------------------------------------------- +describe('viewer tag visibility logic', () => { + const EDITOR_ONLY_TAGS_VAL = ['已审核', '已下架'] + + it('editor sees all tags', () => { + const allTags = ['美容', '儿童', '已审核', '已下架'] + const canEdit = true + const visible = canEdit ? allTags : [] + expect(visible).toEqual(allTags) + }) + + it('viewer sees no public tags', () => { + const canEdit = false + const myDiary = [ + { tags: ['我的标签'] }, + { tags: ['我的标签', '另一个'] }, + ] + // Viewer: collect tags from own diary only + const myTags = new Set() + for (const d of myDiary) { + for (const t of (d.tags || [])) myTags.add(t) + } + const visible = canEdit ? ['美容', '已审核'] : [...myTags] + expect(visible).toContain('我的标签') + expect(visible).toContain('另一个') + expect(visible).not.toContain('美容') + expect(visible).not.toContain('已审核') + }) + + it('viewer with no diary tags sees empty', () => { + const myDiary = [] + const myTags = new Set() + for (const d of myDiary) { + for (const t of (d.tags || [])) myTags.add(t) + } + expect([...myTags]).toHaveLength(0) + }) +}) diff --git a/frontend/src/composables/usePinyinMatch.js b/frontend/src/composables/usePinyinMatch.js index d8684ef..ba447e1 100644 --- a/frontend/src/composables/usePinyinMatch.js +++ b/frontend/src/composables/usePinyinMatch.js @@ -57,6 +57,25 @@ const PINYIN_MAP = { '触': 'c', '修': 'x', '养': 'y', '滋': 'z', '润': 'r', '呼': 'h', '吸': 'x', '消': 'x', '化': 'h', '排': 'p', '毒': 'd', '净': 'j', '纤': 'x', '体': 't', '塑': 's', + // Extended: all oil name chars + '麦': 'm', '卢': 'l', '卡': 'k', '檀': 't', '橘': 'j', + '茅': 'm', '茴': 'h', '芹': 'q', '菜': 'c', '蕾': 'l', + '蜂': 'f', '蓍': 's', '莱': 'l', '姆': 'm', '莎': 's', + '穗': 's', '醇': 'c', '郁': 'y', '没': 'm', '脂': 'z', + '巴': 'b', '样': 'y', '班': 'b', '牙': 'y', '鸡': 'j', + '苍': 'c', '卫': 'w', '畅': 'c', '顺': 's', '释': 's', + '悦': 'y', '柔': 'r', '压': 'y', '定': 'd', '情': 'q', + '绪': 'x', '神': 's', '气': 'q', '宽': 'k', '容': 'r', + '恬': 't', '家': 'j', '欢': 'h', '欣': 'x', '舞': 'w', + '鼓': 'g', '赋': 'f', '谧': 'm', '睡': 's', '烂': 'l', + '绚': 'x', '焕': 'h', '肤': 'f', '年': 'n', '华': 'h', + '完': 'w', '理': 'l', '注': 'z', '贯': 'g', '全': 'q', + '仕': 's', '女': 'nv', '伯': 'b', '斯': 's', '道': 'd', + '格': 'g', '拉': 'l', '元': 'y', '肌': 'j', '栀': 'z', + '鹅': 'e', '掌': 'z', '柴': 'c', '胶': 'j', '囊': 'n', + '空': 'k', '风': 'f', '文': 'w', '月': 'y', '云': 'y', + '五': 'w', '味': 'w', '愈': 'y', '创': 'c', '慰': 'w', + '扁': 'b', '广': 'g', '州': 'z', '热': 'r', } /** diff --git a/frontend/src/views/RecipeManager.vue b/frontend/src/views/RecipeManager.vue index 3ca66d8..1d5b230 100644 --- a/frontend/src/views/RecipeManager.vue +++ b/frontend/src/views/RecipeManager.vue @@ -83,7 +83,7 @@