feat: ssr fixes + switch project page to tanstack (#5192)
* feat: ssr fixes * feat: lazy load non-core data * feat: ssr timing debugging * feat: go back to all parallel * feat: migrate to DI + set up mutators * feat: remove double get versions request, only call v3 * refactor: [version].vue page to use composition API and typescript * feat: gallery.vue start * fix: remove left behind console log * fix: type issues + gallery * fix: versionsummary modal + version page direct join * fix: projectRaw guard * fix: currentMember val fix * fix: actualProjectType * fix: vers summary link same page * fix: lint --------- Co-authored-by: tdgao <mr.trumgao@gmail.com>
This commit is contained in:
73
apps/frontend/src/middleware/project.global.ts
Normal file
73
apps/frontend/src/middleware/project.global.ts
Normal file
@@ -0,0 +1,73 @@
|
||||
import { useGeneratedState } from '~/composables/generated'
|
||||
import { useAppQueryClient } from '~/composables/query-client'
|
||||
import { getProjectTypeForUrlShorthand } from '~/helpers/projects.js'
|
||||
import { useServerModrinthClient } from '~/server/utils/api-client'
|
||||
|
||||
// All valid project type URL segments
|
||||
const PROJECT_TYPES = ['project', 'mod', 'plugin', 'datapack', 'shader', 'resourcepack', 'modpack']
|
||||
|
||||
export default defineNuxtRouteMiddleware(async (to) => {
|
||||
// Only handle project routes
|
||||
if (!to.params.id || !PROJECT_TYPES.includes(to.params.type as string)) {
|
||||
return
|
||||
}
|
||||
|
||||
const queryClient = useAppQueryClient()
|
||||
const authToken = useCookie('auth-token')
|
||||
const client = useServerModrinthClient({ authToken: authToken.value || undefined })
|
||||
const tags = useGeneratedState()
|
||||
const projectId = to.params.id as string
|
||||
|
||||
try {
|
||||
// Fetch v2 project for redirect check AND cache it for the page
|
||||
// Using fetchQuery ensures the page's useQuery gets this cached result
|
||||
const project = await queryClient.fetchQuery({
|
||||
queryKey: ['project', 'v2', projectId],
|
||||
queryFn: () => client.labrinth.projects_v2.get(projectId),
|
||||
staleTime: 1000 * 60 * 5,
|
||||
})
|
||||
|
||||
// Let page handle 404
|
||||
if (!project) return
|
||||
|
||||
// Cache by slug if we looked up by ID (or vice versa)
|
||||
if (projectId !== project.slug) {
|
||||
queryClient.setQueryData(['project', 'v2', project.slug], project)
|
||||
}
|
||||
if (projectId !== project.id) {
|
||||
queryClient.setQueryData(['project', 'v2', project.id], project)
|
||||
}
|
||||
|
||||
// Determine the correct URL type
|
||||
const correctType = getProjectTypeForUrlShorthand(
|
||||
project.project_type,
|
||||
project.loaders,
|
||||
tags.value,
|
||||
)
|
||||
|
||||
// Preserve the rest of the path (subpages like /versions, /settings, etc.)
|
||||
const pathParts = to.path.split('/')
|
||||
pathParts.splice(0, 3) // Remove '', type, and id
|
||||
const remainder = pathParts.filter((x) => x).join('/')
|
||||
|
||||
// Build the canonical path
|
||||
const canonicalPath = `/${correctType}/${project.slug}${remainder ? `/${remainder}` : ''}`
|
||||
|
||||
// Only redirect if the path actually changed
|
||||
if (to.path !== canonicalPath) {
|
||||
return navigateTo(
|
||||
{
|
||||
path: canonicalPath,
|
||||
query: to.query,
|
||||
hash: to.hash,
|
||||
},
|
||||
{
|
||||
redirectCode: 301,
|
||||
replace: true,
|
||||
},
|
||||
)
|
||||
}
|
||||
} catch {
|
||||
// Let the page handle 404s and other errors
|
||||
}
|
||||
})
|
||||
Reference in New Issue
Block a user