mirror of
https://github.com/Litlyx/litlyx
synced 2025-12-10 15:58:38 +01:00
186 lines
7.1 KiB
Vue
186 lines
7.1 KiB
Vue
<script lang="ts" setup>
|
|
import type { TAdminProject } from '~/server/api/admin/projects';
|
|
|
|
import { PREMIUM_PLAN, getPlanFromId } from '@data/PLANS'
|
|
import { useSelectMenuStyle } from '~/composables/ui/useSelectMenuStyle';
|
|
|
|
import { sub, format, isSameDay, type Duration, startOfDay, endOfDay } from 'date-fns'
|
|
|
|
|
|
const page = ref<number>(1);
|
|
|
|
const ordersList = [
|
|
{ label: 'Older', id: '{ "created_at": 1 }' },
|
|
{ label: 'Newer', id: '{ "created_at": -1 }' },
|
|
|
|
{ label: 'Less active', id: '{ "last_log_at": 1 }' },
|
|
{ label: 'More active', id: '{ "last_log_at": -1 }' },
|
|
|
|
{ label: 'Less usage', id: '{ "limit_total": 1 }' },
|
|
{ label: 'More usage', id: '{ "limit_total": -1 }' },
|
|
|
|
{ label: 'Smaller plan', id: '{ "premium_type": 1 }' },
|
|
{ label: 'Bigger plan', id: '{ "premium_type": -1 }' },
|
|
|
|
]
|
|
|
|
const order = ref<string>('{ "created_at": -1 }');
|
|
|
|
const limitList = [
|
|
{ label: '10', id: 10 },
|
|
{ label: '20', id: 20 },
|
|
{ label: '50', id: 50 },
|
|
{ label: '100', id: 100 },
|
|
]
|
|
|
|
const limit = ref<number>(20);
|
|
|
|
const filterList = [
|
|
{ label: 'ALL', id: '{}' },
|
|
{ label: 'PREMIUM', id: '{ "premium_type": { "$gt": 0, "$lt": 1000 } }' },
|
|
{ label: 'APPSUMO', id: '{ "premium_type": { "$gt": 6000, "$lt": 7000 } }' },
|
|
{ label: 'PREMIUM+APPSUMO', id: '{ "premium_type": { "$gt": 0, "$lt": 7000 } }' },
|
|
]
|
|
|
|
|
|
function isRangeSelected(duration: Duration) {
|
|
return isSameDay(selected.value.start, sub(new Date(), duration)) && isSameDay(selected.value.end, new Date())
|
|
}
|
|
|
|
function selectRange(duration: Duration) {
|
|
selected.value = { start: sub(new Date(), duration), end: new Date() }
|
|
}
|
|
|
|
const ranges = [
|
|
{ label: 'Last 7 days', duration: { days: 7 } },
|
|
{ label: 'Last 14 days', duration: { days: 14 } },
|
|
{ label: 'Last 30 days', duration: { days: 30 } },
|
|
{ label: 'Last 3 months', duration: { months: 3 } },
|
|
{ label: 'Last 6 months', duration: { months: 6 } },
|
|
{ label: 'Last year', duration: { years: 1 } }
|
|
]
|
|
const selected = ref<{ start: Date, end: Date }>({ start: sub(new Date(), { days: 14 }), end: new Date() })
|
|
|
|
|
|
onMounted(() => {
|
|
for (const key in PREMIUM_PLAN) {
|
|
filterList.push({ label: key, id: `{"premium_type": ${(PREMIUM_PLAN as any)[key].ID}}` });
|
|
}
|
|
})
|
|
|
|
const filter = ref<string>('{}');
|
|
|
|
const { data: projectsInfo, pending: pendingProjects } = useFetch<{ count: number, projects: TAdminProject[] }>(
|
|
() => `/api/admin/projects?page=${page.value - 1}&limit=${limit.value}&sortQuery=${order.value}&filterQuery=${filter.value}&filterFrom=${selected.value.start.toISOString()}&filterTo=${selected.value.end.toISOString()}`,
|
|
signHeaders()
|
|
);
|
|
|
|
const { data: metrics, pending: pendingMetrics } = useFetch(
|
|
() => `/api/admin/metrics?filterFrom=${selected.value.start.toISOString()}&filterTo=${selected.value.end.toISOString()}`,
|
|
signHeaders()
|
|
);
|
|
|
|
const { uiMenu } = useSelectMenuStyle();
|
|
|
|
|
|
|
|
</script>
|
|
|
|
<template>
|
|
<div class="mt-6 h-full">
|
|
|
|
|
|
<div class="flex flex-col items-center gap-8">
|
|
|
|
<div class="flex items-center gap-10 px-10">
|
|
<div class="flex gap-2 items-center">
|
|
<div>Order:</div>
|
|
<USelectMenu :uiMenu="uiMenu" class="w-[12rem]" placeholder="Order" :options="ordersList"
|
|
value-attribute="id" option-attribute="label" v-model="order">
|
|
</USelectMenu>
|
|
</div>
|
|
|
|
<div class="flex gap-2 items-center">
|
|
<div>Limit:</div>
|
|
<USelectMenu :uiMenu="uiMenu" class="w-[12rem]" placeholder="Limit" :options="limitList"
|
|
value-attribute="id" option-attribute="label" v-model="limit">
|
|
</USelectMenu>
|
|
</div>
|
|
|
|
<div class="flex gap-2 items-center">
|
|
<div>Filter:</div>
|
|
<USelectMenu :uiMenu="uiMenu" class="w-[12rem]" placeholder="Filter" :options="filterList"
|
|
value-attribute="id" option-attribute="label" v-model="filter">
|
|
</USelectMenu>
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div class="flex items-center gap-10 justify-center px-10 w-full">
|
|
|
|
<div class="flex gap-2 items-center shrink-0">
|
|
<div>Page {{ page }} </div>
|
|
<div> {{ Math.min(limit, projectsInfo?.count || 0) }} of {{ projectsInfo?.count || 0
|
|
}}</div>
|
|
</div>
|
|
|
|
<div>
|
|
<UPagination v-model="page" :page-count="limit" :total="projectsInfo?.count || 0" />
|
|
</div>
|
|
|
|
<UPopover class="w-[20rem]" :popper="{ placement: 'bottom' }">
|
|
<UButton class="w-full" color="primary" variant="solid">
|
|
<div class="flex items-center justify-center w-full gap-2">
|
|
<i class="i-heroicons-calendar-days-20-solid"></i>
|
|
{{ selected.start.toLocaleDateString() }} - {{ selected.end.toLocaleDateString() }}
|
|
</div>
|
|
</UButton>
|
|
<template #panel="{ close }">
|
|
<div class="flex items-center sm:divide-x divide-gray-200 dark:divide-gray-800">
|
|
<div class="hidden sm:flex flex-col py-4">
|
|
<UButton v-for="(range, index) in ranges" :key="index" :label="range.label" color="gray"
|
|
variant="ghost" class="rounded-none px-6"
|
|
:class="[isRangeSelected(range.duration) ? 'bg-gray-100 dark:bg-gray-800' : 'hover:bg-gray-50 dark:hover:bg-gray-800/50']"
|
|
truncate @click="selectRange(range.duration)" />
|
|
</div>
|
|
|
|
<DatePicker v-model="selected" @close="close" />
|
|
</div>
|
|
</template>
|
|
</UPopover>
|
|
</div>
|
|
|
|
|
|
<div class="w-[80%]">
|
|
<div v-if="pendingMetrics"> Loading... </div>
|
|
<div class="flex gap-10 flex-wrap" v-if="!pendingMetrics && metrics">
|
|
<div> Projects: {{ metrics.totalProjects }} ({{ metrics.premiumProjects }} premium) </div>
|
|
<div>
|
|
Total visits: {{ formatNumberK(metrics.totalVisits) }}
|
|
</div>
|
|
<div>
|
|
Active: {{ metrics.totalProjects - metrics.deadProjects }} |
|
|
Dead: {{ metrics.deadProjects }}
|
|
</div>
|
|
<div>
|
|
Total events: {{ formatNumberK(metrics.totalEvents) }}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
|
|
|
|
<div
|
|
class="cursor-default flex justify-center flex-wrap gap-6 mb-[4rem] mt-4 overflow-auto h-full pt-6 pb-[20rem]">
|
|
|
|
<AdminOverviewProjectCard v-if="!pendingProjects" :key="project._id.toString()" :project="project"
|
|
class="w-[26rem]" v-for="project of projectsInfo?.projects" />
|
|
|
|
<div v-if="pendingProjects"> Loading...</div>
|
|
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped lang="scss"></style> |