Add incremental loading to manager and client lists

This commit is contained in:
Ruslan Bakiev
2026-04-06 11:49:38 +07:00
parent 7c5d0967a0
commit d119a76ae6
7 changed files with 348 additions and 71 deletions

View File

@@ -0,0 +1,93 @@
import { computed, onBeforeUnmount, onMounted, ref, unref, watch, type ComputedRef, type Ref } from 'vue';
type ReactiveValue<T> = Ref<T> | ComputedRef<T>;
type UseIncrementalListOptions = {
pageSize?: number;
enabled?: ReactiveValue<boolean>;
resetKeys?: Array<ReactiveValue<unknown> | unknown>;
};
export function useIncrementalList<T>(
items: ReactiveValue<T[]>,
options: UseIncrementalListOptions = {},
) {
const pageSize = options.pageSize ?? 24;
const enabled = computed(() => unref(options.enabled ?? true));
const visibleCount = ref(pageSize);
const loadMoreSentinel = ref<HTMLElement | null>(null);
let observer: IntersectionObserver | null = null;
const visibleItems = computed(() => (
enabled.value
? items.value.slice(0, visibleCount.value)
: items.value
));
const canLoadMore = computed(() => (
enabled.value && visibleCount.value < items.value.length
));
const remainingCount = computed(() => Math.max(items.value.length - visibleCount.value, 0));
function loadMore() {
visibleCount.value = Math.min(visibleCount.value + pageSize, items.value.length);
}
function resetVisibleCount() {
visibleCount.value = pageSize;
}
function disconnectObserver() {
observer?.disconnect();
observer = null;
}
function connectObserver() {
disconnectObserver();
if (!canLoadMore.value || !loadMoreSentinel.value) {
return;
}
observer = new IntersectionObserver((entries) => {
if (!entries.some((entry) => entry.isIntersecting)) {
return;
}
loadMore();
}, {
rootMargin: '240px 0px',
});
observer.observe(loadMoreSentinel.value);
}
const resetSignature = computed(() => JSON.stringify([
enabled.value,
items.value.length,
...((options.resetKeys ?? []).map((entry) => unref(entry as ReactiveValue<unknown>))),
]));
watch(resetSignature, resetVisibleCount, { immediate: true });
watch(canLoadMore, () => {
if (!canLoadMore.value) {
disconnectObserver();
}
});
onMounted(() => {
watch([loadMoreSentinel, canLoadMore], connectObserver, { immediate: true });
});
onBeforeUnmount(disconnectObserver);
return {
canLoadMore,
loadMore,
loadMoreSentinel,
remainingCount,
visibleCount,
visibleItems,
};
}

View File

