Simplify manager tab hierarchy
This commit is contained in:
10
app/app.vue
10
app/app.vue
@@ -49,7 +49,7 @@ const managerPageTabs = computed(() => {
|
|||||||
{
|
{
|
||||||
key: 'balances',
|
key: 'balances',
|
||||||
label: 'Балансы',
|
label: 'Балансы',
|
||||||
active: route.query.tab !== 'withdrawals' && route.query.tab !== 'manager',
|
active: route.query.tab !== 'withdrawals' && route.query.tab !== 'products' && route.query.tab !== 'manager',
|
||||||
to: {
|
to: {
|
||||||
path: '/bonus-system',
|
path: '/bonus-system',
|
||||||
query: {
|
query: {
|
||||||
@@ -71,14 +71,14 @@ const managerPageTabs = computed(() => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
key: 'manager',
|
key: 'products',
|
||||||
label: 'Менеджеру',
|
label: 'Товары',
|
||||||
active: route.query.tab === 'manager',
|
active: route.query.tab === 'products' || route.query.tab === 'manager',
|
||||||
to: {
|
to: {
|
||||||
path: '/bonus-system',
|
path: '/bonus-system',
|
||||||
query: {
|
query: {
|
||||||
...route.query,
|
...route.query,
|
||||||
tab: 'manager',
|
tab: 'products',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -20,6 +20,15 @@ type BalanceItem = ManagerBonusBalancesQuery['managerBonusBalances'][number];
|
|||||||
type ReferralLinkItem = ManagerReferralLinksQuery['managerReferralLinks'][number];
|
type ReferralLinkItem = ManagerReferralLinksQuery['managerReferralLinks'][number];
|
||||||
type ManagerUserItem = ManagerUsersQuery['managerUsers'][number];
|
type ManagerUserItem = ManagerUsersQuery['managerUsers'][number];
|
||||||
type WithdrawalItem = ManagerWithdrawalRequestsQuery['managerWithdrawalRequests'][number];
|
type WithdrawalItem = ManagerWithdrawalRequestsQuery['managerWithdrawalRequests'][number];
|
||||||
|
type ProductCard = {
|
||||||
|
id: string;
|
||||||
|
store: string;
|
||||||
|
title: string;
|
||||||
|
amount: number;
|
||||||
|
subtitle: string;
|
||||||
|
gradient: string;
|
||||||
|
tags: string[];
|
||||||
|
};
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const search = ref('');
|
const search = ref('');
|
||||||
@@ -30,21 +39,77 @@ const withdrawalsQuery = useQuery(ManagerWithdrawalRequestsDocument, {
|
|||||||
status: 'PENDING',
|
status: 'PENDING',
|
||||||
});
|
});
|
||||||
|
|
||||||
const activeTab = computed<'balances' | 'withdrawals' | 'manager'>(() => {
|
const activeTab = computed<'balances' | 'withdrawals' | 'products'>(() => {
|
||||||
if (route.query.tab === 'withdrawals') {
|
if (route.query.tab === 'withdrawals') {
|
||||||
return 'withdrawals';
|
return 'withdrawals';
|
||||||
}
|
}
|
||||||
if (route.query.tab === 'manager') {
|
if (route.query.tab === 'products' || route.query.tab === 'manager') {
|
||||||
return 'manager';
|
return 'products';
|
||||||
}
|
}
|
||||||
return 'balances';
|
return 'balances';
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const productCards: ProductCard[] = [
|
||||||
|
{
|
||||||
|
id: 'ozon-3000',
|
||||||
|
store: 'Ozon',
|
||||||
|
title: 'Подарочная карта Ozon',
|
||||||
|
amount: 3000,
|
||||||
|
subtitle: 'Универсальная карта для маркетплейса: техника, дом и повседневные покупки.',
|
||||||
|
gradient: 'linear-gradient(135deg, #38b6ff 0%, #1369ff 55%, #0b2f72 100%)',
|
||||||
|
tags: ['Маркетплейс', 'Электронная карта', '3 000 ₽'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'ozon-5000',
|
||||||
|
store: 'Ozon',
|
||||||
|
title: 'Подарочная карта Ozon',
|
||||||
|
amount: 5000,
|
||||||
|
subtitle: 'Крупный номинал для заметных подарков и сезонных закупок.',
|
||||||
|
gradient: 'linear-gradient(135deg, #65d0ff 0%, #247bff 52%, #12315e 100%)',
|
||||||
|
tags: ['Маркетплейс', 'Топ-номинал', '5 000 ₽'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'wildberries-3000',
|
||||||
|
store: 'Wildberries',
|
||||||
|
title: 'Подарочная карта Wildberries',
|
||||||
|
amount: 3000,
|
||||||
|
subtitle: 'Подходит для одежды, дома и повседневных мелочей в одном каталоге.',
|
||||||
|
gradient: 'linear-gradient(135deg, #d84dff 0%, #8b27ff 52%, #39006a 100%)',
|
||||||
|
tags: ['Fashion', 'Маркетплейс', '3 000 ₽'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'wildberries-4000',
|
||||||
|
store: 'Wildberries',
|
||||||
|
title: 'Подарочная карта Wildberries',
|
||||||
|
amount: 4000,
|
||||||
|
subtitle: 'Средний номинал для fashion-покупок и товаров для дома.',
|
||||||
|
gradient: 'linear-gradient(135deg, #ef7cff 0%, #a12dff 50%, #4c0b7d 100%)',
|
||||||
|
tags: ['Одежда', 'Дом', '4 000 ₽'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'mvideo-4000',
|
||||||
|
store: 'М.Видео',
|
||||||
|
title: 'Подарочная карта М.Видео',
|
||||||
|
amount: 4000,
|
||||||
|
subtitle: 'Для техники, аксессуаров и бытовой электроники.',
|
||||||
|
gradient: 'linear-gradient(135deg, #ff9461 0%, #ff5630 48%, #821414 100%)',
|
||||||
|
tags: ['Техника', 'Электроника', '4 000 ₽'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'mvideo-5000',
|
||||||
|
store: 'М.Видео',
|
||||||
|
title: 'Подарочная карта М.Видео',
|
||||||
|
amount: 5000,
|
||||||
|
subtitle: 'Максимальный номинал для заметных подарков и апгрейдов рабочего места.',
|
||||||
|
gradient: 'linear-gradient(135deg, #ffb17e 0%, #ff6842 50%, #8f1818 100%)',
|
||||||
|
tags: ['Электроника', 'Подарок', '5 000 ₽'],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
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 ?? []);
|
||||||
const withdrawals = computed<WithdrawalItem[]>(() => withdrawalsQuery.result.value?.managerWithdrawalRequests ?? []);
|
const withdrawals = computed<WithdrawalItem[]>(() => withdrawalsQuery.result.value?.managerWithdrawalRequests ?? []);
|
||||||
|
|
||||||
const usersById = computed(() => new Map(users.value.map((user) => [user.id, user])));
|
const usersById = computed(() => new Map(users.value.map((user) => [user.id, user])));
|
||||||
|
|
||||||
const referralLinksByReferrer = computed(() => {
|
const referralLinksByReferrer = computed(() => {
|
||||||
@@ -120,6 +185,27 @@ const filteredWithdrawals = computed(() => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const filteredProducts = computed(() => {
|
||||||
|
const query = search.value.trim().toLowerCase();
|
||||||
|
|
||||||
|
return productCards.filter((item) => {
|
||||||
|
if (!query) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
item.store,
|
||||||
|
item.title,
|
||||||
|
item.subtitle,
|
||||||
|
String(item.amount),
|
||||||
|
...item.tags,
|
||||||
|
]
|
||||||
|
.join(' ')
|
||||||
|
.toLowerCase()
|
||||||
|
.includes(query);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
const {
|
const {
|
||||||
canLoadMore: canLoadMoreBalances,
|
canLoadMore: canLoadMoreBalances,
|
||||||
loadMore: loadMoreBalances,
|
loadMore: loadMoreBalances,
|
||||||
@@ -175,7 +261,7 @@ function formatAmount(value: number) {
|
|||||||
? 'Клиент, связанный клиент, email или процент'
|
? 'Клиент, связанный клиент, email или процент'
|
||||||
: activeTab === 'withdrawals'
|
: activeTab === 'withdrawals'
|
||||||
? 'Пользователь, сумма или статус'
|
? 'Пользователь, сумма или статус'
|
||||||
: 'Сценарии для менеджера'"
|
: 'Магазин, номинал или тип карты'"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<template v-if="activeTab === 'balances'">
|
<template v-if="activeTab === 'balances'">
|
||||||
@@ -211,51 +297,65 @@ function formatAmount(value: number) {
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template v-else-if="activeTab === 'manager'">
|
<template v-else-if="activeTab === 'products'">
|
||||||
<div class="grid gap-4 md:grid-cols-2 xl:grid-cols-4">
|
<div class="surface-card rounded-[32px] p-6">
|
||||||
<NuxtLink
|
<div class="flex flex-col gap-4 lg:flex-row lg:items-end lg:justify-between">
|
||||||
to="/bonus-system/referrals/new"
|
<div class="space-y-2">
|
||||||
class="surface-card surface-card-interactive rounded-3xl p-5"
|
<p class="text-[11px] font-semibold uppercase tracking-[0.18em] text-[#6a8a76]">Витрина магазина</p>
|
||||||
>
|
<h2 class="text-2xl font-black tracking-[-0.03em] text-[#123824]">Подарочные карты для бонусного каталога</h2>
|
||||||
<p class="text-[11px] font-semibold uppercase tracking-[0.18em] text-[#6a8a76]">Связки</p>
|
<p class="max-w-3xl text-sm leading-6 text-[#557562]">
|
||||||
<h2 class="mt-3 text-xl font-black tracking-[-0.03em] text-[#123824]">Добавить бонусную связь</h2>
|
Вынес товары в отдельную вкладку без лишнего вложения. Пока это стартовый сет из популярных магазинов с номиналами от 3 000 до 5 000 рублей.
|
||||||
<p class="mt-2 text-sm leading-6 text-[#557562]">
|
</p>
|
||||||
Связать клиентов карточным сценарием и задать процент бонусной программы.
|
</div>
|
||||||
</p>
|
|
||||||
</NuxtLink>
|
|
||||||
|
|
||||||
<NuxtLink
|
<div class="flex flex-wrap gap-2 text-sm text-[#355947]">
|
||||||
to="/bonus-system/transactions/new"
|
<span class="rounded-full bg-[#eef7f1] px-3 py-2 font-semibold">3 магазина</span>
|
||||||
class="surface-card surface-card-interactive rounded-3xl p-5"
|
<span class="rounded-full bg-[#eef7f1] px-3 py-2 font-semibold">6 карточек</span>
|
||||||
>
|
<span class="rounded-full bg-[#eef7f1] px-3 py-2 font-semibold">3 000-5 000 ₽</span>
|
||||||
<p class="text-[11px] font-semibold uppercase tracking-[0.18em] text-[#6a8a76]">Начисления</p>
|
</div>
|
||||||
<h2 class="mt-3 text-xl font-black tracking-[-0.03em] text-[#123824]">Ручное начисление</h2>
|
</div>
|
||||||
<p class="mt-2 text-sm leading-6 text-[#557562]">
|
</div>
|
||||||
Добавить бонусную транзакцию вручную, если нужно быстро выдать бонус вне автоматического сценария.
|
|
||||||
</p>
|
|
||||||
</NuxtLink>
|
|
||||||
|
|
||||||
<NuxtLink
|
<div v-if="filteredProducts.length === 0" class="manager-empty-state">
|
||||||
to="/bonus-system?tab=withdrawals"
|
По текущему запросу товары не найдены.
|
||||||
class="surface-card surface-card-interactive rounded-3xl p-5"
|
</div>
|
||||||
|
<div v-else class="grid gap-4 md:grid-cols-2 xl:grid-cols-3">
|
||||||
|
<article
|
||||||
|
v-for="product in filteredProducts"
|
||||||
|
:key="product.id"
|
||||||
|
class="surface-card overflow-hidden rounded-[32px]"
|
||||||
>
|
>
|
||||||
<p class="text-[11px] font-semibold uppercase tracking-[0.18em] text-[#6a8a76]">Выводы</p>
|
<div class="p-5 text-white" :style="{ background: product.gradient }">
|
||||||
<h2 class="mt-3 text-xl font-black tracking-[-0.03em] text-[#123824]">Проверить выплаты</h2>
|
<div class="flex items-start justify-between gap-4">
|
||||||
<p class="mt-2 text-sm leading-6 text-[#557562]">
|
<div>
|
||||||
Открыть витрину заявок на вывод и разбирать новые обращения по карточкам.
|
<p class="text-[11px] font-semibold uppercase tracking-[0.18em] text-white/72">{{ product.store }}</p>
|
||||||
</p>
|
<h3 class="mt-3 text-2xl font-black tracking-[-0.03em]">{{ product.title }}</h3>
|
||||||
</NuxtLink>
|
</div>
|
||||||
|
<span class="rounded-full bg-white/14 px-3 py-1 text-sm font-semibold backdrop-blur-sm">Витрина</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
<NuxtLink
|
<div class="mt-10">
|
||||||
to="/messages"
|
<p class="text-sm text-white/72">Номинал</p>
|
||||||
class="surface-card surface-card-interactive rounded-3xl p-5"
|
<p class="mt-2 text-4xl font-black leading-none">{{ formatAmount(product.amount) }} ₽</p>
|
||||||
>
|
</div>
|
||||||
<p class="text-[11px] font-semibold uppercase tracking-[0.18em] text-[#6a8a76]">Шаблоны</p>
|
</div>
|
||||||
<h2 class="mt-3 text-xl font-black tracking-[-0.03em] text-[#123824]">Открыть реестр уведомлений</h2>
|
|
||||||
<p class="mt-2 text-sm leading-6 text-[#557562]">
|
<div class="space-y-4 p-5">
|
||||||
Посмотреть реальные шаблоны из backend-кода и быстро понять, что именно мы отправляем клиенту.
|
<p class="text-sm leading-6 text-[#557562]">
|
||||||
</p>
|
{{ product.subtitle }}
|
||||||
</NuxtLink>
|
</p>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap gap-2">
|
||||||
|
<span
|
||||||
|
v-for="tag in product.tags"
|
||||||
|
:key="tag"
|
||||||
|
class="rounded-full bg-[#eef7f1] px-3 py-1.5 text-xs font-semibold text-[#0d854a]"
|
||||||
|
>
|
||||||
|
{{ tag }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|||||||
@@ -22,9 +22,7 @@ type RequestItem = RegistrationRequestsQuery['registrationRequests'][number];
|
|||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const entityId = computed(() => String(route.params.id || ''));
|
const entityId = computed(() => String(route.params.id || ''));
|
||||||
const isRequestMode = computed(() => route.query.tab === 'requests');
|
const isRequestMode = computed(() => route.query.tab === 'requests');
|
||||||
const backTarget = computed(() => (
|
const backTarget = computed(() => '/clients');
|
||||||
isRequestMode.value ? '/clients?tab=requests' : '/clients'
|
|
||||||
));
|
|
||||||
|
|
||||||
const usersQuery = useQuery(ManagerUsersDetailDocument);
|
const usersQuery = useQuery(ManagerUsersDetailDocument);
|
||||||
const requestsQuery = useQuery(RegistrationRequestsDocument, {
|
const requestsQuery = useQuery(RegistrationRequestsDocument, {
|
||||||
|
|||||||
@@ -1,77 +1,28 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { useQuery } from '@vue/apollo-composable';
|
import { useQuery } from '@vue/apollo-composable';
|
||||||
import {
|
import { ManagerUsersDocument } from '~/composables/graphql/generated';
|
||||||
ManagerUsersDocument,
|
|
||||||
RegistrationRequestsDocument,
|
|
||||||
type RegistrationRequestsQuery,
|
|
||||||
} from '~/composables/graphql/generated';
|
|
||||||
import { messengerConnectionAvatarSrc } from '~/composables/useMessengerConnectionPresentation';
|
import { messengerConnectionAvatarSrc } from '~/composables/useMessengerConnectionPresentation';
|
||||||
|
|
||||||
definePageMeta({
|
definePageMeta({
|
||||||
middleware: ['manager-only'],
|
middleware: ['manager-only'],
|
||||||
});
|
});
|
||||||
|
|
||||||
type RequestItem = RegistrationRequestsQuery['registrationRequests'][number];
|
|
||||||
|
|
||||||
const route = useRoute();
|
|
||||||
const search = ref('');
|
const search = ref('');
|
||||||
|
|
||||||
const usersQuery = useQuery(ManagerUsersDocument);
|
const usersQuery = useQuery(ManagerUsersDocument);
|
||||||
const requestsQuery = useQuery(RegistrationRequestsDocument, {
|
|
||||||
status: null,
|
|
||||||
});
|
|
||||||
|
|
||||||
const activeTab = computed<'users' | 'requests'>(() => (
|
|
||||||
route.query.tab === 'requests' ? 'requests' : 'users'
|
|
||||||
));
|
|
||||||
|
|
||||||
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();
|
||||||
|
|
||||||
return items.filter((item) => {
|
|
||||||
if (!query) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return item.fullName.toLowerCase().includes(query);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
function requestStatusLabel(status: RequestItem['status']) {
|
|
||||||
if (status === 'APPROVED') {
|
|
||||||
return 'Одобрена';
|
|
||||||
}
|
|
||||||
if (status === 'REJECTED') {
|
|
||||||
return 'Отклонена';
|
|
||||||
}
|
|
||||||
return 'На проверке';
|
|
||||||
}
|
|
||||||
|
|
||||||
function requestStatusClass(status: RequestItem['status']) {
|
|
||||||
if (status === 'APPROVED') {
|
|
||||||
return 'badge badge-success border-0';
|
|
||||||
}
|
|
||||||
if (status === 'REJECTED') {
|
|
||||||
return 'badge badge-error border-0';
|
|
||||||
}
|
|
||||||
return 'badge badge-warning border-0';
|
|
||||||
}
|
|
||||||
|
|
||||||
const filteredRequests = computed(() => {
|
|
||||||
const items = requestsQuery.result.value?.registrationRequests ?? [];
|
|
||||||
const query = search.value.trim().toLowerCase();
|
|
||||||
|
|
||||||
return items.filter((item) => {
|
return items.filter((item) => {
|
||||||
if (!query) {
|
if (!query) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
item.companyName,
|
item.fullName,
|
||||||
item.contactName,
|
|
||||||
item.email,
|
item.email,
|
||||||
item.inn || '',
|
item.companyName || '',
|
||||||
]
|
]
|
||||||
.join(' ')
|
.join(' ')
|
||||||
.toLowerCase()
|
.toLowerCase()
|
||||||
@@ -87,20 +38,7 @@ const {
|
|||||||
visibleItems: visibleUsers,
|
visibleItems: visibleUsers,
|
||||||
} = useIncrementalList(filteredUsers, {
|
} = useIncrementalList(filteredUsers, {
|
||||||
pageSize: 24,
|
pageSize: 24,
|
||||||
enabled: computed(() => activeTab.value === 'users'),
|
resetKeys: [search],
|
||||||
resetKeys: [search, activeTab],
|
|
||||||
});
|
|
||||||
|
|
||||||
const {
|
|
||||||
canLoadMore: canLoadMoreRequests,
|
|
||||||
loadMore: loadMoreRequests,
|
|
||||||
loadMoreSentinel: loadMoreRequestsSentinel,
|
|
||||||
remainingCount: remainingRequestsCount,
|
|
||||||
visibleItems: visibleRequests,
|
|
||||||
} = useIncrementalList(filteredRequests, {
|
|
||||||
pageSize: 24,
|
|
||||||
enabled: computed(() => activeTab.value === 'requests'),
|
|
||||||
resetKeys: [search, activeTab],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
function userInitials(fullName: string) {
|
function userInitials(fullName: string) {
|
||||||
@@ -123,7 +61,7 @@ function userInitials(fullName: string) {
|
|||||||
<UiSectionSearchHero
|
<UiSectionSearchHero
|
||||||
v-model="search"
|
v-model="search"
|
||||||
title="Клиенты"
|
title="Клиенты"
|
||||||
:search-placeholder="activeTab === 'users' ? 'Имя пользователя' : 'Компания, контакт, email или ИНН'"
|
search-placeholder="Имя, компания или email"
|
||||||
>
|
>
|
||||||
<template #controls>
|
<template #controls>
|
||||||
<NuxtLink to="/clients/invite" class="btn btn-primary border-0">
|
<NuxtLink to="/clients/invite" class="btn btn-primary border-0">
|
||||||
@@ -132,95 +70,33 @@ function userInitials(fullName: string) {
|
|||||||
</template>
|
</template>
|
||||||
</UiSectionSearchHero>
|
</UiSectionSearchHero>
|
||||||
|
|
||||||
<div class="flex flex-wrap gap-2">
|
<div v-if="usersQuery.loading.value" class="manager-empty-state">
|
||||||
<NuxtLink
|
Загружаем пользователей...
|
||||||
to="/clients?tab=users"
|
|
||||||
class="inline-flex items-center rounded-full px-4 py-2 text-sm font-semibold transition"
|
|
||||||
:class="activeTab === 'users' ? 'bg-[#123824] text-white' : 'bg-white text-[#355947] hover:bg-[#f4faf6]'"
|
|
||||||
>
|
|
||||||
Клиенты
|
|
||||||
</NuxtLink>
|
|
||||||
<NuxtLink
|
|
||||||
to="/clients?tab=requests"
|
|
||||||
class="inline-flex items-center rounded-full px-4 py-2 text-sm font-semibold transition"
|
|
||||||
:class="activeTab === 'requests' ? 'bg-[#123824] text-white' : 'bg-white text-[#355947] hover:bg-[#f4faf6]'"
|
|
||||||
>
|
|
||||||
Заявки
|
|
||||||
</NuxtLink>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div v-else-if="filteredUsers.length === 0" class="manager-empty-state">
|
||||||
|
Пользователи по текущему запросу не найдены.
|
||||||
|
</div>
|
||||||
|
<div v-else class="space-y-4">
|
||||||
|
<div class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
|
||||||
|
<UsersGridCard
|
||||||
|
v-for="user in visibleUsers"
|
||||||
|
:key="user.id"
|
||||||
|
:to="`/clients/${user.id}`"
|
||||||
|
:full-name="user.fullName"
|
||||||
|
:avatar-src="messengerConnectionAvatarSrc(user.telegramConnection)"
|
||||||
|
:initials="userInitials(user.fullName)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<template v-if="activeTab === 'users'">
|
<div
|
||||||
<div v-if="usersQuery.loading.value" class="manager-empty-state">
|
v-if="canLoadMoreUsers"
|
||||||
Загружаем пользователей...
|
ref="loadMoreUsersSentinel"
|
||||||
|
class="flex justify-center"
|
||||||
|
>
|
||||||
|
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMoreUsers">
|
||||||
|
Показать ещё {{ Math.min(remainingUsersCount, 24) }}
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<div v-else-if="filteredUsers.length === 0" class="manager-empty-state">
|
</div>
|
||||||
Пользователи по текущему запросу не найдены.
|
|
||||||
</div>
|
|
||||||
<div v-else class="space-y-4">
|
|
||||||
<div class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
|
|
||||||
<UsersGridCard
|
|
||||||
v-for="user in visibleUsers"
|
|
||||||
:key="user.id"
|
|
||||||
:to="`/clients/${user.id}`"
|
|
||||||
:full-name="user.fullName"
|
|
||||||
:avatar-src="messengerConnectionAvatarSrc(user.telegramConnection)"
|
|
||||||
:initials="userInitials(user.fullName)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
v-if="canLoadMoreUsers"
|
|
||||||
ref="loadMoreUsersSentinel"
|
|
||||||
class="flex justify-center"
|
|
||||||
>
|
|
||||||
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMoreUsers">
|
|
||||||
Показать ещё {{ Math.min(remainingUsersCount, 24) }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<template v-else>
|
|
||||||
<div v-if="requestsQuery.loading.value" class="manager-empty-state">
|
|
||||||
Загружаем заявки...
|
|
||||||
</div>
|
|
||||||
<div v-else-if="filteredRequests.length === 0" class="manager-empty-state">
|
|
||||||
Заявки по текущему запросу не найдены.
|
|
||||||
</div>
|
|
||||||
<div v-else class="space-y-4">
|
|
||||||
<div class="grid gap-4 lg:grid-cols-2 xl:grid-cols-3">
|
|
||||||
<NuxtLink
|
|
||||||
v-for="request in visibleRequests"
|
|
||||||
:key="request.id"
|
|
||||||
:to="`/clients/${request.id}?tab=requests`"
|
|
||||||
class="surface-card surface-card-interactive rounded-3xl p-5"
|
|
||||||
>
|
|
||||||
<div class="flex items-start justify-between gap-3">
|
|
||||||
<div class="space-y-1">
|
|
||||||
<h2 class="text-lg font-bold text-[#123824]">{{ request.companyName }}</h2>
|
|
||||||
<p class="text-sm text-[#466653]">{{ request.contactName }}</p>
|
|
||||||
</div>
|
|
||||||
<span :class="requestStatusClass(request.status)">{{ requestStatusLabel(request.status) }}</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-4 space-y-2 text-sm text-[#355947]">
|
|
||||||
<p>{{ request.email }}</p>
|
|
||||||
<p v-if="request.inn">ИНН: {{ request.inn }}</p>
|
|
||||||
<p>{{ new Date(request.createdAt).toLocaleDateString() }}</p>
|
|
||||||
</div>
|
|
||||||
</NuxtLink>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
v-if="canLoadMoreRequests"
|
|
||||||
ref="loadMoreRequestsSentinel"
|
|
||||||
class="flex justify-center"
|
|
||||||
>
|
|
||||||
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMoreRequests">
|
|
||||||
Показать ещё {{ Math.min(remainingRequestsCount, 24) }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
Reference in New Issue
Block a user