2022-06-20 12:49:51 +02:00
|
|
|
<template>
|
2023-04-08 02:01:42 +02:00
|
|
|
<MkStickyContainer>
|
|
|
|
<template #header
|
|
|
|
><MkPageHeader
|
|
|
|
v-model:tab="tab"
|
|
|
|
:actions="headerActions"
|
|
|
|
:tabs="headerTabs"
|
|
|
|
/></template>
|
|
|
|
<MkSpacer
|
|
|
|
v-if="instance"
|
|
|
|
:content-max="600"
|
|
|
|
:margin-min="16"
|
|
|
|
:margin-max="32"
|
2022-09-10 00:42:17 +02:00
|
|
|
>
|
2023-05-26 04:47:10 +02:00
|
|
|
<swiper
|
2023-05-26 23:02:17 +02:00
|
|
|
:round-lengths="true"
|
2023-05-26 04:47:10 +02:00
|
|
|
:touch-angle="25"
|
|
|
|
:threshold="10"
|
2023-09-02 01:27:33 +02:00
|
|
|
:centered-slides="true"
|
2023-04-08 02:01:42 +02:00
|
|
|
:modules="[Virtual]"
|
|
|
|
:space-between="20"
|
|
|
|
:virtual="true"
|
2023-06-26 21:47:05 +02:00
|
|
|
:allow-touch-move="
|
|
|
|
defaultStore.state.swipeOnMobile &&
|
|
|
|
(deviceKind !== 'desktop' ||
|
|
|
|
defaultStore.state.swipeOnDesktop)
|
|
|
|
"
|
2023-04-08 02:01:42 +02:00
|
|
|
@swiper="setSwiperRef"
|
|
|
|
@slide-change="onSlideChange"
|
|
|
|
>
|
|
|
|
<swiper-slide>
|
|
|
|
<div class="_formRoot">
|
|
|
|
<div class="fnfelxur">
|
|
|
|
<img :src="faviconUrl" alt="" class="icon" />
|
|
|
|
<span class="name">{{
|
|
|
|
instance.name || `(${i18n.ts.unknown})`
|
|
|
|
}}</span>
|
|
|
|
</div>
|
|
|
|
<MkKeyValue :copy="host" oneline style="margin: 1em 0">
|
|
|
|
<template #key>Host</template>
|
|
|
|
<template #value
|
|
|
|
><span class="_monospace"
|
|
|
|
><MkLink :url="`https://${host}`">{{
|
|
|
|
host
|
|
|
|
}}</MkLink></span
|
|
|
|
></template
|
|
|
|
>
|
2022-09-10 00:42:17 +02:00
|
|
|
</MkKeyValue>
|
2023-04-08 02:01:42 +02:00
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{ i18n.ts.software }}</template>
|
|
|
|
<template #value
|
|
|
|
><span class="_monospace"
|
|
|
|
>{{
|
|
|
|
instance.softwareName ||
|
|
|
|
`(${i18n.ts.unknown})`
|
|
|
|
}}
|
|
|
|
/
|
|
|
|
{{
|
|
|
|
instance.softwareVersion ||
|
|
|
|
`(${i18n.ts.unknown})`
|
|
|
|
}}</span
|
|
|
|
></template
|
|
|
|
>
|
2022-09-10 00:42:17 +02:00
|
|
|
</MkKeyValue>
|
2023-04-08 02:01:42 +02:00
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{
|
|
|
|
i18n.ts.administrator
|
|
|
|
}}</template>
|
|
|
|
<template #value
|
|
|
|
>{{
|
|
|
|
instance.maintainerName ||
|
|
|
|
`(${i18n.ts.unknown})`
|
|
|
|
}}
|
|
|
|
({{
|
|
|
|
instance.maintainerEmail ||
|
|
|
|
`(${i18n.ts.unknown})`
|
|
|
|
}})</template
|
|
|
|
>
|
2022-09-10 00:42:17 +02:00
|
|
|
</MkKeyValue>
|
2023-04-08 02:01:42 +02:00
|
|
|
<MkKeyValue>
|
|
|
|
<template #key>{{ i18n.ts.description }}</template>
|
|
|
|
<template #value>{{
|
|
|
|
instance.description
|
|
|
|
}}</template>
|
2022-09-10 00:42:17 +02:00
|
|
|
</MkKeyValue>
|
2022-09-15 22:40:34 +02:00
|
|
|
|
2024-01-11 02:21:21 +01:00
|
|
|
<FormSection v-if="isAdmin">
|
2023-04-08 02:01:42 +02:00
|
|
|
<template #label>Moderation</template>
|
|
|
|
<FormSuspense :p="init">
|
|
|
|
<FormSwitch
|
|
|
|
v-model="suspended"
|
|
|
|
class="_formBlock"
|
|
|
|
@update:modelValue="toggleSuspend"
|
|
|
|
>{{
|
|
|
|
i18n.ts.stopActivityDelivery
|
|
|
|
}}</FormSwitch
|
|
|
|
>
|
|
|
|
<FormSwitch
|
|
|
|
v-model="isBlocked"
|
|
|
|
class="_formBlock"
|
|
|
|
@update:modelValue="toggleBlock"
|
|
|
|
>{{ i18n.ts.blockThisInstance }}</FormSwitch
|
|
|
|
>
|
2023-04-30 15:01:42 +02:00
|
|
|
<FormSwitch
|
|
|
|
v-model="isSilenced"
|
|
|
|
class="_formBlock"
|
|
|
|
@update:modelValue="toggleSilence"
|
|
|
|
>{{
|
|
|
|
i18n.ts.silenceThisInstance
|
|
|
|
}}</FormSwitch
|
|
|
|
>
|
2023-04-08 02:01:42 +02:00
|
|
|
</FormSuspense>
|
|
|
|
<MkButton @click="refreshMetadata"
|
2023-10-17 03:57:20 +02:00
|
|
|
><i :class="icon('ph-arrows-clockwise')"></i>
|
2023-04-08 02:01:42 +02:00
|
|
|
Refresh metadata</MkButton
|
|
|
|
>
|
|
|
|
</FormSection>
|
2022-09-15 22:40:34 +02:00
|
|
|
|
2023-04-08 02:01:42 +02:00
|
|
|
<FormSection>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{
|
|
|
|
i18n.ts.registeredAt
|
|
|
|
}}</template>
|
|
|
|
<template #value
|
|
|
|
><MkTime
|
|
|
|
mode="detail"
|
|
|
|
:time="instance.caughtAt"
|
|
|
|
/></template>
|
|
|
|
</MkKeyValue>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{
|
|
|
|
i18n.ts.updatedAt
|
|
|
|
}}</template>
|
|
|
|
<template #value
|
|
|
|
><MkTime
|
|
|
|
mode="detail"
|
|
|
|
:time="instance.infoUpdatedAt"
|
|
|
|
/></template>
|
|
|
|
</MkKeyValue>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{
|
|
|
|
i18n.ts.latestRequestSentAt
|
|
|
|
}}</template>
|
|
|
|
<template #value
|
|
|
|
><MkTime
|
|
|
|
v-if="instance.latestRequestSentAt"
|
|
|
|
:time="instance.latestRequestSentAt"
|
|
|
|
/><span v-else>N/A</span></template
|
|
|
|
>
|
|
|
|
</MkKeyValue>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{
|
|
|
|
i18n.ts.latestStatus
|
|
|
|
}}</template>
|
|
|
|
<template #value>{{
|
|
|
|
instance.latestStatus
|
|
|
|
? instance.latestStatus
|
|
|
|
: "N/A"
|
|
|
|
}}</template>
|
|
|
|
</MkKeyValue>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>{{
|
|
|
|
i18n.ts.latestRequestReceivedAt
|
|
|
|
}}</template>
|
|
|
|
<template #value
|
|
|
|
><MkTime
|
|
|
|
v-if="instance.latestRequestReceivedAt"
|
|
|
|
:time="instance.latestRequestReceivedAt"
|
|
|
|
/><span v-else>N/A</span></template
|
|
|
|
>
|
|
|
|
</MkKeyValue>
|
|
|
|
</FormSection>
|
|
|
|
|
|
|
|
<FormSection>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>Following (Pub)</template>
|
|
|
|
<template #value>{{
|
|
|
|
number(instance.followingCount)
|
|
|
|
}}</template>
|
|
|
|
</MkKeyValue>
|
|
|
|
<MkKeyValue oneline style="margin: 1em 0">
|
|
|
|
<template #key>Followers (Sub)</template>
|
|
|
|
<template #value>{{
|
|
|
|
number(instance.followersCount)
|
|
|
|
}}</template>
|
|
|
|
</MkKeyValue>
|
|
|
|
</FormSection>
|
|
|
|
|
|
|
|
<FormSection>
|
|
|
|
<template #label>Well-known resources</template>
|
|
|
|
<FormLink
|
|
|
|
:to="`https://${host}/.well-known/host-meta`"
|
|
|
|
external
|
|
|
|
style="margin-bottom: 8px"
|
|
|
|
>host-meta</FormLink
|
|
|
|
>
|
|
|
|
<FormLink
|
|
|
|
:to="`https://${host}/.well-known/host-meta.json`"
|
|
|
|
external
|
|
|
|
style="margin-bottom: 8px"
|
|
|
|
>host-meta.json</FormLink
|
|
|
|
>
|
|
|
|
<FormLink
|
|
|
|
:to="`https://${host}/.well-known/nodeinfo`"
|
|
|
|
external
|
|
|
|
style="margin-bottom: 8px"
|
|
|
|
>nodeinfo</FormLink
|
|
|
|
>
|
|
|
|
<FormLink
|
|
|
|
:to="`https://${host}/robots.txt`"
|
|
|
|
external
|
|
|
|
style="margin-bottom: 8px"
|
|
|
|
>robots.txt</FormLink
|
|
|
|
>
|
|
|
|
<FormLink
|
|
|
|
:to="`https://${host}/manifest.json`"
|
|
|
|
external
|
|
|
|
style="margin-bottom: 8px"
|
|
|
|
>manifest.json</FormLink
|
|
|
|
>
|
|
|
|
</FormSection>
|
|
|
|
</div>
|
|
|
|
</swiper-slide>
|
|
|
|
<swiper-slide>
|
|
|
|
<div class="_formRoot">
|
|
|
|
<MkPagination
|
|
|
|
v-slot="{ items }"
|
|
|
|
:pagination="usersPagination"
|
|
|
|
style="
|
|
|
|
display: grid;
|
|
|
|
grid-template-columns: repeat(
|
|
|
|
auto-fill,
|
|
|
|
minmax(270px, 1fr)
|
|
|
|
);
|
|
|
|
grid-gap: 12px;
|
|
|
|
"
|
|
|
|
>
|
|
|
|
<MkA
|
|
|
|
v-for="user in items"
|
|
|
|
:key="user.id"
|
|
|
|
v-tooltip.mfm="
|
|
|
|
`Last posted: ${new Date(
|
2023-07-06 03:28:27 +02:00
|
|
|
user.updatedAt,
|
2023-04-08 02:01:42 +02:00
|
|
|
).toLocaleString()}`
|
|
|
|
"
|
|
|
|
class="user"
|
|
|
|
:to="`/user-info/${user.id}`"
|
|
|
|
>
|
|
|
|
<MkUserCardMini :user="user" />
|
|
|
|
</MkA>
|
|
|
|
</MkPagination>
|
|
|
|
</div>
|
|
|
|
</swiper-slide>
|
|
|
|
<swiper-slide>
|
|
|
|
<div class="_formRoot">
|
|
|
|
<MkObjectView tall :value="instance"> </MkObjectView>
|
|
|
|
</div>
|
|
|
|
</swiper-slide>
|
|
|
|
</swiper>
|
|
|
|
</MkSpacer>
|
|
|
|
</MkStickyContainer>
|
2021-04-16 10:34:06 +02:00
|
|
|
</template>
|
|
|
|
|
2022-01-18 13:30:17 +01:00
|
|
|
<script lang="ts" setup>
|
2023-09-02 01:27:33 +02:00
|
|
|
import { computed, ref, watch } from "vue";
|
2023-07-11 06:16:56 +02:00
|
|
|
import { Virtual } from "swiper/modules";
|
2023-04-08 02:01:42 +02:00
|
|
|
import { Swiper, SwiperSlide } from "swiper/vue";
|
2024-02-12 16:40:46 +01:00
|
|
|
import type { entities } from "firefish-js";
|
2023-04-08 02:01:42 +02:00
|
|
|
import MkObjectView from "@/components/MkObjectView.vue";
|
|
|
|
import FormLink from "@/components/form/link.vue";
|
|
|
|
import MkLink from "@/components/MkLink.vue";
|
|
|
|
import MkButton from "@/components/MkButton.vue";
|
|
|
|
import FormSection from "@/components/form/section.vue";
|
|
|
|
import MkKeyValue from "@/components/MkKeyValue.vue";
|
|
|
|
import FormSwitch from "@/components/form/switch.vue";
|
|
|
|
import * as os from "@/os";
|
|
|
|
import number from "@/filters/number";
|
2024-03-07 03:06:45 +01:00
|
|
|
import { isAdmin } from "@/me";
|
2023-04-08 02:01:42 +02:00
|
|
|
import { definePageMetadata } from "@/scripts/page-metadata";
|
|
|
|
import { deviceKind } from "@/scripts/device-kind";
|
|
|
|
import { defaultStore } from "@/store";
|
|
|
|
import { i18n } from "@/i18n";
|
|
|
|
import MkUserCardMini from "@/components/MkUserCardMini.vue";
|
|
|
|
import MkPagination from "@/components/MkPagination.vue";
|
2023-10-17 03:57:20 +02:00
|
|
|
import { getProxiedImageUrlNullable } from "@/scripts/media-proxy";
|
|
|
|
import icon from "@/scripts/icon";
|
2023-04-08 02:01:42 +02:00
|
|
|
import "swiper/scss";
|
|
|
|
import "swiper/scss/virtual";
|
2022-01-18 13:30:17 +01:00
|
|
|
|
2024-02-12 16:40:46 +01:00
|
|
|
type AugmentedInstanceMetadata = entities.DetailedInstanceMetadata & {
|
2023-04-04 05:46:23 +02:00
|
|
|
blockedHosts: string[];
|
2023-04-30 15:01:42 +02:00
|
|
|
silencedHosts: string[];
|
2023-04-04 05:46:23 +02:00
|
|
|
};
|
2024-02-12 16:40:46 +01:00
|
|
|
type AugmentedInstance = entities.Instance & {
|
2023-04-04 05:46:23 +02:00
|
|
|
isBlocked: boolean;
|
2023-04-30 15:01:42 +02:00
|
|
|
isSilenced: boolean;
|
2023-04-04 05:46:23 +02:00
|
|
|
};
|
|
|
|
|
2022-01-18 13:30:17 +01:00
|
|
|
const props = defineProps<{
|
|
|
|
host: string;
|
|
|
|
}>();
|
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
const tabs = ["overview"];
|
2024-02-27 18:02:59 +01:00
|
|
|
if (isAdmin) tabs.push("users", "raw");
|
2023-09-02 01:16:23 +02:00
|
|
|
const tab = ref(tabs[0]);
|
2023-08-12 02:44:46 +02:00
|
|
|
watch(tab, () => syncSlide(tabs.indexOf(tab.value)));
|
2022-10-26 04:19:42 +02:00
|
|
|
|
2023-09-02 01:16:23 +02:00
|
|
|
const meta = ref<AugmentedInstanceMetadata | null>(null);
|
|
|
|
const instance = ref<AugmentedInstance | null>(null);
|
|
|
|
const suspended = ref(false);
|
|
|
|
const isBlocked = ref(false);
|
|
|
|
const isSilenced = ref(false);
|
2024-02-27 18:02:59 +01:00
|
|
|
const faviconUrl = ref<string | null>(null);
|
2022-01-18 13:30:17 +01:00
|
|
|
|
2022-06-29 07:14:27 +02:00
|
|
|
const usersPagination = {
|
2024-01-11 02:21:21 +01:00
|
|
|
endpoint: isAdmin ? "admin/show-users" : ("users" as const),
|
2022-06-29 07:14:27 +02:00
|
|
|
limit: 10,
|
|
|
|
params: {
|
2023-04-08 02:01:42 +02:00
|
|
|
sort: "+updatedAt",
|
|
|
|
state: "all",
|
2022-06-29 07:14:27 +02:00
|
|
|
hostname: props.host,
|
|
|
|
},
|
|
|
|
offsetMode: true,
|
|
|
|
};
|
|
|
|
|
2022-01-18 13:30:17 +01:00
|
|
|
async function fetch() {
|
2024-01-11 02:21:21 +01:00
|
|
|
if (isAdmin)
|
2023-08-12 02:44:46 +02:00
|
|
|
meta.value = (await os.api("admin/meta")) as AugmentedInstanceMetadata;
|
|
|
|
instance.value = (await os.api("federation/show-instance", {
|
2022-06-28 15:56:18 +02:00
|
|
|
host: props.host,
|
2023-04-04 05:46:23 +02:00
|
|
|
})) as AugmentedInstance;
|
2023-08-12 02:44:46 +02:00
|
|
|
suspended.value = instance.value.isSuspended;
|
|
|
|
isBlocked.value = instance.value.isBlocked;
|
|
|
|
isSilenced.value = instance.value.isSilenced;
|
|
|
|
faviconUrl.value =
|
|
|
|
getProxiedImageUrlNullable(instance.value.faviconUrl, "preview") ??
|
|
|
|
getProxiedImageUrlNullable(instance.value.iconUrl, "preview");
|
2022-01-18 13:30:17 +01:00
|
|
|
}
|
2021-04-16 10:34:06 +02:00
|
|
|
|
2023-04-04 05:46:23 +02:00
|
|
|
async function toggleBlock() {
|
2023-08-12 02:44:46 +02:00
|
|
|
if (meta.value == null) return;
|
|
|
|
if (!instance.value) {
|
2023-04-04 05:46:23 +02:00
|
|
|
throw new Error(`Instance info not loaded`);
|
|
|
|
}
|
|
|
|
let blockedHosts: string[];
|
2023-08-12 02:44:46 +02:00
|
|
|
if (isBlocked.value) {
|
|
|
|
blockedHosts = meta.value.blockedHosts.concat([instance.value.host]);
|
2023-04-04 05:46:23 +02:00
|
|
|
} else {
|
2023-08-12 02:44:46 +02:00
|
|
|
blockedHosts = meta.value.blockedHosts.filter(
|
|
|
|
(x) => x !== instance.value!.host,
|
|
|
|
);
|
2023-04-04 05:46:23 +02:00
|
|
|
}
|
2023-04-08 02:01:42 +02:00
|
|
|
await os.api("admin/update-meta", {
|
2023-04-04 05:46:23 +02:00
|
|
|
blockedHosts,
|
2022-01-18 13:30:17 +01:00
|
|
|
});
|
|
|
|
}
|
2022-01-04 07:36:14 +01:00
|
|
|
|
2023-04-30 15:01:42 +02:00
|
|
|
async function toggleSilence() {
|
2023-08-12 02:44:46 +02:00
|
|
|
if (meta.value == null) return;
|
|
|
|
if (!instance.value) {
|
2023-04-30 15:01:42 +02:00
|
|
|
throw new Error(`Instance info not loaded`);
|
|
|
|
}
|
|
|
|
let silencedHosts: string[];
|
2023-08-12 02:44:46 +02:00
|
|
|
if (isSilenced.value) {
|
|
|
|
silencedHosts = meta.value.silencedHosts.concat([instance.value.host]);
|
2023-04-30 15:01:42 +02:00
|
|
|
} else {
|
2023-08-12 02:44:46 +02:00
|
|
|
silencedHosts = meta.value.silencedHosts.filter(
|
|
|
|
(x) => x !== instance.value!.host,
|
|
|
|
);
|
2023-04-30 15:01:42 +02:00
|
|
|
}
|
|
|
|
await os.api("admin/update-meta", {
|
|
|
|
silencedHosts,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-01-18 13:30:17 +01:00
|
|
|
async function toggleSuspend(v) {
|
2023-04-08 02:01:42 +02:00
|
|
|
await os.api("admin/federation/update-instance", {
|
2023-08-12 02:44:46 +02:00
|
|
|
host: instance.value.host,
|
|
|
|
isSuspended: suspended.value,
|
2022-01-18 13:30:17 +01:00
|
|
|
});
|
|
|
|
}
|
2022-01-04 07:36:14 +01:00
|
|
|
|
2022-01-31 12:38:26 +01:00
|
|
|
function refreshMetadata() {
|
2023-04-08 02:01:42 +02:00
|
|
|
os.api("admin/federation/refresh-remote-instance-metadata", {
|
2023-08-12 02:44:46 +02:00
|
|
|
host: instance.value.host,
|
2022-01-31 12:38:26 +01:00
|
|
|
});
|
|
|
|
os.alert({
|
2023-04-08 02:01:42 +02:00
|
|
|
text: "Refresh requested",
|
2022-01-31 12:38:26 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-01-18 13:30:17 +01:00
|
|
|
fetch();
|
|
|
|
|
2023-08-12 02:44:46 +02:00
|
|
|
const headerActions = computed(() => [
|
2023-04-08 02:01:42 +02:00
|
|
|
{
|
|
|
|
text: `https://${props.host}`,
|
2023-10-17 03:57:20 +02:00
|
|
|
icon: `${icon("ph-arrow-square-out")}`,
|
2023-04-08 02:01:42 +02:00
|
|
|
handler: () => {
|
|
|
|
window.open(`https://${props.host}`, "_blank");
|
|
|
|
},
|
2022-06-20 12:49:51 +02:00
|
|
|
},
|
2023-04-08 02:01:42 +02:00
|
|
|
]);
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
const theTabs = [
|
2023-04-08 02:01:42 +02:00
|
|
|
{
|
|
|
|
key: "overview",
|
|
|
|
title: i18n.ts.overview,
|
2023-10-17 03:57:20 +02:00
|
|
|
icon: `${icon("ph-info")}`,
|
2023-04-08 02:01:42 +02:00
|
|
|
},
|
|
|
|
];
|
2022-07-26 02:14:37 +02:00
|
|
|
|
2024-01-11 02:21:21 +01:00
|
|
|
if (isAdmin) {
|
2022-07-26 02:27:57 +02:00
|
|
|
theTabs.push(
|
2023-04-08 02:01:42 +02:00
|
|
|
{
|
|
|
|
key: "users",
|
2022-07-26 02:27:57 +02:00
|
|
|
title: i18n.ts.users,
|
2023-10-17 03:57:20 +02:00
|
|
|
icon: `${icon("ph-users")}`,
|
2022-07-26 02:27:57 +02:00
|
|
|
},
|
2023-04-08 02:01:42 +02:00
|
|
|
{
|
|
|
|
key: "raw",
|
|
|
|
title: "Raw",
|
2023-10-17 03:57:20 +02:00
|
|
|
icon: `${icon("ph-code")}`,
|
2023-07-06 03:28:27 +02:00
|
|
|
},
|
2022-07-26 02:27:57 +02:00
|
|
|
);
|
2022-07-26 02:14:37 +02:00
|
|
|
}
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2023-09-02 01:16:23 +02:00
|
|
|
const headerTabs = computed(() => theTabs);
|
2022-07-26 02:27:57 +02:00
|
|
|
|
2022-06-20 10:38:49 +02:00
|
|
|
definePageMetadata({
|
|
|
|
title: props.host,
|
2023-10-17 03:57:20 +02:00
|
|
|
icon: `${icon("ph-hard-drives")}`,
|
2021-04-16 10:34:06 +02:00
|
|
|
});
|
2022-09-10 00:42:17 +02:00
|
|
|
|
|
|
|
let swiperRef = null;
|
|
|
|
|
|
|
|
function setSwiperRef(swiper) {
|
|
|
|
swiperRef = swiper;
|
2023-08-12 02:44:46 +02:00
|
|
|
syncSlide(tabs.indexOf(tab.value));
|
2022-09-10 00:42:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function onSlideChange() {
|
2023-08-12 02:44:46 +02:00
|
|
|
tab.value = tabs[swiperRef.activeIndex];
|
2022-09-10 00:42:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function syncSlide(index) {
|
|
|
|
swiperRef.slideTo(index);
|
|
|
|
}
|
2021-04-16 10:34:06 +02:00
|
|
|
</script>
|
2021-04-16 14:47:12 +02:00
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.fnfelxur {
|
2022-06-26 09:57:12 +02:00
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
|
2021-04-25 05:31:11 +02:00
|
|
|
> .icon {
|
2021-04-16 14:47:12 +02:00
|
|
|
display: block;
|
2022-06-26 09:57:12 +02:00
|
|
|
margin: 0 16px 0 0;
|
2021-04-16 14:47:12 +02:00
|
|
|
height: 64px;
|
|
|
|
border-radius: 8px;
|
|
|
|
}
|
2022-06-26 09:57:12 +02:00
|
|
|
|
|
|
|
> .name {
|
|
|
|
word-break: break-all;
|
|
|
|
}
|
2021-04-16 14:47:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
.cmhjzshl {
|
|
|
|
> .selects {
|
|
|
|
display: flex;
|
2022-01-04 07:36:14 +01:00
|
|
|
margin: 0 0 16px 0;
|
2021-04-16 14:47:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|