You've already forked AstralRinth
forked from didirus/AstralRinth
* feat: init selecting paper+purpur on purchase flow Signed-off-by: Evan Song <theevansong@gmail.com> * feat: properly implement Paper/Purpur in Platform Signed-off-by: Evan Song <theevansong@gmail.com> * chore: correct wording Signed-off-by: Evan Song <theevansong@gmail.com> * feat: redo platform modal Signed-off-by: Evan Song <theevansong@gmail.com> * Switch to HCaptcha for Auth-related captchas (#2945) * Switch to HCaptcha for Auth-related captchas * run fmt * fix hcaptcha not loading * fix: more robust loader dropdown logic Signed-off-by: Evan Song <theevansong@gmail.com> * fix: handle "not yet supported" install err Signed-off-by: Evan Song <theevansong@gmail.com> * chore: fix icon kerfuffles Signed-off-by: Evan Song <theevansong@gmail.com> * chore: improve vanilla install modal title Signed-off-by: Evan Song <theevansong@gmail.com> * fix: spacing Signed-off-by: Evan Song <theevansong@gmail.com> * chore: improve no loader state Signed-off-by: Evan Song <theevansong@gmail.com> * fix: type error Signed-off-by: Evan Song <theevansong@gmail.com> * chore: adjust mod version modal title Signed-off-by: Evan Song <theevansong@gmail.com> * chore: adjust modpack warning copy Signed-off-by: Evan Song <theevansong@gmail.com> * feat: vanilla empty state in content page Signed-off-by: Evan Song <theevansong@gmail.com> * chore: adjust copy Signed-off-by: Evan Song <theevansong@gmail.com> * chore: update icon Signed-off-by: Evan Song <theevansong@gmail.com> * fix: loader type Signed-off-by: Evan Song <theevansong@gmail.com> * fix: loader type Signed-off-by: Evan Song <theevansong@gmail.com> * feat: always show dropdown if possible Signed-off-by: Evan Song <theevansong@gmail.com> * chore: improve spacing Signed-off-by: Evan Song <theevansong@gmail.com> * chore: appear disabled Signed-off-by: Evan Song <theevansong@gmail.com> * h Signed-off-by: Evan Song <theevansong@gmail.com> * chore: if reinstalling, show it on the modal title Signed-off-by: Evan Song <theevansong@gmail.com> * feat: put it in the dropdown, they said Signed-off-by: Evan Song <theevansong@gmail.com> * chore: adjust style Signed-off-by: Evan Song <theevansong@gmail.com> * chore: sort paper-purpur versions desc Signed-off-by: Evan Song <theevansong@gmail.com> * fix: do not consider backup limit in reinstall prompt Signed-off-by: Evan Song <theevansong@gmail.com> * feat: backup locking, plugin support * fix: content type error Signed-off-by: Evan Song <theevansong@gmail.com> * fix: casing Signed-off-by: Evan Song <theevansong@gmail.com> * fix: plugins pt 2 * feat: backups, mrpack * fix: type errors come on Signed-off-by: Evan Song <theevansong@gmail.com> * fix: spacing Signed-off-by: Evan Song <theevansong@gmail.com> * fix: type maxing * chore: show copy button on allocation rows Signed-off-by: Evan Song <theevansong@gmail.com> * feat: suspend improvement --------- Signed-off-by: Evan Song <theevansong@gmail.com> Co-authored-by: Evan Song <theevansong@gmail.com> Co-authored-by: Geometrically <18202329+Geometrically@users.noreply.github.com> Co-authored-by: Jai A <jaiagr+gpg@pm.me> Co-authored-by: Evan Song <52982404+ferothefox@users.noreply.github.com>
117 lines
3.8 KiB
Vue
117 lines
3.8 KiB
Vue
<template>
|
|
<NuxtLink
|
|
class="contents"
|
|
:to="status === 'suspended' ? '' : `/servers/manage/${props.server_id}`"
|
|
>
|
|
<div
|
|
v-tooltip="
|
|
status === 'suspended'
|
|
? `This server is suspended visit the billing page to learn more`
|
|
: ''
|
|
"
|
|
class="flex cursor-pointer flex-row items-center overflow-x-hidden rounded-3xl bg-bg-raised p-4 transition-transform duration-100"
|
|
:class="status === 'suspended' ? 'opacity-50' : 'active:scale-95'"
|
|
data-pyro-server-listing
|
|
:data-pyro-server-listing-id="server_id"
|
|
>
|
|
<UiServersServerIcon v-if="status !== 'suspended'" :image="image" />
|
|
<div
|
|
v-else
|
|
class="bg-bg-secondary flex size-24 items-center justify-center rounded-xl border-[1px] border-solid border-button-border bg-button-bg shadow-sm"
|
|
>
|
|
<LockIcon class="size-20 text-secondary" />
|
|
</div>
|
|
<div class="ml-8 flex flex-col gap-2.5">
|
|
<div class="flex flex-row items-center gap-2">
|
|
<h2 class="m-0 text-xl font-bold text-contrast">{{ name }}</h2>
|
|
<ChevronRightIcon />
|
|
</div>
|
|
|
|
<div
|
|
v-if="projectData?.title"
|
|
class="m-0 flex flex-row items-center gap-2 text-sm font-medium text-secondary"
|
|
>
|
|
<UiAvatar
|
|
:src="iconUrl"
|
|
no-shadow
|
|
style="min-height: 20px; min-width: 20px; height: 20px; width: 20px"
|
|
alt="Server Icon"
|
|
/>
|
|
Using {{ projectData?.title || "Unknown" }}
|
|
</div>
|
|
<div v-else class="min-h-[20px]"></div>
|
|
|
|
<UiServersServerInfoLabels
|
|
:server-data="{ game, mc_version, loader, loader_version, net }"
|
|
:show-game-label="showGameLabel"
|
|
:show-loader-label="showLoaderLabel"
|
|
:show-subdomain-label="showSubdomainLabel"
|
|
:linked="false"
|
|
class="pointer-events-none flex w-full flex-row flex-wrap items-center gap-4 text-secondary *:hidden sm:flex-row sm:*:flex"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</NuxtLink>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ChevronRightIcon, LockIcon } from "@modrinth/assets";
|
|
import type { Project, Server } from "~/types/servers";
|
|
|
|
const props = defineProps<Partial<Server>>();
|
|
|
|
const showGameLabel = computed(() => !!props.game);
|
|
const showLoaderLabel = computed(() => !!props.loader);
|
|
const showSubdomainLabel = computed(() => !!props.net?.domain);
|
|
|
|
let projectData: Ref<Project | null>;
|
|
if (props.upstream) {
|
|
const { data } = await useAsyncData<Project>(
|
|
`server-project-${props.server_id}`,
|
|
async (): Promise<Project> => {
|
|
const result = await useBaseFetch(`project/${props.upstream?.project_id}`);
|
|
return result as Project;
|
|
},
|
|
);
|
|
projectData = data;
|
|
} else {
|
|
projectData = ref(null);
|
|
}
|
|
|
|
const image = ref<string | undefined>();
|
|
|
|
onMounted(async () => {
|
|
const auth = (await usePyroFetch(`servers/${props.server_id}/fs`)) as any;
|
|
try {
|
|
const fileData = await usePyroFetch(`/download?path=/server-icon-original.png`, {
|
|
override: auth,
|
|
});
|
|
|
|
if (fileData instanceof Blob) {
|
|
const canvas = document.createElement("canvas");
|
|
const ctx = canvas.getContext("2d");
|
|
const img = new Image();
|
|
img.src = URL.createObjectURL(fileData);
|
|
await new Promise<void>((resolve) => {
|
|
img.onload = () => {
|
|
canvas.width = 512;
|
|
canvas.height = 512;
|
|
ctx?.drawImage(img, 0, 0, 512, 512);
|
|
const dataURL = canvas.toDataURL("image/png");
|
|
image.value = dataURL;
|
|
resolve();
|
|
};
|
|
});
|
|
}
|
|
} catch (error) {
|
|
if (error instanceof PyroFetchError && error.statusCode === 404) {
|
|
image.value = undefined;
|
|
} else {
|
|
console.error(error);
|
|
}
|
|
}
|
|
});
|
|
|
|
const iconUrl = computed(() => projectData.value?.icon_url || undefined);
|
|
</script>
|