Move manager tabs above content canvas
This commit is contained in:
82
app/app.vue
82
app/app.vue
@@ -9,6 +9,72 @@ const meQuery = useQuery(MeDocument);
|
|||||||
const hasManagerDock = computed(() => (
|
const hasManagerDock = computed(() => (
|
||||||
!isLoginPage.value && hasManagerAccess(meQuery.result.value?.me?.role)
|
!isLoginPage.value && hasManagerAccess(meQuery.result.value?.me?.role)
|
||||||
));
|
));
|
||||||
|
|
||||||
|
const managerPageTabs = computed(() => {
|
||||||
|
if (!hasManagerDock.value) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (route.path === '/clients') {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
key: 'users',
|
||||||
|
label: 'Пользователи',
|
||||||
|
active: route.query.tab !== 'requests',
|
||||||
|
to: {
|
||||||
|
path: '/clients',
|
||||||
|
query: {
|
||||||
|
...route.query,
|
||||||
|
tab: 'users',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'requests',
|
||||||
|
label: 'Заявки',
|
||||||
|
active: route.query.tab === 'requests',
|
||||||
|
to: {
|
||||||
|
path: '/clients',
|
||||||
|
query: {
|
||||||
|
...route.query,
|
||||||
|
tab: 'requests',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (route.path === '/bonus-system') {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
key: 'balances',
|
||||||
|
label: 'Балансы',
|
||||||
|
active: route.query.tab !== 'withdrawals',
|
||||||
|
to: {
|
||||||
|
path: '/bonus-system',
|
||||||
|
query: {
|
||||||
|
...route.query,
|
||||||
|
tab: 'balances',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'withdrawals',
|
||||||
|
label: 'Заявки на выплату',
|
||||||
|
active: route.query.tab === 'withdrawals',
|
||||||
|
to: {
|
||||||
|
path: '/bonus-system',
|
||||||
|
query: {
|
||||||
|
...route.query,
|
||||||
|
tab: 'withdrawals',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [];
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -18,7 +84,21 @@ const hasManagerDock = computed(() => (
|
|||||||
class="mx-auto w-full max-w-[1440px] p-4 pt-[104px] md:p-6 md:pt-[112px] lg:p-8 lg:pt-[118px]"
|
class="mx-auto w-full max-w-[1440px] p-4 pt-[104px] md:p-6 md:pt-[112px] lg:p-8 lg:pt-[118px]"
|
||||||
:class="hasManagerDock ? 'pb-[116px] md:pb-[128px]' : ''"
|
:class="hasManagerDock ? 'pb-[116px] md:pb-[128px]' : ''"
|
||||||
>
|
>
|
||||||
<div class="lk-content-canvas">
|
<div v-if="managerPageTabs.length" class="lk-page-tabs-shell">
|
||||||
|
<nav class="manager-page-tabs" aria-label="Разделы страницы">
|
||||||
|
<NuxtLink
|
||||||
|
v-for="tab in managerPageTabs"
|
||||||
|
:key="tab.key"
|
||||||
|
:to="tab.to"
|
||||||
|
class="manager-page-tab"
|
||||||
|
:class="{ 'manager-page-tab--active': tab.active }"
|
||||||
|
>
|
||||||
|
{{ tab.label }}
|
||||||
|
</NuxtLink>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="lk-content-canvas" :class="{ 'lk-content-canvas--with-tabs': managerPageTabs.length }">
|
||||||
<NuxtPage />
|
<NuxtPage />
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
--brand-surface: #f4fbf7;
|
--brand-surface: #f4fbf7;
|
||||||
--brand-muted: #d8eee1;
|
--brand-muted: #d8eee1;
|
||||||
--brand-ink: #0f2f20;
|
--brand-ink: #0f2f20;
|
||||||
|
--lk-canvas-bg: color-mix(in oklab, #edf3ef 82%, white);
|
||||||
}
|
}
|
||||||
|
|
||||||
[data-theme='aqua'] {
|
[data-theme='aqua'] {
|
||||||
@@ -96,13 +97,18 @@ body {
|
|||||||
.lk-content-canvas {
|
.lk-content-canvas {
|
||||||
border: 0;
|
border: 0;
|
||||||
border-radius: 2rem;
|
border-radius: 2rem;
|
||||||
background: color-mix(in oklab, #edf3ef 82%, white);
|
background: var(--lk-canvas-bg);
|
||||||
box-shadow:
|
box-shadow:
|
||||||
0 18px 46px rgba(16, 73, 44, 0.08),
|
0 18px 46px rgba(16, 73, 44, 0.08),
|
||||||
inset 0 1px 0 rgba(255, 255, 255, 0.32);
|
inset 0 1px 0 rgba(255, 255, 255, 0.32);
|
||||||
padding: clamp(1rem, 1.2vw, 1.5rem);
|
padding: clamp(1rem, 1.2vw, 1.5rem);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.lk-content-canvas--with-tabs {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
.surface-card {
|
.surface-card {
|
||||||
border: 0;
|
border: 0;
|
||||||
background: #fff;
|
background: #fff;
|
||||||
@@ -238,38 +244,44 @@ body {
|
|||||||
color: #557562;
|
color: #557562;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.lk-page-tabs-shell {
|
||||||
|
position: relative;
|
||||||
|
z-index: 2;
|
||||||
|
margin-bottom: -1px;
|
||||||
|
padding-left: clamp(0.75rem, 1.3vw, 1.35rem);
|
||||||
|
}
|
||||||
|
|
||||||
.manager-page-tabs {
|
.manager-page-tabs {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: flex-end;
|
||||||
gap: 0.35rem;
|
gap: 0.45rem;
|
||||||
border: 1px solid #d7e9de;
|
|
||||||
border-radius: 999px;
|
|
||||||
background: rgba(255, 255, 255, 0.72);
|
|
||||||
padding: 0.3rem;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.manager-page-tab {
|
.manager-page-tab {
|
||||||
border: 0;
|
border: 1px solid #d7e9de;
|
||||||
border-radius: 999px;
|
border-bottom: 0;
|
||||||
|
border-radius: 1.15rem 1.15rem 0 0;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
padding: 0.75rem 1.15rem;
|
margin-bottom: -1px;
|
||||||
|
padding: 0.8rem 1.2rem 0.95rem;
|
||||||
font-size: 0.95rem;
|
font-size: 0.95rem;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
color: #5c7b69;
|
color: #5c7b69;
|
||||||
transition:
|
transition:
|
||||||
background-color 0.18s ease,
|
background-color 0.18s ease,
|
||||||
box-shadow 0.18s ease,
|
border-color 0.18s ease,
|
||||||
color 0.18s ease;
|
color 0.18s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.manager-page-tab:hover {
|
.manager-page-tab:hover {
|
||||||
color: #123824;
|
color: #123824;
|
||||||
|
border-color: #bfd8ca;
|
||||||
}
|
}
|
||||||
|
|
||||||
.manager-page-tab--active {
|
.manager-page-tab--active {
|
||||||
background: var(--brand-primary);
|
background: var(--lk-canvas-bg);
|
||||||
color: #fff;
|
color: #123824;
|
||||||
box-shadow: 0 12px 28px rgba(19, 153, 87, 0.22);
|
border-color: #d2e3da;
|
||||||
}
|
}
|
||||||
|
|
||||||
.manager-dock-shell {
|
.manager-dock-shell {
|
||||||
|
|||||||
45
app/components/users/GridCard.vue
Normal file
45
app/components/users/GridCard.vue
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
withDefaults(defineProps<{
|
||||||
|
to: string;
|
||||||
|
fullName: string;
|
||||||
|
avatarSrc?: string;
|
||||||
|
initials?: string;
|
||||||
|
metaLabel?: string;
|
||||||
|
metaValue?: string;
|
||||||
|
}>(), {
|
||||||
|
avatarSrc: '',
|
||||||
|
initials: 'FR',
|
||||||
|
metaLabel: '',
|
||||||
|
metaValue: '',
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<NuxtLink
|
||||||
|
:to="to"
|
||||||
|
class="surface-card surface-card-interactive flex flex-col items-center rounded-[32px] p-6 text-center"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
v-if="avatarSrc"
|
||||||
|
:src="avatarSrc"
|
||||||
|
:alt="fullName"
|
||||||
|
class="h-24 w-24 rounded-[32px] object-cover shadow-[0_12px_30px_rgba(18,56,36,0.14)]"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-else
|
||||||
|
class="flex h-24 w-24 items-center justify-center rounded-[32px] bg-[linear-gradient(135deg,#dff7e9_0%,#c2ead3_100%)] text-3xl font-black text-[#123824] shadow-[inset_0_1px_0_rgba(255,255,255,0.65)]"
|
||||||
|
>
|
||||||
|
{{ initials }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h2 class="mt-8 text-lg font-bold leading-tight text-[#123824]">{{ fullName }}</h2>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="metaLabel && metaValue"
|
||||||
|
class="mt-4 inline-flex flex-wrap items-center justify-center gap-2 rounded-full border border-[#e1c15a] bg-[#fff8dc] px-4 py-2 text-sm text-[#7a5b00]"
|
||||||
|
>
|
||||||
|
<span class="font-semibold">{{ metaLabel }}</span>
|
||||||
|
<span class="font-bold text-[#123824]">{{ metaValue }}</span>
|
||||||
|
</div>
|
||||||
|
</NuxtLink>
|
||||||
|
</template>
|
||||||
@@ -22,7 +22,6 @@ type ManagerUserItem = ManagerUsersQuery['managerUsers'][number];
|
|||||||
type WithdrawalItem = ManagerWithdrawalRequestsQuery['managerWithdrawalRequests'][number];
|
type WithdrawalItem = ManagerWithdrawalRequestsQuery['managerWithdrawalRequests'][number];
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const router = useRouter();
|
|
||||||
const search = ref('');
|
const search = ref('');
|
||||||
const balancesQuery = useQuery(ManagerBonusBalancesDocument);
|
const balancesQuery = useQuery(ManagerBonusBalancesDocument);
|
||||||
const referralLinksQuery = useQuery(ManagerReferralLinksDocument);
|
const referralLinksQuery = useQuery(ManagerReferralLinksDocument);
|
||||||
@@ -35,15 +34,6 @@ const activeTab = computed<'balances' | 'withdrawals'>(() => (
|
|||||||
route.query.tab === 'withdrawals' ? 'withdrawals' : 'balances'
|
route.query.tab === 'withdrawals' ? 'withdrawals' : 'balances'
|
||||||
));
|
));
|
||||||
|
|
||||||
function setTab(tab: 'balances' | 'withdrawals') {
|
|
||||||
void router.replace({
|
|
||||||
query: {
|
|
||||||
...route.query,
|
|
||||||
tab,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const balances = computed<BalanceItem[]>(() => balancesQuery.result.value?.managerBonusBalances ?? []);
|
const balances = computed<BalanceItem[]>(() => balancesQuery.result.value?.managerBonusBalances ?? []);
|
||||||
const referralLinks = computed<ReferralLinkItem[]>(() => referralLinksQuery.result.value?.managerReferralLinks ?? []);
|
const referralLinks = computed<ReferralLinkItem[]>(() => referralLinksQuery.result.value?.managerReferralLinks ?? []);
|
||||||
const users = computed<ManagerUserItem[]>(() => usersQuery.result.value?.managerUsers ?? []);
|
const users = computed<ManagerUserItem[]>(() => usersQuery.result.value?.managerUsers ?? []);
|
||||||
@@ -130,29 +120,17 @@ function userInitials(fullName: string) {
|
|||||||
|
|
||||||
return parts.map((part) => part[0]?.toUpperCase() ?? '').join('');
|
return parts.map((part) => part[0]?.toUpperCase() ?? '').join('');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function formatAmount(value: number) {
|
||||||
|
return new Intl.NumberFormat('ru-RU', {
|
||||||
|
minimumFractionDigits: 0,
|
||||||
|
maximumFractionDigits: 2,
|
||||||
|
}).format(value);
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<section class="space-y-6">
|
<section class="space-y-6">
|
||||||
<div class="manager-page-tabs">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="manager-page-tab"
|
|
||||||
:class="{ 'manager-page-tab--active': activeTab === 'balances' }"
|
|
||||||
@click="setTab('balances')"
|
|
||||||
>
|
|
||||||
Балансы
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="manager-page-tab"
|
|
||||||
:class="{ 'manager-page-tab--active': activeTab === 'withdrawals' }"
|
|
||||||
@click="setTab('withdrawals')"
|
|
||||||
>
|
|
||||||
Заявки на выплату
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<UiSectionSearchHero
|
<UiSectionSearchHero
|
||||||
v-model="search"
|
v-model="search"
|
||||||
title="Бонусы"
|
title="Бонусы"
|
||||||
@@ -173,20 +151,16 @@ function userInitials(fullName: string) {
|
|||||||
Бонусных связок пока нет.
|
Бонусных связок пока нет.
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
|
<div v-else class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
|
||||||
<NuxtLink
|
<UsersGridCard
|
||||||
v-for="item in filteredBalances"
|
v-for="item in filteredBalances"
|
||||||
:key="item.userId"
|
:key="item.userId"
|
||||||
:to="`/bonus-system/${item.userId}`"
|
:to="`/bonus-system/${item.userId}`"
|
||||||
class="block surface-card-interactive"
|
:full-name="item.fullName"
|
||||||
>
|
:avatar-src="messengerConnectionAvatarSrc(usersById.get(item.userId)?.telegramConnection)"
|
||||||
<BonusAccountCard
|
:initials="userInitials(item.fullName)"
|
||||||
:full-name="item.fullName"
|
meta-label="Доступный бонус"
|
||||||
:balance="item.balance"
|
:meta-value="formatAmount(item.balance)"
|
||||||
:avatar-src="messengerConnectionAvatarSrc(usersById.get(item.userId)?.telegramConnection)"
|
/>
|
||||||
:initials="userInitials(item.fullName)"
|
|
||||||
compact
|
|
||||||
/>
|
|
||||||
</NuxtLink>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import { useQuery } from '@vue/apollo-composable';
|
|||||||
import {
|
import {
|
||||||
ManagerUsersDocument,
|
ManagerUsersDocument,
|
||||||
RegistrationRequestsDocument,
|
RegistrationRequestsDocument,
|
||||||
type ManagerUsersQuery,
|
|
||||||
type RegistrationRequestsQuery,
|
type RegistrationRequestsQuery,
|
||||||
} from '~/composables/graphql/generated';
|
} from '~/composables/graphql/generated';
|
||||||
import { messengerConnectionAvatarSrc } from '~/composables/useMessengerConnectionPresentation';
|
import { messengerConnectionAvatarSrc } from '~/composables/useMessengerConnectionPresentation';
|
||||||
@@ -12,11 +11,9 @@ definePageMeta({
|
|||||||
middleware: ['manager-only'],
|
middleware: ['manager-only'],
|
||||||
});
|
});
|
||||||
|
|
||||||
type ManagerUserItem = ManagerUsersQuery['managerUsers'][number];
|
|
||||||
type RequestItem = RegistrationRequestsQuery['registrationRequests'][number];
|
type RequestItem = RegistrationRequestsQuery['registrationRequests'][number];
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const router = useRouter();
|
|
||||||
const search = ref('');
|
const search = ref('');
|
||||||
|
|
||||||
const usersQuery = useQuery(ManagerUsersDocument);
|
const usersQuery = useQuery(ManagerUsersDocument);
|
||||||
@@ -28,15 +25,6 @@ const activeTab = computed<'users' | 'requests'>(() => (
|
|||||||
route.query.tab === 'requests' ? 'requests' : 'users'
|
route.query.tab === 'requests' ? 'requests' : 'users'
|
||||||
));
|
));
|
||||||
|
|
||||||
function setTab(tab: 'users' | 'requests') {
|
|
||||||
void router.replace({
|
|
||||||
query: {
|
|
||||||
...route.query,
|
|
||||||
tab,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const filteredUsers = computed(() => {
|
const filteredUsers = computed(() => {
|
||||||
const items = usersQuery.result.value?.managerUsers ?? [];
|
const items = usersQuery.result.value?.managerUsers ?? [];
|
||||||
const query = search.value.trim().toLowerCase();
|
const query = search.value.trim().toLowerCase();
|
||||||
@@ -108,25 +96,6 @@ function userInitials(fullName: string) {
|
|||||||
|
|
||||||
<template>
|
<template>
|
||||||
<section class="space-y-6">
|
<section class="space-y-6">
|
||||||
<div class="manager-page-tabs">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="manager-page-tab"
|
|
||||||
:class="{ 'manager-page-tab--active': activeTab === 'users' }"
|
|
||||||
@click="setTab('users')"
|
|
||||||
>
|
|
||||||
Пользователи
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="manager-page-tab"
|
|
||||||
:class="{ 'manager-page-tab--active': activeTab === 'requests' }"
|
|
||||||
@click="setTab('requests')"
|
|
||||||
>
|
|
||||||
Заявки
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<UiSectionSearchHero
|
<UiSectionSearchHero
|
||||||
v-model="search"
|
v-model="search"
|
||||||
title="Пользователи"
|
title="Пользователи"
|
||||||
@@ -147,33 +116,14 @@ function userInitials(fullName: string) {
|
|||||||
Пользователи по текущему запросу не найдены.
|
Пользователи по текущему запросу не найдены.
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
|
<div v-else class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
|
||||||
<NuxtLink
|
<UsersGridCard
|
||||||
v-for="user in filteredUsers"
|
v-for="user in filteredUsers"
|
||||||
:key="user.id"
|
:key="user.id"
|
||||||
:to="`/clients/${user.id}`"
|
:to="`/clients/${user.id}`"
|
||||||
class="surface-card surface-card-interactive flex min-h-[280px] flex-col rounded-[32px] p-6"
|
:full-name="user.fullName"
|
||||||
>
|
:avatar-src="messengerConnectionAvatarSrc(user.telegramConnection)"
|
||||||
<div class="flex justify-center">
|
:initials="userInitials(user.fullName)"
|
||||||
<img
|
/>
|
||||||
v-if="messengerConnectionAvatarSrc(user.telegramConnection)"
|
|
||||||
:src="messengerConnectionAvatarSrc(user.telegramConnection)"
|
|
||||||
:alt="user.fullName"
|
|
||||||
class="h-24 w-24 rounded-[32px] object-cover shadow-[0_12px_30px_rgba(18,56,36,0.14)]"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
v-else
|
|
||||||
class="flex h-24 w-24 items-center justify-center rounded-[32px] bg-[linear-gradient(135deg,#dff7e9_0%,#c2ead3_100%)] text-3xl font-black text-[#123824] shadow-[inset_0_1px_0_rgba(255,255,255,0.65)]"
|
|
||||||
>
|
|
||||||
{{ userInitials(user.fullName) }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex-1" />
|
|
||||||
|
|
||||||
<div class="pt-8 text-center">
|
|
||||||
<h2 class="text-lg font-bold leading-tight text-[#123824]">{{ user.fullName }}</h2>
|
|
||||||
</div>
|
|
||||||
</NuxtLink>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user