Compare commits

...

4 Commits

Author SHA1 Message Date
Tim
76dd57b858 Merge pull request #1139 from nagisa77/feature/fix-category-back-history
Feature/fix category back history
2026-02-06 17:44:55 +08:00
Tim
18edde64c3 fix: sync home filter dropdown state to URL history 2026-02-06 17:42:35 +08:00
Tim
ebc79f36e7 fix: keep browser history for home category and tag filters 2026-02-06 17:35:59 +08:00
Tim
a7fbd1eb75 Merge pull request #1138 from nagisa77/feature/fix-notification-mark-all-read
fix: mark all unread notifications across all pages
2026-02-06 17:19:57 +08:00
6 changed files with 55 additions and 7 deletions

View File

@@ -21,7 +21,7 @@ const props = defineProps({
const gotoCategory = async () => {
if (!props.category) return
const value = encodeURIComponent(props.category.id ?? props.category.name)
await navigateTo({ path: '/', query: { category: value } }, { replace: true })
await navigateTo({ path: '/', query: { category: value } })
}
const isImageIcon = (icon) => {

View File

@@ -30,7 +30,7 @@ defineProps({
const gotoTag = async (tag) => {
const value = encodeURIComponent(tag.id ?? tag.name)
await navigateTo({ path: '/', query: { tags: value } }, { replace: true })
await navigateTo({ path: '/', query: { tags: value } })
}
const isImageIcon = (icon) => {

View File

@@ -357,13 +357,13 @@ const isImageIcon = (icon) => {
const gotoCategory = (c) => {
const value = encodeURIComponent(c.id ?? c.name)
navigateTo({ path: '/', query: { category: value } }, { replace: true })
navigateTo({ path: '/', query: { category: value } })
handleItemClick()
}
const gotoTag = (t) => {
const value = encodeURIComponent(t.id ?? t.name)
navigateTo({ path: '/', query: { tags: value } }, { replace: true })
navigateTo({ path: '/', query: { tags: value } })
handleItemClick()
}
</script>

View File

@@ -170,9 +170,9 @@ watch(selected, (val) => {
navigateTo(`/posts/${opt.postId}#comment-${opt.id}`, { replace: true })
}
} else if (opt.type === 'category') {
navigateTo({ path: '/', query: { category: opt.id } }, { replace: true })
navigateTo({ path: '/', query: { category: opt.id } })
} else if (opt.type === 'tag') {
navigateTo({ path: '/', query: { tags: opt.id } }, { replace: true })
navigateTo({ path: '/', query: { tags: opt.id } })
}
selected.value = null
keyword.value = ''

View File

@@ -202,6 +202,28 @@ const selectedTagsSet = (tags) => {
.map((v) => (isNaN(v) ? v : Number(v)))
}
const normalizeCategoryFromQuery = (category) => {
if (category == null || category === '') return ''
const raw = Array.isArray(category) ? category[0] : category
const decoded = decodeURIComponent(raw)
return isNaN(decoded) ? decoded : Number(decoded)
}
const normalizeTagsFromQuery = (tags) => {
if (tags == null || tags === '') return []
const raw = Array.isArray(tags) ? tags.join(',') : tags
return raw
.split(',')
.filter((v) => v)
.map((v) => decodeURIComponent(v))
.map((v) => (isNaN(v) ? v : Number(v)))
}
const arraysShallowEqual = (a = [], b = []) => {
if (a.length !== b.length) return false
return a.every((v, idx) => String(v) === String(b[idx]))
}
/** 初始化:仅在客户端首渲染时根据路由同步一次 **/
onMounted(() => {
const { category, tags } = route.query
@@ -239,6 +261,32 @@ watch(
},
)
// 从筛选器变更回写到 URL确保浏览器历史可回退到上一个筛选状态。
watch([selectedCategory, selectedTags], async ([category, tags]) => {
const routeCategory = normalizeCategoryFromQuery(route.query.category)
const routeTags = normalizeTagsFromQuery(route.query.tags)
const categoryChanged = String(category ?? '') !== String(routeCategory ?? '')
const tagsChanged = !arraysShallowEqual(tags || [], routeTags)
if (!categoryChanged && !tagsChanged) return
const nextQuery = { ...route.query }
if (category == null || category === '') {
delete nextQuery.category
} else {
nextQuery.category = encodeURIComponent(String(category))
}
if (!Array.isArray(tags) || tags.length === 0) {
delete nextQuery.tags
} else {
nextQuery.tags = tags.map((v) => encodeURIComponent(String(v))).join(',')
}
await navigateTo({ path: '/', query: nextQuery })
})
/** 选项加载(分类/标签名称回填) **/
const loadOptions = async () => {
if (selectedCategory.value && !isNaN(selectedCategory.value)) {

View File

@@ -643,7 +643,7 @@ const sendMessage = async () => {
const gotoTag = (tag) => {
const value = encodeURIComponent(tag.id ?? tag.name)
navigateTo({ path: '/', query: { tags: value } }, { replace: true })
navigateTo({ path: '/', query: { tags: value } })
}
const init = async () => {