* fix: files.vue bugs before styling changes * feat: move files tab to shared layout structure * fix: qa * fix: qa * fix: bugs * fix: lint * fix: admonition cleanup with progress + actions * fix: cleanup * fix: modals * fix: admon title * fix: i18n standard * fix: lint + i18n pass * fix: remove transition * fix: type errors * feat: files tab in app * fix: qa * fix: backup item minmax * fix: use ContentPageHeader for server panel * fix: lint * fix: lint * fix: lint * feat: page leave safety * fix: lint * fix: cargo fmt fix * fix: blank in prod * fix: content card table stuff * Revert "fix: blank in prod" This reverts commit 74758fe185cf85a4a20355857f889cb091b97ace. * fix: import * feat: browse worlds/servers flow * fix: worlds tab parity with content tab * fix: perf bug + shader filter pill copy * feat: singleplayer filter * fix: ordering * fix: breadcrumbs * fix: lint * fix: qa * feat: store server proj id when adding to a non-linked instance * fix: lint * fix: i18n + qa * fix: conflict * qa: already installed modal + placeholders not server-specific * fix: qa * fix: add + edit server modals * fix: qa * fix: security * fix: devin flags * fix: lint * chore: change file to break build cache * fix: admon * fix: import path stuff * feat: qa * fix: fmt fmt idiot --------- Signed-off-by: Calum H. <calum@modrinth.com>
753 lines
23 KiB
Vue
753 lines
23 KiB
Vue
<template>
|
|
<AddServerModal
|
|
ref="addServerModal"
|
|
:instance="instance"
|
|
@submit="
|
|
(server, start) => {
|
|
addServer(server)
|
|
if (start) {
|
|
joinWorld(server)
|
|
}
|
|
}
|
|
"
|
|
/>
|
|
<EditServerModal ref="editServerModal" :instance="instance" @submit="editServer" />
|
|
<EditWorldModal ref="editWorldModal" :instance="instance" @submit="editWorld" />
|
|
<ConfirmModalWrapper
|
|
ref="removeServerModal"
|
|
:title="
|
|
formatMessage(messages.removeServerTitle, {
|
|
name: serverToRemove?.name ?? formatMessage(messages.thisServer),
|
|
})
|
|
"
|
|
:description="
|
|
serverToRemove?.address === serverToRemove?.name
|
|
? formatMessage(messages.removeServerDescription, { name: serverToRemove?.name })
|
|
: formatMessage(messages.removeServerDescriptionWithAddress, {
|
|
name: serverToRemove?.name,
|
|
address: serverToRemove?.address,
|
|
})
|
|
"
|
|
:markdown="false"
|
|
@proceed="proceedRemoveServer"
|
|
/>
|
|
<ConfirmModalWrapper
|
|
ref="deleteWorldModal"
|
|
:title="formatMessage(messages.deleteWorldTitle)"
|
|
:description="formatMessage(messages.deleteWorldDescription, { name: worldToDelete?.name })"
|
|
@proceed="proceedDeleteWorld"
|
|
/>
|
|
<div v-if="dedupedWorlds.length > 0" class="flex flex-col gap-4">
|
|
<div class="flex flex-wrap items-center gap-2">
|
|
<StyledInput
|
|
v-model="searchFilter"
|
|
:icon="SearchIcon"
|
|
type="text"
|
|
autocomplete="off"
|
|
:spellcheck="false"
|
|
input-class="!h-10"
|
|
wrapper-class="flex-1 min-w-0"
|
|
clearable
|
|
:placeholder="
|
|
formatMessage(messages.searchWorldsPlaceholder, { count: dedupedWorlds.length })
|
|
"
|
|
/>
|
|
<div class="flex gap-2">
|
|
<ButtonStyled type="outlined">
|
|
<button class="!h-10 !border-button-bg !border-[1px]" @click="addServerModal?.show()">
|
|
<PlusIcon class="size-5" />
|
|
{{ formatMessage(messages.addServer) }}
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled color="brand">
|
|
<button
|
|
class="!h-10 flex items-center gap-2"
|
|
@click="
|
|
router.push({ path: '/browse/server', query: { i: instance.path, from: 'worlds' } })
|
|
"
|
|
>
|
|
<CompassIcon class="size-5" />
|
|
<span>{{ formatMessage(messages.browseServers) }}</span>
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
<div class="flex flex-wrap items-center justify-between gap-2">
|
|
<div class="flex flex-wrap items-center gap-1.5">
|
|
<FilterIcon class="size-5 text-secondary" />
|
|
<button
|
|
:class="filterPillClass(selectedFilters.length === 0)"
|
|
@click="selectedFilters = []"
|
|
>
|
|
{{ formatMessage(commonMessages.allProjectType) }}
|
|
</button>
|
|
<button
|
|
v-for="option in filterOptions"
|
|
:key="option.id"
|
|
:class="filterPillClass(selectedFilters.includes(option.id))"
|
|
@click="toggleFilter(option.id)"
|
|
>
|
|
{{ option.label }}
|
|
</button>
|
|
</div>
|
|
<ButtonStyled type="transparent" hover-color-fill="none">
|
|
<button :disabled="refreshingAll" @click="refreshAllWorlds">
|
|
<RefreshCwIcon :class="refreshingAll ? 'animate-spin' : ''" />
|
|
{{ formatMessage(commonMessages.refreshButton) }}
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
<div class="flex flex-col w-full gap-2">
|
|
<WorldItem
|
|
v-for="world in filteredWorlds"
|
|
:key="`world-${world.type}-${world.type == 'singleplayer' ? world.path : `${world.address}-${world.index}`}`"
|
|
:world="world"
|
|
:managed="world.type === 'server' ? isManagedServerWorld(world) : false"
|
|
:highlighted="highlightedWorld === getWorldIdentifier(world)"
|
|
:supports-server-quick-play="supportsServerQuickPlay"
|
|
:supports-world-quick-play="supportsWorldQuickPlay"
|
|
:current-protocol="protocolVersion"
|
|
:playing-instance="playing"
|
|
:playing-world="worldsMatch(world, worldPlaying)"
|
|
:starting-instance="startingInstance"
|
|
:refreshing="world.type === 'server' ? serverData[world.address]?.refreshing : undefined"
|
|
:server-status="world.type === 'server' ? serverData[world.address]?.status : undefined"
|
|
:rendered-motd="
|
|
world.type === 'server' ? serverData[world.address]?.renderedMotd : undefined
|
|
"
|
|
:game-mode="world.type === 'singleplayer' ? GAME_MODES[world.game_mode] : undefined"
|
|
@play="() => joinWorld(world)"
|
|
@stop="() => emit('stop')"
|
|
@refresh="() => refreshServer((world as ServerWorld).address)"
|
|
@edit="
|
|
() =>
|
|
world.type === 'singleplayer'
|
|
? editWorldModal?.show(world)
|
|
: isManagedServerWorld(world)
|
|
? undefined
|
|
: editServerModal?.show(world)
|
|
"
|
|
@delete="() => !isManagedServerWorld(world) && promptToRemoveWorld(world)"
|
|
@open-folder="(world: SingleplayerWorld) => showWorldInFolder(instance.path, world.path)"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<EmptyState
|
|
v-else
|
|
type="empty-inbox"
|
|
:heading="formatMessage(messages.noWorldsHeading)"
|
|
:description="formatMessage(messages.noWorldsDescription)"
|
|
>
|
|
<template #actions>
|
|
<ButtonStyled type="outlined">
|
|
<button class="!h-10 !border-button-bg !border-[1px]" @click="addServerModal?.show()">
|
|
<PlusIcon class="size-5" />
|
|
{{ formatMessage(messages.addServer) }}
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled color="brand">
|
|
<button
|
|
class="!h-10 flex items-center gap-2"
|
|
@click="
|
|
router.push({ path: '/browse/server', query: { i: instance.path, from: 'worlds' } })
|
|
"
|
|
>
|
|
<CompassIcon class="size-5" />
|
|
<span>{{ formatMessage(messages.browseServers) }}</span>
|
|
</button>
|
|
</ButtonStyled>
|
|
</template>
|
|
</EmptyState>
|
|
</template>
|
|
<script setup lang="ts">
|
|
import { CompassIcon, FilterIcon, PlusIcon, RefreshCwIcon, SearchIcon } from '@modrinth/assets'
|
|
import {
|
|
ButtonStyled,
|
|
commonMessages,
|
|
defineMessages,
|
|
EmptyState,
|
|
GAME_MODES,
|
|
type GameVersion,
|
|
injectNotificationManager,
|
|
StyledInput,
|
|
useVIntl,
|
|
} from '@modrinth/ui'
|
|
import { platform } from '@tauri-apps/plugin-os'
|
|
import { computed, onUnmounted, ref, watch } from 'vue'
|
|
import { useRoute, useRouter } from 'vue-router'
|
|
|
|
import type ContextMenu from '@/components/ui/ContextMenu.vue'
|
|
import ConfirmModalWrapper from '@/components/ui/modal/ConfirmModalWrapper.vue'
|
|
import AddServerModal from '@/components/ui/world/modal/AddServerModal.vue'
|
|
import EditServerModal from '@/components/ui/world/modal/EditServerModal.vue'
|
|
import EditWorldModal from '@/components/ui/world/modal/EditSingleplayerWorldModal.vue'
|
|
import WorldItem from '@/components/ui/world/WorldItem.vue'
|
|
import { trackEvent } from '@/helpers/analytics'
|
|
import { get_project, get_project_v3 } from '@/helpers/cache.js'
|
|
import { profile_listener } from '@/helpers/events'
|
|
import { get_game_versions } from '@/helpers/tags'
|
|
import type { GameInstance } from '@/helpers/types'
|
|
import {
|
|
delete_world,
|
|
get_profile_protocol_version,
|
|
getServerDomainKey,
|
|
getWorldIdentifier,
|
|
handleDefaultProfileUpdateEvent,
|
|
hasServerQuickPlaySupport,
|
|
hasWorldQuickPlaySupport,
|
|
normalizeServerAddress,
|
|
type ProfileEvent,
|
|
type ProtocolVersion,
|
|
refreshServerData,
|
|
refreshServers,
|
|
refreshWorld,
|
|
refreshWorlds,
|
|
remove_server_from_profile,
|
|
resolveManagedServerWorld,
|
|
type ServerData,
|
|
type ServerWorld,
|
|
showWorldInFolder,
|
|
type SingleplayerWorld,
|
|
sortWorlds,
|
|
start_join_server,
|
|
start_join_singleplayer_world,
|
|
type World,
|
|
} from '@/helpers/worlds.ts'
|
|
import { injectServerInstall } from '@/providers/server-install'
|
|
import { handleSevereError } from '@/store/error.js'
|
|
import { ensureManagedServerWorldExists, getServerAddress } from '@/store/install'
|
|
|
|
const messages = defineMessages({
|
|
removeServerTitle: {
|
|
id: 'app.instance.worlds.remove-server-title',
|
|
defaultMessage: 'Are you sure you want to remove {name}?',
|
|
},
|
|
removeServerDescription: {
|
|
id: 'app.instance.worlds.remove-server-description',
|
|
defaultMessage:
|
|
"'{name}' will be removed from your list, including in-game, and there will be no way to recover it.",
|
|
},
|
|
removeServerDescriptionWithAddress: {
|
|
id: 'app.instance.worlds.remove-server-description-with-address',
|
|
defaultMessage:
|
|
"'{name}' ({address}) will be removed from your list, including in-game, and there will be no way to recover it.",
|
|
},
|
|
deleteWorldTitle: {
|
|
id: 'app.instance.worlds.delete-world-title',
|
|
defaultMessage: 'Are you sure you want to permanently delete this world?',
|
|
},
|
|
deleteWorldDescription: {
|
|
id: 'app.instance.worlds.delete-world-description',
|
|
defaultMessage:
|
|
"'{name}' will be **permanently deleted**, and there will be no way to recover it.",
|
|
},
|
|
searchWorldsPlaceholder: {
|
|
id: 'app.instance.worlds.search-worlds-placeholder',
|
|
defaultMessage: 'Search {count} worlds...',
|
|
},
|
|
addServer: {
|
|
id: 'app.instance.worlds.add-server',
|
|
defaultMessage: 'Add server',
|
|
},
|
|
browseServers: {
|
|
id: 'app.instance.worlds.browse-servers',
|
|
defaultMessage: 'Browse servers',
|
|
},
|
|
noWorldsHeading: {
|
|
id: 'app.instance.worlds.no-worlds-heading',
|
|
defaultMessage: 'No servers or worlds added',
|
|
},
|
|
noWorldsDescription: {
|
|
id: 'app.instance.worlds.no-worlds-description',
|
|
defaultMessage: 'Add a server or browse to get started',
|
|
},
|
|
thisServer: {
|
|
id: 'app.instance.worlds.this-server',
|
|
defaultMessage: 'this server',
|
|
},
|
|
vanillaFilter: {
|
|
id: 'app.instance.worlds.filter-vanilla',
|
|
defaultMessage: 'Vanilla',
|
|
},
|
|
moddedFilter: {
|
|
id: 'app.instance.worlds.filter-modded',
|
|
defaultMessage: 'Modded',
|
|
},
|
|
onlineFilter: {
|
|
id: 'app.instance.worlds.filter-online',
|
|
defaultMessage: 'Online',
|
|
},
|
|
offlineFilter: {
|
|
id: 'app.instance.worlds.filter-offline',
|
|
defaultMessage: 'Offline',
|
|
},
|
|
})
|
|
|
|
const { formatMessage } = useVIntl()
|
|
const { handleError } = injectNotificationManager()
|
|
const { playServerProject } = injectServerInstall()
|
|
const route = useRoute()
|
|
const router = useRouter()
|
|
|
|
const addServerModal = ref<InstanceType<typeof AddServerModal>>()
|
|
const editServerModal = ref<InstanceType<typeof EditServerModal>>()
|
|
const editWorldModal = ref<InstanceType<typeof EditWorldModal>>()
|
|
const removeServerModal = ref<InstanceType<typeof ConfirmModalWrapper>>()
|
|
const deleteWorldModal = ref<InstanceType<typeof ConfirmModalWrapper>>()
|
|
|
|
const serverToRemove = ref<ServerWorld>()
|
|
const worldToDelete = ref<SingleplayerWorld>()
|
|
|
|
const emit = defineEmits<{
|
|
(event: 'play', world: World): void
|
|
(event: 'stop'): void
|
|
}>()
|
|
|
|
const props = defineProps<{
|
|
instance: GameInstance
|
|
options: InstanceType<typeof ContextMenu> | null
|
|
offline: boolean
|
|
playing: boolean
|
|
installed: boolean
|
|
}>()
|
|
|
|
const instance = computed(() => props.instance)
|
|
const playing = computed(() => props.playing)
|
|
|
|
function play(world: World) {
|
|
emit('play', world)
|
|
}
|
|
|
|
const selectedFilters = ref<string[]>([])
|
|
const searchFilter = ref('')
|
|
|
|
function filterPillClass(isActive: boolean) {
|
|
return [
|
|
'cursor-pointer rounded-full border border-solid px-3 py-1.5 text-base font-semibold leading-5 transition-all duration-100 active:scale-[0.97]',
|
|
isActive
|
|
? 'border-green bg-brand-highlight text-brand'
|
|
: 'border-surface-5 bg-surface-4 text-primary hover:bg-surface-5',
|
|
]
|
|
}
|
|
|
|
function toggleFilter(id: string) {
|
|
const idx = selectedFilters.value.indexOf(id)
|
|
if (idx >= 0) {
|
|
selectedFilters.value.splice(idx, 1)
|
|
} else {
|
|
selectedFilters.value.push(id)
|
|
if (id === 'singleplayer') {
|
|
selectedFilters.value = selectedFilters.value.filter((f) => f !== 'online' && f !== 'offline')
|
|
} else if (id === 'online' || id === 'offline') {
|
|
selectedFilters.value = selectedFilters.value.filter((f) => f !== 'singleplayer')
|
|
}
|
|
}
|
|
}
|
|
|
|
const refreshingAll = ref(false)
|
|
const hadNoWorlds = ref(true)
|
|
const startingInstance = ref(false)
|
|
const worldPlaying = ref<World>()
|
|
|
|
const worlds = ref<World[]>([])
|
|
const serverData = ref<Record<string, ServerData>>({})
|
|
|
|
// Track servers_updated calls on Linux to prevent server ping spam
|
|
const MAX_LINUX_REFRESHES = 3
|
|
const isLinux = platform() === 'linux'
|
|
const linuxRefreshCount = ref(0)
|
|
|
|
const protocolVersion = ref<ProtocolVersion | null>(
|
|
await get_profile_protocol_version(instance.value.path),
|
|
)
|
|
const managedServerName = ref<string | null>(null)
|
|
const managedServerAddress = ref<string | null>(null)
|
|
|
|
const managedServerWorld = computed(() =>
|
|
resolveManagedServerWorld(worlds.value, managedServerName.value, managedServerAddress.value),
|
|
)
|
|
|
|
function isManagedServerWorld(world: World): world is ServerWorld {
|
|
return world.type === 'server' && managedServerWorld.value?.index === world.index
|
|
}
|
|
|
|
async function refreshManagedServerMetadata() {
|
|
await ensureManagedServerWorldExists(
|
|
instance.value.path,
|
|
managedServerName.value,
|
|
managedServerAddress.value,
|
|
)
|
|
|
|
const projectId = instance.value.linked_data?.project_id
|
|
if (!projectId) {
|
|
managedServerName.value = null
|
|
managedServerAddress.value = null
|
|
return
|
|
}
|
|
|
|
try {
|
|
const [project, projectV3] = await Promise.all([
|
|
get_project(projectId, 'bypass'),
|
|
get_project_v3(projectId, 'bypass'),
|
|
])
|
|
|
|
if (projectV3?.minecraft_server == null) {
|
|
managedServerName.value = null
|
|
managedServerAddress.value = null
|
|
return
|
|
}
|
|
|
|
const serverAddress = getServerAddress(projectV3.minecraft_java_server)
|
|
if (!serverAddress) {
|
|
managedServerName.value = null
|
|
managedServerAddress.value = null
|
|
return
|
|
}
|
|
|
|
managedServerName.value = project.title
|
|
managedServerAddress.value = serverAddress
|
|
} catch (err) {
|
|
console.error(
|
|
`Failed to resolve managed server metadata for profile: ${instance.value.path}`,
|
|
err,
|
|
)
|
|
managedServerName.value = null
|
|
managedServerAddress.value = null
|
|
}
|
|
}
|
|
|
|
watch(
|
|
() => instance.value.linked_data?.project_id,
|
|
async () => {
|
|
await refreshManagedServerMetadata()
|
|
},
|
|
{ immediate: true },
|
|
)
|
|
|
|
const unlistenProfile = await profile_listener(async (e: ProfileEvent) => {
|
|
if (e.profile_path_id !== instance.value.path) return
|
|
|
|
console.info(`Handling profile event '${e.event}' for profile: ${e.profile_path_id}`)
|
|
|
|
if (e.event === 'servers_updated') {
|
|
if (isLinux && linuxRefreshCount.value >= MAX_LINUX_REFRESHES) return
|
|
if (isLinux) linuxRefreshCount.value++
|
|
|
|
await refreshAllWorlds()
|
|
}
|
|
|
|
await handleDefaultProfileUpdateEvent(worlds.value, instance.value.path, e)
|
|
})
|
|
|
|
await refreshAllWorlds()
|
|
|
|
async function refreshServer(address: string) {
|
|
if (!serverData.value[address]) {
|
|
serverData.value[address] = {
|
|
refreshing: true,
|
|
}
|
|
}
|
|
await refreshServerData(serverData.value[address], protocolVersion.value, address)
|
|
}
|
|
|
|
async function refreshAllWorlds() {
|
|
if (refreshingAll.value) {
|
|
console.log(`Already refreshing, cancelling refresh.`)
|
|
return
|
|
}
|
|
await refreshManagedServerMetadata()
|
|
|
|
refreshingAll.value = true
|
|
|
|
worlds.value = await refreshWorlds(instance.value.path).finally(
|
|
() => (refreshingAll.value = false),
|
|
)
|
|
refreshServers(worlds.value, serverData.value, protocolVersion.value)
|
|
|
|
const hasNoWorlds = worlds.value.length === 0
|
|
|
|
if (hadNoWorlds.value && hasNoWorlds) {
|
|
setTimeout(() => {
|
|
refreshingAll.value = false
|
|
}, 1000)
|
|
} else {
|
|
refreshingAll.value = false
|
|
}
|
|
|
|
hadNoWorlds.value = hasNoWorlds
|
|
}
|
|
|
|
async function addServer(server: ServerWorld) {
|
|
worlds.value.push(server)
|
|
sortWorlds(worlds.value)
|
|
await refreshServer(server.address)
|
|
}
|
|
|
|
async function editServer(server: ServerWorld) {
|
|
const index = worlds.value.findIndex((w) => w.type === 'server' && w.index === server.index)
|
|
if (index !== -1) {
|
|
const oldServer = worlds.value[index] as ServerWorld
|
|
worlds.value[index] = server
|
|
sortWorlds(worlds.value)
|
|
if (oldServer.address !== server.address) {
|
|
await refreshServer(server.address)
|
|
}
|
|
} else {
|
|
handleError(new Error(`Error refreshing server, refreshing all worlds`))
|
|
await refreshAllWorlds()
|
|
}
|
|
}
|
|
|
|
async function removeServer(server: ServerWorld) {
|
|
await remove_server_from_profile(instance.value.path, server.index).catch(handleError)
|
|
worlds.value = worlds.value.filter((w) => w.type !== 'server' || w.index !== server.index)
|
|
let serverIdx = 0
|
|
for (const w of worlds.value) {
|
|
if (w.type === 'server') {
|
|
w.index = serverIdx++
|
|
}
|
|
}
|
|
}
|
|
|
|
async function editWorld(path: string, name: string, removeIcon: boolean) {
|
|
const world = worlds.value.find((world) => world.type === 'singleplayer' && world.path === path)
|
|
if (world) {
|
|
world.name = name
|
|
if (removeIcon) {
|
|
world.icon = undefined
|
|
}
|
|
sortWorlds(worlds.value)
|
|
} else {
|
|
handleError(new Error(`Error finding world in list, refreshing all worlds`))
|
|
await refreshAllWorlds()
|
|
}
|
|
}
|
|
|
|
async function deleteWorld(world: SingleplayerWorld) {
|
|
await delete_world(instance.value.path, world.path).catch(handleError)
|
|
worlds.value = worlds.value.filter((w) => w.type !== 'singleplayer' || w.path !== world.path)
|
|
}
|
|
|
|
function handleJoinError(err: Error) {
|
|
handleSevereError(err, { profilePath: instance.value.path })
|
|
startingInstance.value = false
|
|
worldPlaying.value = undefined
|
|
}
|
|
|
|
async function joinWorld(world: World) {
|
|
console.log(`Joining world ${getWorldIdentifier(world)}`)
|
|
startingInstance.value = true
|
|
worldPlaying.value = world
|
|
if (world.type === 'server') {
|
|
const managedProjectId = instance.value.linked_data?.project_id
|
|
if (managedProjectId && isManagedServerWorld(world)) {
|
|
await playServerProject(managedProjectId).catch(handleJoinError)
|
|
trackEvent('InstanceStart', {
|
|
loader: instance.value.loader,
|
|
game_version: instance.value.game_version,
|
|
source: 'WorldsPage',
|
|
})
|
|
startingInstance.value = false
|
|
return
|
|
}
|
|
await start_join_server(instance.value.path, world.address).catch(handleJoinError)
|
|
trackEvent('InstanceStart', {
|
|
loader: instance.value.loader,
|
|
game_version: instance.value.game_version,
|
|
source: 'WorldsPage',
|
|
})
|
|
} else if (world.type === 'singleplayer') {
|
|
await start_join_singleplayer_world(instance.value.path, world.path).catch(handleJoinError)
|
|
}
|
|
play(world)
|
|
startingInstance.value = false
|
|
}
|
|
|
|
watch(
|
|
() => playing.value,
|
|
(playing) => {
|
|
if (!playing) {
|
|
worldPlaying.value = undefined
|
|
|
|
setTimeout(async () => {
|
|
for (const world of worlds.value) {
|
|
if (world.type === 'singleplayer' && world.locked) {
|
|
await refreshWorld(worlds.value, instance.value.path, world.path)
|
|
}
|
|
}
|
|
}, 1000)
|
|
}
|
|
},
|
|
)
|
|
|
|
function worldsMatch(world: World, other: World | undefined) {
|
|
if (world.type === 'server' && other?.type === 'server') {
|
|
return world.address === other.address
|
|
} else if (world.type === 'singleplayer' && other?.type === 'singleplayer') {
|
|
return world.path === other.path
|
|
}
|
|
return false
|
|
}
|
|
|
|
const gameVersions = ref<GameVersion[]>(await get_game_versions().catch(() => []))
|
|
const supportsServerQuickPlay = computed(() =>
|
|
hasServerQuickPlaySupport(gameVersions.value, instance.value.game_version),
|
|
)
|
|
const supportsWorldQuickPlay = computed(() =>
|
|
hasWorldQuickPlaySupport(gameVersions.value, instance.value.game_version),
|
|
)
|
|
|
|
const dedupedWorlds = computed(() => {
|
|
const visibleWorlds: World[] = []
|
|
const serverIndexByDomain = new Map<string, number>()
|
|
|
|
for (const world of worlds.value) {
|
|
if (world.type !== 'server') {
|
|
visibleWorlds.push(world)
|
|
continue
|
|
}
|
|
|
|
const domainKey =
|
|
getServerDomainKey(world.address) ||
|
|
normalizeServerAddress(world.address) ||
|
|
`server-${world.index}`
|
|
const existingIndex = serverIndexByDomain.get(domainKey)
|
|
|
|
if (existingIndex == null) {
|
|
serverIndexByDomain.set(domainKey, visibleWorlds.length)
|
|
visibleWorlds.push(world)
|
|
continue
|
|
}
|
|
|
|
// replace world with managed world if applicable
|
|
const existingWorld = visibleWorlds[existingIndex]
|
|
if (
|
|
existingWorld?.type === 'server' &&
|
|
!isManagedServerWorld(existingWorld) &&
|
|
isManagedServerWorld(world)
|
|
) {
|
|
visibleWorlds[existingIndex] = world
|
|
}
|
|
}
|
|
|
|
return visibleWorlds
|
|
})
|
|
|
|
const filterOptions = computed(() => {
|
|
const options: { id: string; label: string }[] = []
|
|
const hasSingleplayer = dedupedWorlds.value.some((x) => x.type === 'singleplayer')
|
|
const hasServer = dedupedWorlds.value.some((x) => x.type === 'server')
|
|
|
|
const hasStatusFilter =
|
|
selectedFilters.value.includes('online') || selectedFilters.value.includes('offline')
|
|
|
|
if (hasSingleplayer && hasServer && !hasStatusFilter) {
|
|
options.push({ id: 'singleplayer', label: formatMessage(commonMessages.singleplayerLabel) })
|
|
}
|
|
|
|
if (hasServer) {
|
|
const servers = dedupedWorlds.value.filter((x) => x.type === 'server')
|
|
const hasVanilla = servers.some((x) => x.content_kind !== 'modpack')
|
|
const hasModded = servers.some((x) => x.content_kind === 'modpack')
|
|
if (hasVanilla && hasModded) {
|
|
options.push({ id: 'vanilla', label: formatMessage(messages.vanillaFilter) })
|
|
options.push({ id: 'modded', label: formatMessage(messages.moddedFilter) })
|
|
}
|
|
if (!selectedFilters.value.includes('singleplayer')) {
|
|
const hasOnline = servers.some((x) => !!serverData.value[x.address]?.status)
|
|
const hasOffline = servers.some((x) => !serverData.value[x.address]?.status)
|
|
if (hasOnline && hasOffline) {
|
|
options.push({ id: 'online', label: formatMessage(messages.onlineFilter) })
|
|
options.push({ id: 'offline', label: formatMessage(messages.offlineFilter) })
|
|
}
|
|
}
|
|
}
|
|
|
|
return options
|
|
})
|
|
|
|
watch(filterOptions, (options) => {
|
|
const validIds = new Set(options.map((opt) => opt.id))
|
|
const cleaned = selectedFilters.value.filter((f) => validIds.has(f))
|
|
if (cleaned.length !== selectedFilters.value.length) {
|
|
selectedFilters.value = cleaned
|
|
}
|
|
})
|
|
|
|
const filteredWorlds = computed(() =>
|
|
dedupedWorlds.value.filter((x) => {
|
|
if (searchFilter.value && !x.name.toLowerCase().includes(searchFilter.value.toLowerCase())) {
|
|
return false
|
|
}
|
|
|
|
if (selectedFilters.value.length === 0) return true
|
|
|
|
const hasSingleplayerFilter = selectedFilters.value.includes('singleplayer')
|
|
const typeFilters = selectedFilters.value.filter((f) => f === 'vanilla' || f === 'modded')
|
|
const statusFilters = selectedFilters.value.filter((f) => f === 'online' || f === 'offline')
|
|
|
|
if (x.type === 'singleplayer') {
|
|
return hasSingleplayerFilter || (typeFilters.length === 0 && statusFilters.length === 0)
|
|
}
|
|
|
|
if (hasSingleplayerFilter && typeFilters.length === 0 && statusFilters.length === 0) {
|
|
return false
|
|
}
|
|
|
|
let passesType = true
|
|
if (typeFilters.length > 0) {
|
|
const isModded = x.content_kind === 'modpack'
|
|
passesType =
|
|
(typeFilters.includes('modded') && isModded) ||
|
|
(typeFilters.includes('vanilla') && !isModded)
|
|
}
|
|
|
|
let passesStatus = true
|
|
if (statusFilters.length > 0) {
|
|
const isOnline = !!serverData.value[x.address]?.status
|
|
passesStatus =
|
|
(statusFilters.includes('online') && isOnline) ||
|
|
(statusFilters.includes('offline') && !isOnline)
|
|
}
|
|
|
|
return passesType && passesStatus
|
|
}),
|
|
)
|
|
|
|
const highlightedWorld = ref(route.query.highlight)
|
|
|
|
function promptToRemoveWorld(world: World): boolean {
|
|
if (world.type === 'server') {
|
|
serverToRemove.value = world
|
|
removeServerModal.value?.show()
|
|
return !!removeServerModal.value
|
|
} else {
|
|
worldToDelete.value = world
|
|
deleteWorldModal.value?.show()
|
|
return !!deleteWorldModal.value
|
|
}
|
|
}
|
|
|
|
async function proceedRemoveServer() {
|
|
if (!serverToRemove.value) {
|
|
handleError(new Error(`Error removing server, no server marked for removal.`))
|
|
return
|
|
}
|
|
await removeServer(serverToRemove.value)
|
|
serverToRemove.value = undefined
|
|
}
|
|
|
|
async function proceedDeleteWorld() {
|
|
if (!worldToDelete.value) {
|
|
handleError(new Error(`Error deleting world, no world marked for removal.`))
|
|
return
|
|
}
|
|
await deleteWorld(worldToDelete.value)
|
|
worldToDelete.value = undefined
|
|
}
|
|
|
|
onUnmounted(() => {
|
|
unlistenProfile()
|
|
})
|
|
</script>
|