yumechi-no-kuni/packages/frontend/src/pages/admin/overview.federation.vue

186 lines
4.4 KiB
Vue
Raw Normal View History

2022-06-25 09:01:40 -05:00
<template>
<div>
2022-06-25 09:01:40 -05:00
<MkLoading v-if="fetching"/>
<div v-show="!fetching" :class="$style.root">
<div v-if="topSubInstancesForPie && topPubInstancesForPie" class="pies">
<div class="pie deliver _panel">
<div class="title">Sub</div>
<XPie :data="topSubInstancesForPie" class="chart"/>
<div class="subTitle">Top 10</div>
2022-06-25 09:01:40 -05:00
</div>
<div class="pie inbox _panel">
<div class="title">Pub</div>
<XPie :data="topPubInstancesForPie" class="chart"/>
<div class="subTitle">Top 10</div>
</div>
</div>
<div v-if="!fetching" class="items">
<div class="item _panel sub">
<div class="icon"><i class="ti ti-world-download"></i></div>
<div class="body">
<div class="value">
{{ number(federationSubActive) }}
<MkNumberDiff v-tooltip="i18n.ts.dayOverDayChanges" class="diff" :value="federationSubActiveDiff"></MkNumberDiff>
</div>
<div class="label">Sub</div>
</div>
</div>
<div class="item _panel pub">
<div class="icon"><i class="ti ti-world-upload"></i></div>
<div class="body">
<div class="value">
{{ number(federationPubActive) }}
<MkNumberDiff v-tooltip="i18n.ts.dayOverDayChanges" class="diff" :value="federationPubActiveDiff"></MkNumberDiff>
</div>
<div class="label">Pub</div>
</div>
</div>
</div>
</div>
2022-06-25 09:01:40 -05:00
</div>
</template>
<script lang="ts" setup>
import { onMounted, onUnmounted, ref } from 'vue';
import XPie from './overview.pie.vue';
import MkMiniChart from '@/components/MkMiniChart.vue';
2022-06-25 09:01:40 -05:00
import * as os from '@/os';
import number from '@/filters/number';
import MkNumberDiff from '@/components/MkNumberDiff.vue';
import { i18n } from '@/i18n';
import { useChartTooltip } from '@/scripts/use-chart-tooltip';
2022-06-25 09:01:40 -05:00
let topSubInstancesForPie: any = $ref(null);
let topPubInstancesForPie: any = $ref(null);
let federationPubActive = $ref<number | null>(null);
let federationPubActiveDiff = $ref<number | null>(null);
let federationSubActive = $ref<number | null>(null);
let federationSubActiveDiff = $ref<number | null>(null);
let fetching = $ref(true);
2022-06-25 09:01:40 -05:00
const { handler: externalTooltipHandler } = useChartTooltip();
onMounted(async () => {
const chart = await os.apiGet('charts/federation', { limit: 2, span: 'day' });
federationPubActive = chart.pubActive[0];
federationPubActiveDiff = chart.pubActive[0] - chart.pubActive[1];
federationSubActive = chart.subActive[0];
federationSubActiveDiff = chart.subActive[0] - chart.subActive[1];
os.apiGet('federation/stats', { limit: 10 }).then(res => {
topSubInstancesForPie = res.topSubInstances.map(x => ({
name: x.host,
color: x.themeColor,
value: x.followersCount,
onClick: () => {
os.pageWindow(`/instance-info/${x.host}`);
},
})).concat([{ name: '(other)', color: '#80808080', value: res.otherFollowersCount }]);
topPubInstancesForPie = res.topPubInstances.map(x => ({
name: x.host,
color: x.themeColor,
value: x.followingCount,
onClick: () => {
os.pageWindow(`/instance-info/${x.host}`);
},
})).concat([{ name: '(other)', color: '#80808080', value: res.otherFollowingCount }]);
2022-06-25 09:01:40 -05:00
});
fetching = false;
2022-06-25 09:01:40 -05:00
});
</script>
<style lang="scss" module>
.root {
2022-06-25 09:01:40 -05:00
&:global {
> .pies {
display: grid;
2022-12-25 00:36:03 -06:00
grid-template-columns: repeat(auto-fill, minmax(190px, 1fr));
2022-12-25 00:43:46 -06:00
grid-gap: 12px;
margin-bottom: 12px;
2022-06-25 09:01:40 -05:00
> .pie {
position: relative;
padding: 12px;
2022-06-25 09:01:40 -05:00
> .title {
position: absolute;
top: 20px;
left: 20px;
font-size: 90%;
}
2022-06-25 09:01:40 -05:00
> .chart {
max-height: 150px;
2022-06-25 09:01:40 -05:00
}
> .subTitle {
position: absolute;
bottom: 20px;
right: 20px;
font-size: 85%;
2022-06-25 09:01:40 -05:00
}
}
}
> .items {
display: grid;
2022-12-25 00:36:03 -06:00
grid-template-columns: repeat(auto-fill, minmax(190px, 1fr));
2022-12-25 00:43:46 -06:00
grid-gap: 12px;
> .item {
display: flex;
box-sizing: border-box;
padding: 12px;
2022-06-25 09:01:40 -05:00
> .icon {
display: grid;
place-items: center;
height: 100%;
aspect-ratio: 1;
margin-right: 12px;
background: var(--accentedBg);
color: var(--accent);
border-radius: 10px;
}
&.sub {
> .icon {
background: #d5ba0026;
color: #dfc300;
}
}
&.pub {
> .icon {
background: #00cf2326;
color: #00cd5b;
}
}
> .body {
2022-12-25 00:36:03 -06:00
padding: 2px 0;
> .value {
2023-01-01 18:21:44 -06:00
font-size: 1.2em;
font-weight: bold;
> .diff {
font-size: 0.65em;
font-weight: normal;
}
}
> .label {
font-size: 0.8em;
opacity: 0.5;
}
}
2022-06-25 09:01:40 -05:00
}
}
}
}
</style>