@@ -56,33 +56,40 @@ const referralLinksByReferrer = computed(() => {
const filteredBalances = computed(() => { const filteredBalances = computed(() => {
const query = search.value.trim().toLowerCase(); const query = search.value.trim().toLowerCase();
return balances.value.filter((item) => { return balances.value
const links = referralLinksByReferrer.value.get(item.userId); .filter((item) => {
const links = referralLinksByReferrer.value.get(item.userId);
if (!links?.length) { if (!links?.length) {
return false; return false;
} }
if (!query) { if (!query) {
return true; return true;
} }
return [ return [
item.fullName, item.fullName,
item.email, item.email,
item.companyName || '', item.companyName || '',
String(item.balance), String(item.balance),
...links.flatMap((link) => [ ...links.flatMap((link) => [
link.refereeName, link.refereeName,
link.refereeEmail, link.refereeEmail,
link.refereeCompanyName || '', link.refereeCompanyName || '',
String(link.bonusPercent), String(link.bonusPercent),
]), ]),
] ]
.join(' ') .join(' ')
.toLowerCase() .toLowerCase()
.includes(query); .includes(query);
}); })
.slice()
.sort((left, right) => {
const leftLatest = referralLinksByReferrer.value.get(left.userId)?.[0]?.createdAt ?? '';
const rightLatest = referralLinksByReferrer.value.get(right.userId)?.[0]?.createdAt ?? '';
return rightLatest.localeCompare(leftLatest);
});
}); });
const filteredWithdrawals = computed(() => { const filteredWithdrawals = computed(() => {
@@ -107,6 +114,30 @@ const filteredWithdrawals = computed(() => {
}); });
}); });
const {
canLoadMore: canLoadMoreBalances,
loadMore: loadMoreBalances,
loadMoreSentinel: loadMoreBalancesSentinel,
remainingCount: remainingBalancesCount,
visibleItems: visibleBalances,
} = useIncrementalList(filteredBalances, {
pageSize: 24,
enabled: computed(() => activeTab.value === 'balances'),
resetKeys: [search, activeTab],
});
const {
canLoadMore: canLoadMoreWithdrawals,
loadMore: loadMoreWithdrawals,
loadMoreSentinel: loadMoreWithdrawalsSentinel,
remainingCount: remainingWithdrawalsCount,
visibleItems: visibleWithdrawals,
} = useIncrementalList(filteredWithdrawals, {
pageSize: 24,
enabled: computed(() => activeTab.value === 'withdrawals'),
resetKeys: [search, activeTab],
});
function userInitials(fullName: string) { function userInitials(fullName: string) {
const parts = fullName const parts = fullName
.trim() .trim()
@@ -150,17 +181,29 @@ function formatAmount(value: number) {
<div v-else-if="filteredBalances.length === 0" class="manager-empty-state"> <div v-else-if="filteredBalances.length === 0" class="manager-empty-state">
Бонусных связок пока нет. Бонусных связок пока нет.
</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="space-y-4">
<UsersGridCard <div class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
v-for="item in filteredBalances" <UsersGridCard
:key="item.userId" v-for="item in visibleBalances"
:to="`/bonus-system/${item.userId}`" :key="item.userId"
:full-name="item.fullName" :to="`/bonus-system/${item.userId}`"
:avatar-src="messengerConnectionAvatarSrc(usersById.get(item.userId)?.telegramConnection)" :full-name="item.fullName"
:initials="userInitials(item.fullName)" :avatar-src="messengerConnectionAvatarSrc(usersById.get(item.userId)?.telegramConnection)"
meta-label="Доступный бонус" :initials="userInitials(item.fullName)"
:meta-value="formatAmount(item.balance)" meta-label="Доступный бонус"
/> :meta-value="formatAmount(item.balance)"
/>
</div>
<div
v-if="canLoadMoreBalances"
ref="loadMoreBalancesSentinel"
class="flex justify-center"
>
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMoreBalances">
Показать ещё {{ Math.min(remainingBalancesCount, 24) }}
</button>
</div>
</div> </div>
</template> </template>
@@ -173,7 +216,7 @@ function formatAmount(value: number) {
</div> </div>
<div v-else class="space-y-4"> <div v-else class="space-y-4">
<article <article
v-for="withdrawal in filteredWithdrawals" v-for="withdrawal in visibleWithdrawals"
:key="withdrawal.id" :key="withdrawal.id"
class="surface-card rounded-3xl px-5 py-5" class="surface-card rounded-3xl px-5 py-5"
> >
@@ -190,6 +233,16 @@ function formatAmount(value: number) {
</NuxtLink> </NuxtLink>
</div> </div>
</article> </article>
<div
v-if="canLoadMoreWithdrawals"
ref="loadMoreWithdrawalsSentinel"
class="flex justify-center"
>
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMoreWithdrawals">
Показать ещё {{ Math.min(remainingWithdrawalsCount, 24) }}
</button>
</div>
</div> </div>
</template> </template>
</section> </section>

View File

@@ -27,14 +27,32 @@ const createReferralMutation = useMutation(CreateReferralDocument);
const clientOptions = computed<ManagerUserItem[]>(() => ( const clientOptions = computed<ManagerUserItem[]>(() => (
(usersQuery.result.value?.managerUsers ?? []) (usersQuery.result.value?.managerUsers ?? [])
.filter((user) => user.role === 'CLIENT') .filter((user) => user.role === 'CLIENT')
.slice() ));
.sort((left, right) => left.fullName.localeCompare(right.fullName, 'ru'))
const referrerOptions = computed<ManagerUserItem[]>(() => (
clientOptions.value.filter((user) => user.id !== refereeUserId.value)
));
const refereeOptions = computed<ManagerUserItem[]>(() => (
clientOptions.value.filter((user) => user.id !== referrerUserId.value)
)); ));
const referralLinks = computed<ManagerReferralLinkItem[]>(() => ( const referralLinks = computed<ManagerReferralLinkItem[]>(() => (
linksQuery.result.value?.managerReferralLinks ?? [] linksQuery.result.value?.managerReferralLinks ?? []
)); ));
watch(referrerUserId, (value) => {
if (value && value === refereeUserId.value) {
refereeUserId.value = '';
}
});
watch(refereeUserId, (value) => {
if (value && value === referrerUserId.value) {
referrerUserId.value = '';
}
});
function userOptionLabel(user: ManagerUserItem) { function userOptionLabel(user: ManagerUserItem) {
return [user.fullName, user.companyName || user.email] return [user.fullName, user.companyName || user.email]
.filter(Boolean) .filter(Boolean)
@@ -96,7 +114,7 @@ async function createReferral() {
<span class="label-text">Клиент, который получает бонус</span> <span class="label-text">Клиент, который получает бонус</span>
<select v-model="referrerUserId" class="select manager-field w-full"> <select v-model="referrerUserId" class="select manager-field w-full">
<option value="">Выберите клиента</option> <option value="">Выберите клиента</option>
<option v-for="user in clientOptions" :key="user.id" :value="user.id"> <option v-for="user in referrerOptions" :key="user.id" :value="user.id">
{{ userOptionLabel(user) }} {{ userOptionLabel(user) }}
</option> </option>
</select> </select>
@@ -106,7 +124,7 @@ async function createReferral() {
<span class="label-text">Клиент, с чьих заказов начисляется бонус</span> <span class="label-text">Клиент, с чьих заказов начисляется бонус</span>
<select v-model="refereeUserId" class="select manager-field w-full"> <select v-model="refereeUserId" class="select manager-field w-full">
<option value="">Выберите клиента</option> <option value="">Выберите клиента</option>
<option v-for="user in clientOptions" :key="user.id" :value="user.id"> <option v-for="user in refereeOptions" :key="user.id" :value="user.id">
{{ userOptionLabel(user) }} {{ userOptionLabel(user) }}
</option> </option>
</select> </select>

View File

@@ -143,6 +143,18 @@ const searchedOrders = computed<ManagerOrderItem[]>(() => {
const filteredOrders = computed<ManagerOrderItem[]>(() => searchedOrders.value.filter((order) => matchesFilter(order))); const filteredOrders = computed<ManagerOrderItem[]>(() => searchedOrders.value.filter((order) => matchesFilter(order)));
const {
canLoadMore,
loadMore,
loadMoreSentinel,
remainingCount,
visibleItems: visibleOrders,
} = useIncrementalList(filteredOrders, {
pageSize: 24,
enabled: computed(() => viewMode.value === 'list'),
resetKeys: [search, statusFilter, viewMode],
});
function escapeHtml(value: string) { function escapeHtml(value: string) {
return String(value) return String(value)
.replaceAll('&', '&amp;') .replaceAll('&', '&amp;')
@@ -315,7 +327,7 @@ const calendarOptions = computed(() => ({
<div v-else class="space-y-4"> <div v-else class="space-y-4">
<OrdersOrderSummaryCard <OrdersOrderSummaryCard
v-for="order in filteredOrders" v-for="order in visibleOrders"
:key="order.id" :key="order.id"
:to="`/client-orders/${order.id}`" :to="`/client-orders/${order.id}`"
:code="order.code" :code="order.code"
@@ -324,6 +336,16 @@ const calendarOptions = computed(() => ({
:total-price="order.totalPrice" :total-price="order.totalPrice"
:items="order.items" :items="order.items"
/> />
<div
v-if="canLoadMore"
ref="loadMoreSentinel"
class="flex justify-center pt-2"
>
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMore">
Показать ещё {{ Math.min(remainingCount, 24) }}
</button>
</div>
</div> </div>
</section> </section>
</template> </template>

View File

@@ -46,6 +46,18 @@ const currentRequest = computed(() =>
const currentUserOrders = computed<ManagerOrderItem[]>(() => userOrdersQuery.result.value?.managerOrders ?? []); const currentUserOrders = computed<ManagerOrderItem[]>(() => userOrdersQuery.result.value?.managerOrders ?? []);
const {
canLoadMore: canLoadMoreUserOrders,
loadMore: loadMoreUserOrders,
loadMoreSentinel: loadMoreUserOrdersSentinel,
remainingCount: remainingUserOrdersCount,
visibleItems: visibleUserOrders,
} = useIncrementalList(currentUserOrders, {
pageSize: 24,
enabled: computed(() => !isRequestMode.value),
resetKeys: [entityId, isRequestMode],
});
function userInitials(fullName: string) { function userInitials(fullName: string) {
const parts = fullName const parts = fullName
.trim() .trim()
@@ -213,7 +225,7 @@ async function rejectRequest() {
</div> </div>
<div v-else class="mt-4 space-y-3"> <div v-else class="mt-4 space-y-3">
<OrdersOrderSummaryCard <OrdersOrderSummaryCard
v-for="order in currentUserOrders" v-for="order in visibleUserOrders"
:key="order.id" :key="order.id"
:to="`/client-orders/${order.id}`" :to="`/client-orders/${order.id}`"
:code="order.code" :code="order.code"
@@ -222,6 +234,16 @@ async function rejectRequest() {
:total-price="order.totalPrice" :total-price="order.totalPrice"
:items="order.items" :items="order.items"
/> />
<div
v-if="canLoadMoreUserOrders"
ref="loadMoreUserOrdersSentinel"
class="flex justify-center pt-2"
>
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMoreUserOrders">
Показать ещё {{ Math.min(remainingUserOrdersCount, 24) }}
</button>
</div>
</div> </div>
</div> </div>
</template> </template>

View File

@@ -79,6 +79,30 @@ const filteredRequests = computed(() => {
}); });
}); });
const {
canLoadMore: canLoadMoreUsers,
loadMore: loadMoreUsers,
loadMoreSentinel: loadMoreUsersSentinel,
remainingCount: remainingUsersCount,
visibleItems: visibleUsers,
} = useIncrementalList(filteredUsers, {
pageSize: 24,
enabled: computed(() => activeTab.value === 'users'),
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) {
const parts = fullName const parts = fullName
.trim() .trim()
@@ -132,15 +156,27 @@ function userInitials(fullName: string) {
<div v-else-if="filteredUsers.length === 0" class="manager-empty-state"> <div v-else-if="filteredUsers.length === 0" class="manager-empty-state">
Пользователи по текущему запросу не найдены. Пользователи по текущему запросу не найдены.
</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="space-y-4">
<UsersGridCard <div class="grid gap-4 sm:grid-cols-2 xl:grid-cols-4 2xl:grid-cols-6">
v-for="user in filteredUsers" <UsersGridCard
:key="user.id" v-for="user in visibleUsers"
:to="`/clients/${user.id}`" :key="user.id"
:full-name="user.fullName" :to="`/clients/${user.id}`"
:avatar-src="messengerConnectionAvatarSrc(user.telegramConnection)" :full-name="user.fullName"
:initials="userInitials(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> </div>
</template> </template>
@@ -151,27 +187,39 @@ function userInitials(fullName: string) {
<div v-else-if="filteredRequests.length === 0" class="manager-empty-state"> <div v-else-if="filteredRequests.length === 0" class="manager-empty-state">
Заявки по текущему запросу не найдены. Заявки по текущему запросу не найдены.
</div> </div>
<div v-else class="grid gap-4 lg:grid-cols-2 xl:grid-cols-3"> <div v-else class="space-y-4">
<NuxtLink <div class="grid gap-4 lg:grid-cols-2 xl:grid-cols-3">
v-for="request in filteredRequests" <NuxtLink
:key="request.id" v-for="request in visibleRequests"
:to="`/clients/${request.id}?tab=requests`" :key="request.id"
class="surface-card surface-card-interactive rounded-3xl p-5" :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"> <div class="flex items-start justify-between gap-3">
<h2 class="text-lg font-bold text-[#123824]">{{ request.companyName }}</h2> <div class="space-y-1">
<p class="text-sm text-[#466653]">{{ request.contactName }}</p> <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>
<span :class="requestStatusClass(request.status)">{{ requestStatusLabel(request.status) }}</span>
</div>
<div class="mt-4 space-y-2 text-sm text-[#355947]"> <div class="mt-4 space-y-2 text-sm text-[#355947]">
<p>{{ request.email }}</p> <p>{{ request.email }}</p>
<p v-if="request.inn">ИНН: {{ request.inn }}</p> <p v-if="request.inn">ИНН: {{ request.inn }}</p>
<p>{{ new Date(request.createdAt).toLocaleDateString() }}</p> <p>{{ new Date(request.createdAt).toLocaleDateString() }}</p>
</div> </div>
</NuxtLink> </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> </div>
</template> </template>
</section> </section>

View File

@@ -45,6 +45,17 @@ const filteredOrders = computed(() => {
return matchSearch && matchesFilter(order); return matchSearch && matchesFilter(order);
}); });
}); });
const {
canLoadMore,
loadMore,
loadMoreSentinel,
remainingCount,
visibleItems: visibleOrders,
} = useIncrementalList(filteredOrders, {
pageSize: 24,
resetKeys: [search, statusFilter],
});
</script> </script>
<template> <template>
@@ -76,7 +87,7 @@ const filteredOrders = computed(() => {
<div v-else class="space-y-3"> <div v-else class="space-y-3">
<OrdersOrderSummaryCard <OrdersOrderSummaryCard
v-for="order in filteredOrders" v-for="order in visibleOrders"
:key="order.id" :key="order.id"
:to="`/orders/${order.id}`" :to="`/orders/${order.id}`"
:code="order.code" :code="order.code"
@@ -85,6 +96,16 @@ const filteredOrders = computed(() => {
:total-price="order.totalPrice" :total-price="order.totalPrice"
:items="order.items" :items="order.items"
/> />
<div
v-if="canLoadMore"
ref="loadMoreSentinel"
class="flex justify-center pt-2"
>
<button class="btn btn-outline border-[#d7e9de] bg-white" @click="loadMore">
Показать ещё {{ Math.min(remainingCount, 24) }}
</button>
</div>
</div> </div>
</section> </section>
</template> </template>