* feat: base content card component * fix: tooltips + colors * feat: fix orgs * feat: base content tab internals rewrite * feat: fix invalidmodal * feat: add ContentModpackCard * fix: extract types * draft: layout * feat: unlink modal * feat: impl content tab * fix: lint * fix: toggling * temp: disable updating stuff * feat: selection v-model * feat: bulk selection * feat: mods tab rough draft * feat: use fuse.js * feat: add project combobox * clean up project combobox * feat: start install to play modal * fix: events * feat: use v-on * feat: bulk actions + fix floating action bar width * feat: figma alignments * feat: migrate toggle to tailwind * fix: row borders * feat: disabled state * feat: virtual list impl for card table based on window scroll * fix: lint * feat: virtualization + smaller contentcard items * feat: use ContentCardTable + ContentCardItems * feat: fix gap + border issues on last elm * feat: cleanup + use proper searching * fix: use TeleportOverflowMenu * fix: fallback to svg if src is invalid on avatar component * fix: storybook * feat: start on updater modal * feat: finish content updater modal * feat: i18n pass * feat: impl modal * feat(app): backend changes for content tab refactor (#5237) * feat: include_changelog=false for updater modal * fix: hash overrides * feat: update checking for modpack * feat: qa * feat: modpack content modal * fix: padding in table to match modals + tightness * fix: lint * feat: delete modal * feat: fix toggle bugs * fix: prepr * fix: duplicate messages * qa: full width search * qa: use bg-surface-1.5 * qa: animation for filter pills * qa: standardize hover colors * fix: border-[1px] is border * qa: mass de-select actually mass selecting * qa: match figma designs for floating action bar * qa: modal fixes * q: modal fixes x2 * fix: table border * qa: confirm modals * qa: modal alignment * qa: re-add stuck heading + dedupe logic * qa: dedupe virtual scrolling + remove dead components * qa: responsiveness for content table + link fixes * qa: version column link, tooltips + lint fixes * qa: instance busy protections * fix: installation freeze bug * chore: remove old mods page * refactor: deduplicate layout * chore: delete old content page(s) * qa * qa * qa * feat: sort btn - to iterate * fix: ml * feat: date added * fix: lint * fix: formatting.ts removal * feat: get_dependencies_as_content_items * qa: final QA changes * refactor: deduplicate + polish content.rs * feat: hook up content.vue with v1 * feat: hide v1 content api behind frontend feature flag * fix: query keys + copy on empty state * chore: i18n pass * feat: reimpl unlink + upload endpoint * feat: use bulk endpoints v1 * fix: lint * fix: flags * fix: responsiveness via container queries * fix: lint * qa: 1 * qa: fixes * qa: fix ssr issues with browse content * qa: header page divider * qa: modals * fix: prepr * fix: issues * fix: lint * fix: toggle v1 ff * qa: 5 * qa: delete modal copy * feat: creation flow modals (#5383) * refactor: delete content v0 usages + impl * feat: qa + fixes * feat: installing banner using state event * feat: fix modpack card bugs + filtering issues * refactor: delete backups v0 api module * feat: v1 servers GET endpoint * fix: backups * feat: swap to kyros upload v1 addon * fix: use tanstack for loader.vue * feat: finish install from discovery modal * qa: bug fixes * feat: set up installation settings * fix: lint * fix: typos * fix: bugs * fix: disable inline content * feat: content tab improvements — upload UX, installation settings, and client-only indicators Upload cancellation and navigation guard: - Add ConfirmLeaveModal that prompts when navigating away during upload - Cancel in-flight XHR uploads when user confirms leaving the page - Add beforeunload handler to warn on browser/tab close during upload - Track uploadedBytes/totalBytes in UploadState for progress display - Replace Collapsible with Transition for upload progress admonition - Show byte progress and percentage in upload banner - Clamp upload progress to prevent exceeding 100% Installation settings (server.properties): - Add KnownPropertiesFields and PropertiesFields types to Archon types - Add buildProperties() to creation flow context to collect gamemode, difficulty, seed, world type, structures, and generator settings - Pass properties through installContent on onboarding, discovery, and ServerSetupModal flows Server setup and discovery flow improvements: - Migrate ServerSetupModal from servers_v0.reinstall to content_v1.installContent - Replace loaderApiNames lookup with toApiLoader() helper - Remove eraseDataOnInstall toggle — always use soft_override: false - Simplify modpack install on discovery page to use first available version and route through creation flow modal for both onboarding and non-onboarding - Differentiate post-install navigation: content page for onboarding, loader options for existing servers Modpack update flow: - Replace updateModpack() call with installContent() using soft_override: true to support version selection in the content updater modal Client-only mod indicators: - Add environment field to AddonVersion (reuses Labrinth.Projects.v3.Environment) - Add environment to ContentItem and isClientOnly to ContentCardTableItem - Show orange TriangleAlertIcon with tooltip on client-only mods in content table - Add "Client-only" filter pill to content filtering (controlled via showClientOnlyFilter on ContentManagerContext) - Apply client-only indicators in both ContentPageLayout and ModpackContentModal Misc: - Add CLAUDE.md note about using prepr commands for lint checks - Export ConfirmLeaveModal from instances barrel * fix: piping * fix: switch content disable for linked server instances * feat: client only filter * fix: prepr * feat: hasUpdate shape update * feat: bulk update endpoint impl for content in panel * feat: websocket state impl again with new phases * fix: ws * fix: use timeout fn for sync admon + fix content card layout scroll for browsers with overflow anchor bug * fix: qa bugs * fix: lint, a11y and i18n * refactor: set up layouts folder properly * fix: linked data cache stuff + lint * feat: move installationsettings to shared layout * fix: lint * fix: issues * feat: temp fuck staging up * fix: lockfile * fix: data sync issues on loader.vue * fix: lint * Hide shader configuration files from content list (#5499) * feat: workaround search problem + split out reset * fix: qa * fix: changelog not showing on first open * fix: qa + optimistic updating improvements * fix: prepr+lint * fix: qa * feat: qa * fix: lint * fix: lint * fix: build * fix: build * fix: type errors * fix: fade and JAVA_HOME passthrough * feat: qa * feat: impl diff shit * fix: qa * fix: app qa * feat: update diff modal * fix: endpoint * fix: qa * fix: qa * fix: use bulk in modpack modal * feat: abort signal impl + fix issues * fix: diff modal trunc * feat: qa * fix: qa * feat: tooltip content tab * fix: prepr * fix: dismiss on settings btn * feat: qa * feat: dont clear handlers on disconnect * fix: lint * fix: wrangler + introduce staging-archon env file --------- Signed-off-by: Calum H. <calum@modrinth.com> Co-authored-by: tdgao <mr.trumgao@gmail.com> Co-authored-by: Artyom Ezri <61311568+Artezon@users.noreply.github.com>
373 lines
9.7 KiB
TypeScript
373 lines
9.7 KiB
TypeScript
import { GenericModrinthClient, type Labrinth } from '@modrinth/api-client'
|
|
import serverSidedVue from '@vitejs/plugin-vue'
|
|
import fs from 'fs/promises'
|
|
import { defineNuxtConfig } from 'nuxt/config'
|
|
import svgLoader from 'vite-svg-loader'
|
|
|
|
const STAGING_API_URL = 'https://staging-api.modrinth.com/v2/'
|
|
|
|
const preloadedFonts = [
|
|
'inter/Inter-Regular.woff2',
|
|
'inter/Inter-Medium.woff2',
|
|
'inter/Inter-SemiBold.woff2',
|
|
'inter/Inter-Bold.woff2',
|
|
]
|
|
|
|
const favicons = {
|
|
'(prefers-color-scheme:no-preference)': '/favicon-light.ico',
|
|
'(prefers-color-scheme:light)': '/favicon-light.ico',
|
|
'(prefers-color-scheme:dark)': '/favicon.ico',
|
|
}
|
|
|
|
const PROD_MODRINTH_URL = 'https://modrinth.com'
|
|
const STAGING_MODRINTH_URL = 'https://staging.modrinth.com'
|
|
|
|
export default defineNuxtConfig({
|
|
srcDir: 'src/',
|
|
app: {
|
|
head: {
|
|
htmlAttrs: {
|
|
lang: 'en',
|
|
},
|
|
title: 'Modrinth',
|
|
link: [
|
|
// The type is necessary because the linter can't always compare this very nested/complex type on itself
|
|
...preloadedFonts.map((font): object => {
|
|
return {
|
|
rel: 'preload',
|
|
href: `https://cdn-raw.modrinth.com/fonts/${font}?v=3.19`,
|
|
as: 'font',
|
|
type: 'font/woff2',
|
|
crossorigin: 'anonymous',
|
|
}
|
|
}),
|
|
...Object.entries(favicons).map(([media, href]): object => {
|
|
return { rel: 'icon', type: 'image/x-icon', href, media }
|
|
}),
|
|
...Object.entries(favicons).map(([media, href]): object => {
|
|
return { rel: 'apple-touch-icon', type: 'image/x-icon', href, media, sizes: '64x64' }
|
|
}),
|
|
{
|
|
rel: 'search',
|
|
type: 'application/opensearchdescription+xml',
|
|
href: '/opensearch.xml',
|
|
title: 'Modrinth mods',
|
|
},
|
|
],
|
|
},
|
|
},
|
|
vite: {
|
|
css: {
|
|
preprocessorOptions: {
|
|
scss: {
|
|
// TODO: dont forget about this
|
|
silenceDeprecations: ['import'],
|
|
},
|
|
},
|
|
},
|
|
ssr: {
|
|
// https://github.com/Akryum/floating-vue/issues/809#issuecomment-1002996240
|
|
noExternal: ['v-tooltip'],
|
|
optimizeDeps: {
|
|
include: ['vue-router'],
|
|
},
|
|
},
|
|
define: {
|
|
global: {},
|
|
},
|
|
esbuild: {
|
|
define: {
|
|
global: 'globalThis',
|
|
},
|
|
},
|
|
cacheDir: '../../node_modules/.vite/apps/knossos',
|
|
resolve: {
|
|
dedupe: ['vue'],
|
|
},
|
|
plugins: [
|
|
svgLoader({
|
|
svgoConfig: {
|
|
plugins: [
|
|
{
|
|
name: 'preset-default',
|
|
params: {
|
|
overrides: {
|
|
removeViewBox: false,
|
|
},
|
|
},
|
|
},
|
|
],
|
|
},
|
|
}),
|
|
],
|
|
build: {
|
|
rollupOptions: {
|
|
external: ['cloudflare:workers'],
|
|
},
|
|
},
|
|
},
|
|
hooks: {
|
|
async 'nitro:config'(nitroConfig) {
|
|
const emailTemplates = Object.keys(
|
|
await import('./src/templates/emails/index.ts').then((m) => m.default),
|
|
)
|
|
const docTemplates = Object.keys(
|
|
await import('./src/templates/docs/index.ts').then((m) => m.default),
|
|
)
|
|
const blogArticles = await import('@modrinth/blog').then((m) => m.articles)
|
|
const { getChangelog } = await import('@modrinth/utils')
|
|
|
|
nitroConfig.prerender = nitroConfig.prerender || {}
|
|
nitroConfig.prerender.routes = nitroConfig.prerender.routes || []
|
|
for (const template of emailTemplates) {
|
|
nitroConfig.prerender.routes.push(`/_internal/templates/email/${template}`)
|
|
}
|
|
for (const template of docTemplates) {
|
|
nitroConfig.prerender.routes.push(`/_internal/templates/doc/${template}`)
|
|
}
|
|
nitroConfig.prerender.routes.push('/news')
|
|
for (const article of blogArticles) {
|
|
nitroConfig.prerender.routes.push(`/news/article/${article.slug}`)
|
|
}
|
|
nitroConfig.prerender.routes.push('/news/changelog')
|
|
for (const entry of getChangelog()) {
|
|
const id = entry.version ?? entry.date.unix()
|
|
nitroConfig.prerender.routes.push(`/news/changelog/${entry.product}/${id}`)
|
|
}
|
|
},
|
|
async 'build:before'() {
|
|
// 30 minutes
|
|
const TTL = 30 * 60 * 1000
|
|
|
|
let state: Partial<Labrinth.State.GeneratedState & Record<string, any>> = {}
|
|
|
|
try {
|
|
state = JSON.parse(await fs.readFile('./src/generated/state.json', 'utf8'))
|
|
} catch {
|
|
// File doesn't exist, create folder
|
|
await fs.mkdir('./src/generated', { recursive: true })
|
|
}
|
|
|
|
const API_URL = getApiUrl()
|
|
|
|
if (
|
|
// Skip regeneration if within TTL...
|
|
state.lastGenerated &&
|
|
new Date(state.lastGenerated).getTime() + TTL > new Date().getTime() &&
|
|
// ...but only if the API URL is the same
|
|
state.apiUrl === API_URL &&
|
|
// ...and if no errors were caught during the last generation
|
|
(state.errors ?? []).length === 0
|
|
) {
|
|
console.log(
|
|
'Tags already recently generated. Delete apps/frontend/src/generated/state.json to force regeneration.',
|
|
)
|
|
return
|
|
}
|
|
|
|
const client = new GenericModrinthClient({
|
|
labrinthBaseUrl: API_URL.replace('/v2/', ''),
|
|
userAgent: 'Knossos generator (support@modrinth.com)',
|
|
})
|
|
|
|
const generatedState = await client.labrinth.state.build()
|
|
state.lastGenerated = new Date().toISOString()
|
|
state.apiUrl = API_URL
|
|
state = {
|
|
...state,
|
|
...generatedState,
|
|
}
|
|
|
|
await fs.writeFile('./src/generated/state.json', JSON.stringify(state))
|
|
|
|
// throw if errors and building for prod (preview & staging allowed to have errors)
|
|
if (
|
|
process.env.BUILD_ENV === 'production' &&
|
|
process.env.PREVIEW !== 'true' &&
|
|
generatedState.errors.length > 0
|
|
) {
|
|
throw new Error(
|
|
`Production build failed: State generation encountered errors. Error codes: ${JSON.stringify(generatedState.errors)}; API URL: ${API_URL}`,
|
|
)
|
|
}
|
|
|
|
console.log('Tags generated!')
|
|
|
|
const robotsContent =
|
|
getDomain() === PROD_MODRINTH_URL && process.env.PREVIEW !== 'true'
|
|
? 'User-agent: *\nDisallow: /_internal/'
|
|
: 'User-agent: *\nDisallow: /'
|
|
|
|
await fs.writeFile('./src/public/robots.txt', robotsContent)
|
|
},
|
|
},
|
|
runtimeConfig: {
|
|
// @ts-ignore
|
|
apiBaseUrl: process.env.BASE_URL ?? globalThis.BASE_URL ?? getApiUrl(),
|
|
// @ts-ignore
|
|
rateLimitKey: process.env.RATE_LIMIT_IGNORE_KEY ?? globalThis.RATE_LIMIT_IGNORE_KEY,
|
|
pyroBaseUrl: process.env.PYRO_BASE_URL,
|
|
public: {
|
|
apiBaseUrl: getApiUrl(),
|
|
pyroBaseUrl: process.env.PYRO_BASE_URL,
|
|
siteUrl: getDomain(),
|
|
production: isProduction(),
|
|
buildEnv: process.env.BUILD_ENV,
|
|
preview: process.env.PREVIEW === 'true',
|
|
featureFlagOverrides: getFeatureFlagOverrides(),
|
|
|
|
owner: process.env.VERCEL_GIT_REPO_OWNER || 'modrinth',
|
|
slug: process.env.VERCEL_GIT_REPO_SLUG || 'code',
|
|
branch:
|
|
process.env.VERCEL_GIT_COMMIT_REF ||
|
|
process.env.CF_PAGES_BRANCH ||
|
|
// @ts-ignore
|
|
globalThis.CF_PAGES_BRANCH ||
|
|
'main',
|
|
hash:
|
|
process.env.VERCEL_GIT_COMMIT_SHA ||
|
|
process.env.CF_PAGES_COMMIT_SHA ||
|
|
// @ts-ignore
|
|
globalThis.CF_PAGES_COMMIT_SHA ||
|
|
'unknown',
|
|
|
|
stripePublishableKey:
|
|
process.env.STRIPE_PUBLISHABLE_KEY ||
|
|
globalThis.STRIPE_PUBLISHABLE_KEY ||
|
|
'pk_test_51JbFxJJygY5LJFfKV50mnXzz3YLvBVe2Gd1jn7ljWAkaBlRz3VQdxN9mXcPSrFbSqxwAb0svte9yhnsmm7qHfcWn00R611Ce7b',
|
|
},
|
|
},
|
|
typescript: {
|
|
shim: false,
|
|
strict: true,
|
|
typeCheck: false,
|
|
tsConfig: {
|
|
compilerOptions: {
|
|
moduleResolution: 'bundler',
|
|
allowImportingTsExtensions: true,
|
|
},
|
|
},
|
|
},
|
|
modules: [
|
|
'@pinia/nuxt',
|
|
'floating-vue/nuxt',
|
|
// Sentry causes rollup-plugin-inject errors in dev, only enable in production
|
|
...(isProduction() ? ['@sentry/nuxt/module'] : []),
|
|
],
|
|
floatingVue: {
|
|
themes: {
|
|
'ribbit-popout': {
|
|
$extend: 'dropdown',
|
|
placement: 'bottom-end',
|
|
instantMove: true,
|
|
distance: 8,
|
|
},
|
|
'dismissable-prompt': {
|
|
$extend: 'dropdown',
|
|
placement: 'bottom-start',
|
|
},
|
|
},
|
|
},
|
|
nitro: {
|
|
rollupConfig: {
|
|
// @ts-expect-error because of rolldown-vite - completely fine though
|
|
plugins: [serverSidedVue()],
|
|
external: ['cloudflare:workers'],
|
|
},
|
|
preset: 'cloudflare_module',
|
|
cloudflare: {
|
|
nodeCompat: true,
|
|
},
|
|
replace: {
|
|
__SENTRY_RELEASE__: JSON.stringify(process.env.CF_PAGES_COMMIT_SHA || 'unknown'),
|
|
__SENTRY_ENVIRONMENT__: JSON.stringify(process.env.BUILD_ENV || 'development'),
|
|
},
|
|
},
|
|
devtools: {
|
|
enabled: true,
|
|
},
|
|
css: ['~/assets/styles/tailwind.css'],
|
|
postcss: {
|
|
plugins: {
|
|
tailwindcss: {},
|
|
autoprefixer: {},
|
|
},
|
|
},
|
|
routeRules: {
|
|
'/**': {
|
|
headers: {
|
|
'Accept-CH': 'Sec-CH-Prefers-Color-Scheme',
|
|
'Critical-CH': 'Sec-CH-Prefers-Color-Scheme',
|
|
},
|
|
},
|
|
'/dashboard/revenue/withdraw': {
|
|
redirect: {
|
|
to: '/dashboard/revenue',
|
|
statusCode: 410,
|
|
},
|
|
},
|
|
'/email/**': {
|
|
redirect: '/_internal/templates/email/**',
|
|
},
|
|
'/_internal/templates/email/**': {
|
|
prerender: true,
|
|
headers: {
|
|
'Content-Type': 'text/html',
|
|
'Cache-Control': 'public, max-age=3600',
|
|
},
|
|
},
|
|
'/_internal/templates/doc/**': {
|
|
prerender: true,
|
|
headers: {
|
|
'Content-Type': 'text/html',
|
|
'Cache-Control': 'public, max-age=3600',
|
|
},
|
|
},
|
|
},
|
|
compatibilityDate: '2025-01-01',
|
|
telemetry: false,
|
|
experimental: {
|
|
asyncContext: true,
|
|
},
|
|
sourcemap: { client: 'hidden' },
|
|
sentry: {
|
|
sourcemaps: {
|
|
disable: true,
|
|
},
|
|
},
|
|
})
|
|
|
|
function getApiUrl() {
|
|
// @ts-ignore
|
|
return process.env.BROWSER_BASE_URL ?? globalThis.BROWSER_BASE_URL ?? STAGING_API_URL
|
|
}
|
|
|
|
function isProduction() {
|
|
return process.env.NODE_ENV === 'production'
|
|
}
|
|
|
|
function getFeatureFlagOverrides() {
|
|
return JSON.parse(process.env.FLAG_OVERRIDES ?? '{}')
|
|
}
|
|
|
|
function getDomain() {
|
|
if (process.env.NODE_ENV === 'production') {
|
|
// @ts-ignore
|
|
if (process.env.CF_PAGES_URL || globalThis.CF_PAGES_URL) {
|
|
// @ts-ignore
|
|
return process.env.CF_PAGES_URL ?? globalThis.CF_PAGES_URL
|
|
} else if (process.env.HEROKU_APP_NAME) {
|
|
return `https://${process.env.HEROKU_APP_NAME}.herokuapp.com`
|
|
} else if (process.env.VERCEL_URL) {
|
|
return `https://${process.env.VERCEL_URL}`
|
|
} else if (getApiUrl() === STAGING_API_URL) {
|
|
return STAGING_MODRINTH_URL
|
|
} else {
|
|
return PROD_MODRINTH_URL
|
|
}
|
|
} else {
|
|
const port = process.env.PORT || 3000
|
|
return `http://localhost:${port}`
|
|
}
|
|
}
|