mirror of
https://github.com/certd/certd.git
synced 2026-04-16 14:00:51 +08:00
perf: 管理控制台数据统计
This commit is contained in:
@@ -105,7 +105,7 @@ const vipTypeDefine = {
|
||||
comm: {
|
||||
title: "商业版",
|
||||
type: "comm",
|
||||
privilege: ["拥有专业版所有特权", "允许商用", "修改logo、标题", "多用户无限制", "支持用户支付(敬请期待)"]
|
||||
privilege: ["拥有专业版所有特权", "允许商用,可修改logo、标题", "数据统计", "插件管理", "多用户无限制", "支持用户支付(敬请期待)"]
|
||||
}
|
||||
};
|
||||
|
||||
@@ -256,7 +256,7 @@ function openUpgrade() {
|
||||
padding: 10px;
|
||||
border: 1px solid #eee;
|
||||
border-radius: 5px;
|
||||
height: 160px;
|
||||
height: 170px;
|
||||
//background-color: rgba(250, 237, 167, 0.79);
|
||||
&.current {
|
||||
border-color: green;
|
||||
|
||||
@@ -13,6 +13,20 @@ export const sysResources = [
|
||||
permission: "sys:settings:view"
|
||||
},
|
||||
children: [
|
||||
{
|
||||
title: "控制台",
|
||||
name: "SysConsole",
|
||||
path: "/sys/console",
|
||||
component: "/sys/console/index.vue",
|
||||
meta: {
|
||||
show: () => {
|
||||
const settingStore = useSettingStore();
|
||||
return settingStore.isComm;
|
||||
},
|
||||
icon: "ion:speedometer-outline",
|
||||
permission: "sys:auth:user:view"
|
||||
}
|
||||
},
|
||||
{
|
||||
title: "用户管理",
|
||||
name: "UserManager",
|
||||
|
||||
@@ -6,6 +6,12 @@
|
||||
<div class="p-10">
|
||||
<a-descriptions title="" bordered>
|
||||
<a-descriptions-item label="用户名">{{ userInfo.username }}</a-descriptions-item>
|
||||
<a-descriptions-item label="头像">
|
||||
<a-avatar v-if="userInfo.avatar" size="large" :src="'/api/basic/file/download?&key=' + userInfo.avatar" style="background-color: #eee"> </a-avatar>
|
||||
<a-avatar v-else size="large" style="background-color: #00b4f5">
|
||||
{{ userInfo.username }}
|
||||
</a-avatar>
|
||||
</a-descriptions-item>
|
||||
<a-descriptions-item label="昵称">{{ userInfo.nickName }}</a-descriptions-item>
|
||||
<a-descriptions-item label="邮箱">{{ userInfo.email }}</a-descriptions-item>
|
||||
<a-descriptions-item label="手机号">{{ userInfo.phoneCode }}{{ userInfo.mobile }}</a-descriptions-item>
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
<template>
|
||||
<v-chart class="chart" :option="option" autoresize />
|
||||
<v-chart v-if="props.data" class="chart" :option="option" autoresize />
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
@@ -8,45 +8,48 @@ import { CanvasRenderer } from "echarts/renderers";
|
||||
import { PieChart, LineChart } from "echarts/charts";
|
||||
import { TitleComponent, TooltipComponent, LegendComponent, GridComponent } from "echarts/components";
|
||||
import VChart, { THEME_KEY } from "vue-echarts";
|
||||
import { ref, provide, defineProps } from "vue";
|
||||
import { ref, provide, defineProps, computed } from "vue";
|
||||
import { ChartItem } from "/@/views/framework/home/dashboard/charts/d";
|
||||
|
||||
use([CanvasRenderer, PieChart, LineChart, TitleComponent, TooltipComponent, GridComponent, LegendComponent]);
|
||||
|
||||
provide(THEME_KEY, "");
|
||||
|
||||
const props = defineProps<{
|
||||
data: ChartItem[];
|
||||
}>();
|
||||
|
||||
const dates = props.data.map((item) => {
|
||||
return item.name;
|
||||
});
|
||||
const counts = props.data.map((item) => {
|
||||
return item.value;
|
||||
});
|
||||
|
||||
var noDataOption = {
|
||||
// 使用系统提供的 noData 图形
|
||||
graphic: {
|
||||
type: "text",
|
||||
left: "center",
|
||||
top: "center",
|
||||
style: {
|
||||
text: "无数据",
|
||||
textAlign: "center",
|
||||
fill: "#ccc"
|
||||
const props = withDefaults(
|
||||
defineProps<{
|
||||
data: ChartItem[];
|
||||
title: string;
|
||||
}>(),
|
||||
{
|
||||
data: () => {
|
||||
return [];
|
||||
}
|
||||
}
|
||||
};
|
||||
);
|
||||
|
||||
const dates = computed(() => {
|
||||
return props.data.map((item) => {
|
||||
return item.name;
|
||||
});
|
||||
});
|
||||
const counts = computed(() => {
|
||||
return props.data.map((item) => {
|
||||
return item.value;
|
||||
});
|
||||
});
|
||||
const option = ref({
|
||||
noDataSchema: noDataOption,
|
||||
color: ["#91cc75", "#73c0de", "#ee6666", "#fac858", "#3ba272", "#fc8452", "#9a60b4", "#ea7ccc", "#5470c6"],
|
||||
// title: {
|
||||
// text: "",
|
||||
// left: "center"
|
||||
// },
|
||||
title: {
|
||||
show: props.data.length === 0, // 没数据才显示
|
||||
extStyle: {
|
||||
color: "grey",
|
||||
fontSize: 20
|
||||
},
|
||||
text: "暂无数据",
|
||||
left: "center",
|
||||
top: "center"
|
||||
},
|
||||
|
||||
tooltip: {
|
||||
trigger: "item"
|
||||
},
|
||||
@@ -83,7 +86,7 @@ const option = ref({
|
||||
],
|
||||
series: [
|
||||
{
|
||||
name: "运行次数",
|
||||
name: props.title,
|
||||
type: "line",
|
||||
stack: "Total",
|
||||
label: {
|
||||
|
||||
@@ -62,7 +62,7 @@
|
||||
</a-col>
|
||||
<a-col :span="6">
|
||||
<statistic-card title="最近运行统计" :footer="false">
|
||||
<day-count v-if="count.runningCount" :data="count.runningCount"></day-count>
|
||||
<day-count v-if="count.historyCountPerDay" :data="count.historyCountPerDay" title="运行次数"></day-count>
|
||||
</statistic-card>
|
||||
</a-col>
|
||||
<a-col :span="6">
|
||||
@@ -153,9 +153,9 @@ function transformStatusCount() {
|
||||
];
|
||||
const result = [];
|
||||
for (const item of sorted) {
|
||||
const find = data.find((v: any) => v.name === item.name);
|
||||
const find = data.find((v: any) => v.status === item.name);
|
||||
if (find) {
|
||||
result.push({ name: item.label, value: find.value });
|
||||
result.push({ name: item.label, value: find.count });
|
||||
} else {
|
||||
result.push({ name: item.label, value: 0 });
|
||||
}
|
||||
@@ -165,6 +165,12 @@ function transformStatusCount() {
|
||||
async function loadCount() {
|
||||
count.value = await GetStatisticCount();
|
||||
transformStatusCount();
|
||||
count.value.historyCountPerDay = count.value.historyCountPerDay.map((item) => {
|
||||
return {
|
||||
name: item.date,
|
||||
value: item.count
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
async function loadPluginGroups() {
|
||||
|
||||
@@ -8,7 +8,8 @@
|
||||
</div>
|
||||
<div class="content">
|
||||
<div v-if="!slots.default" class="statistic">
|
||||
<div class="value">{{ count }}</div>
|
||||
<div v-if="count !== 0" class="value">{{ count }}</div>
|
||||
<a-empty v-else></a-empty>
|
||||
</div>
|
||||
<slot></slot>
|
||||
</div>
|
||||
|
||||
8
packages/ui/certd-client/src/views/sys/console/api.ts
Normal file
8
packages/ui/certd-client/src/views/sys/console/api.ts
Normal file
@@ -0,0 +1,8 @@
|
||||
import { request } from "/@/api/service";
|
||||
|
||||
export async function GetStatisticCount() {
|
||||
return await request({
|
||||
url: "/sys/statistic/count",
|
||||
method: "POST"
|
||||
});
|
||||
}
|
||||
77
packages/ui/certd-client/src/views/sys/console/index.vue
Normal file
77
packages/ui/certd-client/src/views/sys/console/index.vue
Normal file
@@ -0,0 +1,77 @@
|
||||
<template>
|
||||
<fs-page class="page-sys-console">
|
||||
<template #header>
|
||||
<div class="title">控制台</div>
|
||||
</template>
|
||||
<div>
|
||||
<div class="statistic-data m-20">
|
||||
<a-row :gutter="20">
|
||||
<a-col :span="6">
|
||||
<statistic-card title="用户总数" :count="count.userCount">
|
||||
<template #footer>
|
||||
<router-link to="/sys/authority/user" class="flex"><fs-icon icon="ion:settings-outline" class="mr-5 fs-16" /> 管理用户</router-link>
|
||||
</template>
|
||||
</statistic-card>
|
||||
</a-col>
|
||||
<a-col :span="6">
|
||||
<statistic-card title="用户增长趋势">
|
||||
<day-count v-if="count.userRegisterCountPerDay" :data="count.userRegisterCountPerDay" title="新增用户"></day-count>
|
||||
</statistic-card>
|
||||
</a-col>
|
||||
<a-col :span="6">
|
||||
<statistic-card title="全站流水线总数" :count="count.pipelineCount">
|
||||
<template #footer>
|
||||
<router-link to="/certd/pipeline" class="flex"><fs-icon icon="ion:settings-outline" class="mr-5 fs-16" /> 管理流水线</router-link>
|
||||
</template>
|
||||
</statistic-card>
|
||||
</a-col>
|
||||
<a-col :span="6">
|
||||
<statistic-card title="流水线增长趋势">
|
||||
<day-count v-if="count.pipelineCreateCountPerDay" :data="count.pipelineCreateCountPerDay" title="新增流水线"></day-count>
|
||||
</statistic-card>
|
||||
</a-col>
|
||||
<a-col :span="6">
|
||||
<statistic-card title="最近运行统计" :footer="false">
|
||||
<day-count v-if="count.historyCountPerDay" :data="count.historyCountPerDay" title="运行次数"></day-count>
|
||||
</statistic-card>
|
||||
</a-col>
|
||||
</a-row>
|
||||
</div>
|
||||
</div>
|
||||
</fs-page>
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import { onMounted, ref } from "vue";
|
||||
import { FsIcon } from "@fast-crud/fast-crud";
|
||||
import StatisticCard from "/@/views/framework/home/dashboard/statistic-card.vue";
|
||||
import DayCount from "/@/views/framework/home/dashboard/charts/day-count.vue";
|
||||
import { GetStatisticCount } from "./api";
|
||||
|
||||
const count = ref({});
|
||||
function transformCountPerDayToChartData(key) {
|
||||
count.value[key] = count.value[key].map((item) => {
|
||||
return {
|
||||
name: item.date,
|
||||
value: item.count
|
||||
};
|
||||
});
|
||||
}
|
||||
async function loadCount() {
|
||||
count.value = await GetStatisticCount();
|
||||
transformCountPerDayToChartData("userRegisterCountPerDay");
|
||||
transformCountPerDayToChartData("pipelineCreateCountPerDay");
|
||||
transformCountPerDayToChartData("historyCountPerDay");
|
||||
}
|
||||
|
||||
onMounted(async () => {
|
||||
await loadCount();
|
||||
});
|
||||
</script>
|
||||
<style lang="less">
|
||||
.page-sys-console {
|
||||
.fs-page-content {
|
||||
background-color: #eee;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user