mirror of
https://github.com/Megghy/vtsuru.live.git
synced 2025-12-06 18:36:55 +08:00
update
This commit is contained in:
@@ -1,2 +1,251 @@
|
||||
<script setup lang="ts">
|
||||
</script>
|
||||
import { useAccount } from '@/api/account'
|
||||
import { QueryGetAPI } from '@/api/query'
|
||||
import { HISTORY_API_URL } from '@/data/constants'
|
||||
import { NCard, useMessage } from 'naive-ui'
|
||||
import { onMounted, ref } from 'vue'
|
||||
import { use } from 'echarts/core'
|
||||
import { CanvasRenderer } from 'echarts/renderers'
|
||||
import { LineChart } from 'echarts/charts'
|
||||
import { TitleComponent, TooltipComponent, LegendComponent, GridComponent, DataZoomComponent } from 'echarts/components'
|
||||
import VChart, { THEME_KEY } from 'vue-echarts'
|
||||
import { format } from 'date-fns'
|
||||
|
||||
use([CanvasRenderer, LineChart, TitleComponent, TooltipComponent, LegendComponent, GridComponent, DataZoomComponent, LineChart])
|
||||
|
||||
const accountInfo = useAccount()
|
||||
const message = useMessage()
|
||||
|
||||
const fansHistory = ref<{ time: number; count: number }[]>()
|
||||
const guardHistory = ref<{ time: number; count: number }[]>()
|
||||
const fansOption = ref()
|
||||
const guardsOption = ref()
|
||||
|
||||
async function getFansHistory() {
|
||||
await QueryGetAPI<
|
||||
{
|
||||
time: number
|
||||
count: number
|
||||
}[]
|
||||
>(HISTORY_API_URL + 'fans', {
|
||||
id: accountInfo.value?.id,
|
||||
})
|
||||
.then((data) => {
|
||||
if (data.code == 200) {
|
||||
fansHistory.value = data.data
|
||||
} else {
|
||||
message.error('加载失败: ' + data.message)
|
||||
}
|
||||
})
|
||||
.catch((err) => {
|
||||
console.error(err)
|
||||
message.error('加载失败')
|
||||
})
|
||||
}
|
||||
async function getGuardsHistory() {
|
||||
await QueryGetAPI<
|
||||
{
|
||||
time: number
|
||||
count: number
|
||||
}[]
|
||||
>(HISTORY_API_URL + 'guards', {
|
||||
id: accountInfo.value?.id,
|
||||
})
|
||||
.then((data) => {
|
||||
if (data.code == 200) {
|
||||
guardHistory.value = data.data
|
||||
} else {
|
||||
message.error('加载失败: ' + data.message)
|
||||
}
|
||||
})
|
||||
.catch((err) => {
|
||||
console.error(err)
|
||||
message.error('加载失败')
|
||||
})
|
||||
}
|
||||
function isSameDay(time1: number, time2: number) {
|
||||
const time1Date = new Date(time1)
|
||||
const time2Date = new Date(time2)
|
||||
return time1Date.getFullYear() === time2Date.getFullYear() && time1Date.getMonth() === time2Date.getMonth() && time1Date.getDate() === time2Date.getDate()
|
||||
}
|
||||
function getOptions() {
|
||||
let fansIncreacement = [] as { time: number; count: number; timeString: string }[]
|
||||
let guards = [] as { time: number; count: number; timeString: string }[]
|
||||
|
||||
let fansIncreacementLastHour = 0
|
||||
let lastHourFans = 0
|
||||
fansHistory.value?.forEach((f) => {
|
||||
if (!isSameDay(f.time, fansIncreacementLastHour)) {
|
||||
fansIncreacement.push({
|
||||
time: fansIncreacementLastHour,
|
||||
count: fansIncreacementLastHour == 0 ? 0 : f.count - lastHourFans,
|
||||
//将timeString转换为yyyy-MM-dd HH
|
||||
timeString: format(f.time, 'yyyy-MM-dd'),
|
||||
})
|
||||
fansIncreacementLastHour = f.time
|
||||
lastHourFans = f.count
|
||||
}
|
||||
})
|
||||
let lastDayGuards = 0
|
||||
let lastDay = 0
|
||||
guardHistory.value?.forEach((g) => {
|
||||
if (!isSameDay(g.time, lastDayGuards)) {
|
||||
guards.push({
|
||||
time: lastDayGuards,
|
||||
count: lastDay == 0 ? 0 : g.count - lastDayGuards,
|
||||
//将timeString转换为yyyy-MM-dd HH
|
||||
timeString: format(g.time, 'yyyy-MM-dd'),
|
||||
})
|
||||
lastDay = g.time
|
||||
lastDayGuards = g.count
|
||||
}
|
||||
})
|
||||
|
||||
fansOption.value = {
|
||||
title: {
|
||||
text: '粉丝数',
|
||||
left: 'left',
|
||||
},
|
||||
tooltip: {
|
||||
trigger: 'axis',
|
||||
},
|
||||
legend: {},
|
||||
yAxis: [
|
||||
{
|
||||
type: 'value',
|
||||
},
|
||||
{
|
||||
type: 'value',
|
||||
},
|
||||
],
|
||||
xAxis: [
|
||||
{
|
||||
type: 'category',
|
||||
axisTick: {
|
||||
alignWithLabel: true,
|
||||
},
|
||||
axisLine: {
|
||||
onZero: false,
|
||||
lineStyle: {
|
||||
color: '#EE6666',
|
||||
},
|
||||
},
|
||||
// prettier-ignore
|
||||
data: fansIncreacement.map((f) => f.timeString),
|
||||
},
|
||||
{
|
||||
type: 'category',
|
||||
axisTick: {
|
||||
alignWithLabel: true,
|
||||
},
|
||||
axisLine: {
|
||||
onZero: false,
|
||||
lineStyle: {
|
||||
color: '#5470C6',
|
||||
},
|
||||
},
|
||||
// prettier-ignore
|
||||
data: fansHistory.value?.map(f => format(f.time, 'yyyy-MM-dd HH:mm')),
|
||||
},
|
||||
],
|
||||
series: [
|
||||
{
|
||||
name: '粉丝数',
|
||||
type: 'line',
|
||||
xAxisIndex: 1,
|
||||
yAxisIndex: 1,
|
||||
emphasis: {
|
||||
focus: 'series',
|
||||
},
|
||||
data: fansHistory.value?.map((f) => f.count),
|
||||
},
|
||||
{
|
||||
name: '增量 /日',
|
||||
type: 'line',
|
||||
smooth: true,
|
||||
showSymbol: false,
|
||||
emphasis: {
|
||||
focus: 'series',
|
||||
},
|
||||
data: fansIncreacement.map((f) => f.count),
|
||||
},
|
||||
],
|
||||
dataZoom: [
|
||||
{
|
||||
show: true,
|
||||
realtime: true,
|
||||
start: 0,
|
||||
end: 100,
|
||||
xAxisIndex: [0, 1],
|
||||
},
|
||||
],
|
||||
}
|
||||
guardsOption.value = {
|
||||
title: {
|
||||
text: '舰长数',
|
||||
left: 'left',
|
||||
},
|
||||
tooltip: {
|
||||
trigger: 'axis',
|
||||
},
|
||||
legend: {},
|
||||
yAxis: [
|
||||
{
|
||||
type: 'value',
|
||||
},
|
||||
{
|
||||
type: 'value',
|
||||
},
|
||||
],
|
||||
xAxis: [
|
||||
{
|
||||
type: 'category',
|
||||
axisTick: {
|
||||
alignWithLabel: true,
|
||||
},
|
||||
axisLine: {
|
||||
onZero: false,
|
||||
lineStyle: {
|
||||
color: '#EE6666',
|
||||
},
|
||||
},
|
||||
// prettier-ignore
|
||||
data: guards.map((f) => f.timeString ),
|
||||
},
|
||||
],
|
||||
series: [
|
||||
{
|
||||
name: '舰长数',
|
||||
type: 'line',
|
||||
step: 'start',
|
||||
emphasis: {
|
||||
focus: 'series',
|
||||
},
|
||||
data: guards.map((f) => f.count),
|
||||
},
|
||||
],
|
||||
dataZoom: [
|
||||
{
|
||||
show: true,
|
||||
realtime: true,
|
||||
start: 0,
|
||||
end: 100,
|
||||
xAxisIndex: [0, 1],
|
||||
},
|
||||
],
|
||||
}
|
||||
}
|
||||
|
||||
onMounted(async () => {
|
||||
await getFansHistory()
|
||||
await getGuardsHistory()
|
||||
getOptions()
|
||||
})
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<NCard size="small">
|
||||
<VChart :option="fansOption" style="height: 200px" />
|
||||
<VChart :option="guardsOption" style="height: 200px" />
|
||||
</NCard>
|
||||
</template>
|
||||
|
||||
Reference in New Issue
Block a user