Compare commits
7 Commits
fix1707
...
1da9c6ac09
Author | SHA1 | Date | |
---|---|---|---|
1da9c6ac09 | |||
649ddbfa8a | |||
f3d21959c9 | |||
61b50d10ba | |||
ea76106caa | |||
b7edd73ce0 | |||
b6f9d017d6 |
@ -63,6 +63,7 @@ const FiltersPanelMobile: React.FC<FiltersPanelMobileProps> = ({
|
|||||||
setLocalFilterValues({});
|
setLocalFilterValues({});
|
||||||
onSearchChange('');
|
onSearchChange('');
|
||||||
// Сбрасываем фильтры в родительском компоненте
|
// Сбрасываем фильтры в родительском компоненте
|
||||||
|
// Используем пустые массивы для правильной очистки
|
||||||
Object.keys(filterValues).forEach(key => {
|
Object.keys(filterValues).forEach(key => {
|
||||||
onFilterChange?.(key, []);
|
onFilterChange?.(key, []);
|
||||||
});
|
});
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { PartsIndexCatalogsResponse, PartsIndexGroup, PartsIndexEntityInfoResponse } from '@/types/partsindex';
|
import { PartsIndexCatalogsResponse, PartsIndexGroup, PartsIndexEntityInfoResponse } from '@/types/partsindex';
|
||||||
|
|
||||||
const PARTS_INDEX_API_BASE = 'https://api.parts-index.com';
|
const PARTS_INDEX_API_BASE = process.env.PARTSAPI_URL+"/v1" || 'https://api.parts-index.com/v1';
|
||||||
const API_KEY = 'PI-E1C0ADB7-E4A8-4960-94A0-4D9C0A074DAE';
|
const API_KEY = 'PI-E1C0ADB7-E4A8-4960-94A0-4D9C0A074DAE';
|
||||||
|
|
||||||
class PartsIndexService {
|
class PartsIndexService {
|
||||||
|
@ -38,8 +38,8 @@ const mockData = Array(12).fill({
|
|||||||
brand: "Borsehung",
|
brand: "Borsehung",
|
||||||
});
|
});
|
||||||
|
|
||||||
const ITEMS_PER_PAGE = 50; // Целевое количество товаров на странице
|
const ITEMS_PER_PAGE = 50; // Уменьшено для быстрой загрузки и лучшего UX
|
||||||
const PARTSINDEX_PAGE_SIZE = 25; // Размер страницы PartsIndex API (фиксированный)
|
const PARTSINDEX_PAGE_SIZE = 25; // Синхронизировано для оптимальной скорости
|
||||||
const MAX_BRANDS_DISPLAY = 10; // Сколько брендов показывать изначально
|
const MAX_BRANDS_DISPLAY = 10; // Сколько брендов показывать изначально
|
||||||
|
|
||||||
export default function Catalog() {
|
export default function Catalog() {
|
||||||
@ -56,6 +56,36 @@ export default function Catalog() {
|
|||||||
const [showSortMobile, setShowSortMobile] = useState(false);
|
const [showSortMobile, setShowSortMobile] = useState(false);
|
||||||
const [searchQuery, setSearchQuery] = useState('');
|
const [searchQuery, setSearchQuery] = useState('');
|
||||||
const [selectedFilters, setSelectedFilters] = useState<{[key: string]: string[]}>({});
|
const [selectedFilters, setSelectedFilters] = useState<{[key: string]: string[]}>({});
|
||||||
|
|
||||||
|
// Инициализация фильтров из URL при загрузке
|
||||||
|
useEffect(() => {
|
||||||
|
if (router.isReady) {
|
||||||
|
const urlFilters: {[key: string]: string[]} = {};
|
||||||
|
const urlSearchQuery = router.query.q as string || '';
|
||||||
|
|
||||||
|
// Восстанавливаем фильтры из URL
|
||||||
|
Object.keys(router.query).forEach(key => {
|
||||||
|
if (key.startsWith('filter_')) {
|
||||||
|
const filterName = key.replace('filter_', '');
|
||||||
|
const filterValue = router.query[key];
|
||||||
|
if (typeof filterValue === 'string') {
|
||||||
|
urlFilters[filterName] = [filterValue];
|
||||||
|
} else if (Array.isArray(filterValue)) {
|
||||||
|
urlFilters[filterName] = filterValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log('🔗 Восстанавливаем фильтры из URL:', { urlFilters, urlSearchQuery });
|
||||||
|
|
||||||
|
if (Object.keys(urlFilters).length > 0) {
|
||||||
|
setSelectedFilters(urlFilters);
|
||||||
|
}
|
||||||
|
if (urlSearchQuery) {
|
||||||
|
setSearchQuery(urlSearchQuery);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [router.isReady]);
|
||||||
const [visibleArticles, setVisibleArticles] = useState<PartsAPIArticle[]>([]);
|
const [visibleArticles, setVisibleArticles] = useState<PartsAPIArticle[]>([]);
|
||||||
const [visibleEntities, setVisibleEntities] = useState<PartsIndexEntity[]>([]);
|
const [visibleEntities, setVisibleEntities] = useState<PartsIndexEntity[]>([]);
|
||||||
const [currentPage, setCurrentPage] = useState(1);
|
const [currentPage, setCurrentPage] = useState(1);
|
||||||
@ -80,6 +110,7 @@ export default function Catalog() {
|
|||||||
const [isAutoLoading, setIsAutoLoading] = useState(false); // Автоматическая подгрузка в процессе
|
const [isAutoLoading, setIsAutoLoading] = useState(false); // Автоматическая подгрузка в процессе
|
||||||
const [currentUserPage, setCurrentUserPage] = useState(1); // Текущая пользовательская страница
|
const [currentUserPage, setCurrentUserPage] = useState(1); // Текущая пользовательская страница
|
||||||
const [entitiesCache, setEntitiesCache] = useState<Map<number, PartsIndexEntity[]>>(new Map()); // Кэш страниц
|
const [entitiesCache, setEntitiesCache] = useState<Map<number, PartsIndexEntity[]>>(new Map()); // Кэш страниц
|
||||||
|
const [isFilterChanging, setIsFilterChanging] = useState(false); // Флаг изменения фильтров
|
||||||
|
|
||||||
// Карта видимости товаров по индексу
|
// Карта видимости товаров по индексу
|
||||||
const [visibilityMap, setVisibilityMap] = useState<Map<number, boolean>>(new Map());
|
const [visibilityMap, setVisibilityMap] = useState<Map<number, boolean>>(new Map());
|
||||||
@ -175,16 +206,17 @@ export default function Catalog() {
|
|||||||
// Хук для загрузки цен товаров PartsIndex
|
// Хук для загрузки цен товаров PartsIndex
|
||||||
const { getPrice, isLoadingPrice, ensurePriceLoaded } = useProductPrices();
|
const { getPrice, isLoadingPrice, ensurePriceLoaded } = useProductPrices();
|
||||||
|
|
||||||
// Загружаем цены для видимых товаров PartsIndex
|
// Загружаем цены для видимых товаров PartsIndex (для отображения конкретных цен)
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (isPartsIndexMode && visibleEntities.length > 0) {
|
if (isPartsIndexMode && visibleEntities.length > 0) {
|
||||||
|
// Загружаем цены только для видимых товаров для отображения точных цен
|
||||||
visibleEntities.forEach((entity, index) => {
|
visibleEntities.forEach((entity, index) => {
|
||||||
const productForPrice = {
|
const productForPrice = {
|
||||||
id: entity.id,
|
id: entity.id,
|
||||||
code: entity.code,
|
code: entity.code,
|
||||||
brand: entity.brand.name
|
brand: entity.brand.name
|
||||||
};
|
};
|
||||||
// Загружаем с небольшой задержкой чтобы не перегружать сервер
|
// Загружаем с небольшой задержкой
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
ensurePriceLoaded(productForPrice);
|
ensurePriceLoaded(productForPrice);
|
||||||
}, index * 50);
|
}, index * 50);
|
||||||
@ -208,9 +240,16 @@ export default function Catalog() {
|
|||||||
console.log('📊 Обновляем entitiesData:', {
|
console.log('📊 Обновляем entitiesData:', {
|
||||||
listLength: entitiesData.partsIndexCatalogEntities.list.length,
|
listLength: entitiesData.partsIndexCatalogEntities.list.length,
|
||||||
pagination: entitiesData.partsIndexCatalogEntities.pagination,
|
pagination: entitiesData.partsIndexCatalogEntities.pagination,
|
||||||
currentPage: entitiesData.partsIndexCatalogEntities.pagination?.page?.current || 1
|
currentPage: entitiesData.partsIndexCatalogEntities.pagination?.page?.current || 1,
|
||||||
|
isFilterChanging
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Если изменяются фильтры, сбрасываем флаг после получения новых данных
|
||||||
|
if (isFilterChanging) {
|
||||||
|
setIsFilterChanging(false);
|
||||||
|
console.log('🔄 Сброшен флаг isFilterChanging - получены новые отфильтрованные данные');
|
||||||
|
}
|
||||||
|
|
||||||
const newEntities = entitiesData.partsIndexCatalogEntities.list;
|
const newEntities = entitiesData.partsIndexCatalogEntities.list;
|
||||||
const pagination = entitiesData.partsIndexCatalogEntities.pagination;
|
const pagination = entitiesData.partsIndexCatalogEntities.pagination;
|
||||||
|
|
||||||
@ -228,9 +267,13 @@ export default function Catalog() {
|
|||||||
// Если это первая страница или сброс, заменяем накопленные товары
|
// Если это первая страница или сброс, заменяем накопленные товары
|
||||||
if (currentPage === 1) {
|
if (currentPage === 1) {
|
||||||
setAccumulatedEntities(newEntities);
|
setAccumulatedEntities(newEntities);
|
||||||
// Устанавливаем visibleEntities сразу, не дожидаясь проверки цен
|
// Устанавливаем visibleEntities сразу, только если не идет изменение фильтров
|
||||||
setVisibleEntities(newEntities);
|
if (!isFilterChanging) {
|
||||||
console.log('✅ Установлены visibleEntities для первой страницы:', newEntities.length);
|
setVisibleEntities(newEntities);
|
||||||
|
console.log('✅ Установлены visibleEntities для первой страницы:', newEntities.length);
|
||||||
|
} else {
|
||||||
|
console.log('🔄 Пропускаем установку visibleEntities - фильтры изменяются');
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// Добавляем к накопленным товарам
|
// Добавляем к накопленным товарам
|
||||||
setAccumulatedEntities(prev => [...prev, ...newEntities]);
|
setAccumulatedEntities(prev => [...prev, ...newEntities]);
|
||||||
@ -245,7 +288,7 @@ export default function Catalog() {
|
|||||||
|
|
||||||
console.log('✅ Пагинация обновлена:', { currentPage, hasNext, hasPrev });
|
console.log('✅ Пагинация обновлена:', { currentPage, hasNext, hasPrev });
|
||||||
}
|
}
|
||||||
}, [entitiesData]);
|
}, [entitiesData, isFilterChanging]);
|
||||||
|
|
||||||
// Преобразование выбранных фильтров в формат PartsIndex API
|
// Преобразование выбранных фильтров в формат PartsIndex API
|
||||||
const convertFiltersToPartsIndexParams = useMemo((): Record<string, any> => {
|
const convertFiltersToPartsIndexParams = useMemo((): Record<string, any> => {
|
||||||
@ -284,25 +327,18 @@ export default function Catalog() {
|
|||||||
// Восстанавливаем автоподгрузку
|
// Восстанавливаем автоподгрузку
|
||||||
console.log('🔄 Автоподгрузка активна');
|
console.log('🔄 Автоподгрузка активна');
|
||||||
|
|
||||||
// Подсчитываем текущее количество товаров с предложениями
|
// Подсчитываем текущее количество товаров (все уже отфильтрованы на сервере)
|
||||||
const currentEntitiesWithOffers = accumulatedEntities.filter(entity => {
|
const currentEntitiesCount = accumulatedEntities.length;
|
||||||
const productForPrice = { id: entity.id, code: entity.code, brand: entity.brand.name };
|
|
||||||
const priceData = getPrice(productForPrice);
|
|
||||||
const isLoadingPriceData = isLoadingPrice(productForPrice);
|
|
||||||
// Товар считается "с предложениями" если у него есть реальная цена (не null и не undefined)
|
|
||||||
return (priceData && priceData.price && priceData.price > 0) || isLoadingPriceData;
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('📊 Автоподгрузка: текущее состояние:', {
|
console.log('📊 Автоподгрузка: текущее состояние:', {
|
||||||
накопленоТоваров: accumulatedEntities.length,
|
накопленоТоваров: currentEntitiesCount,
|
||||||
сПредложениями: currentEntitiesWithOffers.length,
|
|
||||||
целевоеКоличество: ITEMS_PER_PAGE,
|
целевоеКоличество: ITEMS_PER_PAGE,
|
||||||
естьЕщеТовары: hasMoreEntities
|
естьЕщеТовары: hasMoreEntities
|
||||||
});
|
});
|
||||||
|
|
||||||
// Если у нас уже достаточно товаров с предложениями, не загружаем
|
// Если у нас уже достаточно товаров, не загружаем
|
||||||
if (currentEntitiesWithOffers.length >= ITEMS_PER_PAGE) {
|
if (currentEntitiesCount >= ITEMS_PER_PAGE) {
|
||||||
console.log('✅ Автоподгрузка: достаточно товаров с предложениями');
|
console.log('✅ Автоподгрузка: достаточно товаров');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -444,27 +480,26 @@ export default function Catalog() {
|
|||||||
}
|
}
|
||||||
}, [isPartsIndexMode, entitiesWithOffers.length, hasMoreEntities, isAutoLoading]);
|
}, [isPartsIndexMode, entitiesWithOffers.length, hasMoreEntities, isAutoLoading]);
|
||||||
|
|
||||||
// Обновляем список товаров с предложениями при изменении накопленных товаров или цен
|
// Обновляем список товаров при изменении накопленных товаров (серверная фильтрация)
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isPartsIndexMode) {
|
if (!isPartsIndexMode) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Показываем все товары, но отдельно считаем те, у которых есть цены
|
// Если фильтры изменяются, не обновляем отображение старых данных
|
||||||
|
if (isFilterChanging) {
|
||||||
|
console.log('🔄 Пропускаем обновление entitiesWithOffers - фильтры изменяются');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Все товары уже отфильтрованы на сервере - показываем все накопленные
|
||||||
const entitiesWithOffers = accumulatedEntities;
|
const entitiesWithOffers = accumulatedEntities;
|
||||||
|
|
||||||
// Подсчитываем количество товаров с реальными ценами для автоподгрузки
|
console.log('📊 Обновляем entitiesWithOffers (серверная фильтрация):', {
|
||||||
const entitiesWithRealPrices = accumulatedEntities.filter(entity => {
|
|
||||||
const productForPrice = { id: entity.id, code: entity.code, brand: entity.brand.name };
|
|
||||||
const priceData = getPrice(productForPrice);
|
|
||||||
return priceData && priceData.price && priceData.price > 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('📊 Обновляем entitiesWithOffers:', {
|
|
||||||
накопленоТоваров: accumulatedEntities.length,
|
накопленоТоваров: accumulatedEntities.length,
|
||||||
отображаемыхТоваров: entitiesWithOffers.length,
|
отображаемыхТоваров: entitiesWithOffers.length,
|
||||||
сРеальнымиЦенами: entitiesWithRealPrices.length,
|
целевоеКоличество: ITEMS_PER_PAGE,
|
||||||
целевоеКоличество: ITEMS_PER_PAGE
|
isFilterChanging
|
||||||
});
|
});
|
||||||
|
|
||||||
setEntitiesWithOffers(entitiesWithOffers);
|
setEntitiesWithOffers(entitiesWithOffers);
|
||||||
@ -484,31 +519,9 @@ export default function Catalog() {
|
|||||||
|
|
||||||
setVisibleEntities(visibleForCurrentPage);
|
setVisibleEntities(visibleForCurrentPage);
|
||||||
|
|
||||||
}, [isPartsIndexMode, accumulatedEntities, currentUserPage]);
|
}, [isPartsIndexMode, accumulatedEntities, currentUserPage, isFilterChanging]);
|
||||||
|
|
||||||
// Отдельный useEffect для обновления статистики цен (без влияния на visibleEntities)
|
|
||||||
useEffect(() => {
|
|
||||||
if (!isPartsIndexMode || accumulatedEntities.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Обновляем статистику каждые 2 секунды
|
|
||||||
const timer = setTimeout(() => {
|
|
||||||
const entitiesWithRealPrices = accumulatedEntities.filter(entity => {
|
|
||||||
const productForPrice = { id: entity.id, code: entity.code, brand: entity.brand.name };
|
|
||||||
const priceData = getPrice(productForPrice);
|
|
||||||
return priceData && priceData.price && priceData.price > 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('💰 Обновление статистики цен:', {
|
|
||||||
накопленоТоваров: accumulatedEntities.length,
|
|
||||||
сРеальнымиЦенами: entitiesWithRealPrices.length,
|
|
||||||
процентЗагрузки: Math.round((entitiesWithRealPrices.length / accumulatedEntities.length) * 100)
|
|
||||||
});
|
|
||||||
}, 2000);
|
|
||||||
|
|
||||||
return () => clearTimeout(timer);
|
|
||||||
}, [isPartsIndexMode, accumulatedEntities.length, getPrice]);
|
|
||||||
|
|
||||||
// Генерируем динамические фильтры для PartsAPI
|
// Генерируем динамические фильтры для PartsAPI
|
||||||
const generatePartsAPIFilters = useCallback((): FilterConfig[] => {
|
const generatePartsAPIFilters = useCallback((): FilterConfig[] => {
|
||||||
@ -595,27 +608,94 @@ export default function Catalog() {
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Функция для обновления URL с фильтрами
|
||||||
|
const updateUrlWithFilters = useCallback((filters: {[key: string]: string[]}, search: string) => {
|
||||||
|
const query: any = { ...router.query };
|
||||||
|
|
||||||
|
// Удаляем старые фильтры из URL
|
||||||
|
Object.keys(query).forEach(key => {
|
||||||
|
if (key.startsWith('filter_') || key === 'q') {
|
||||||
|
delete query[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Добавляем новые фильтры
|
||||||
|
Object.entries(filters).forEach(([filterName, values]) => {
|
||||||
|
if (values.length > 0) {
|
||||||
|
query[`filter_${filterName}`] = values.length === 1 ? values[0] : values;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Добавляем поисковый запрос
|
||||||
|
if (search.trim()) {
|
||||||
|
query.q = search;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Обновляем URL без перезагрузки страницы
|
||||||
|
router.push({
|
||||||
|
pathname: router.pathname,
|
||||||
|
query
|
||||||
|
}, undefined, { shallow: true });
|
||||||
|
}, [router]);
|
||||||
|
|
||||||
const handleDesktopFilterChange = (filterTitle: string, value: string | string[]) => {
|
const handleDesktopFilterChange = (filterTitle: string, value: string | string[]) => {
|
||||||
setSelectedFilters(prev => ({
|
setSelectedFilters(prev => {
|
||||||
...prev,
|
const newFilters = { ...prev };
|
||||||
[filterTitle]: Array.isArray(value) ? value : [value]
|
|
||||||
}));
|
// Если значение пустое (пустой массив или пустая строка), удаляем фильтр
|
||||||
|
if (Array.isArray(value) && value.length === 0) {
|
||||||
|
delete newFilters[filterTitle];
|
||||||
|
} else if (!value || (typeof value === 'string' && value.trim() === '')) {
|
||||||
|
delete newFilters[filterTitle];
|
||||||
|
} else {
|
||||||
|
// Иначе устанавливаем значение
|
||||||
|
newFilters[filterTitle] = Array.isArray(value) ? value : [value];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Обновляем URL
|
||||||
|
updateUrlWithFilters(newFilters, searchQuery);
|
||||||
|
|
||||||
|
return newFilters;
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleMobileFilterChange = (type: string, value: any) => {
|
const handleMobileFilterChange = (type: string, value: any) => {
|
||||||
setSelectedFilters(prev => ({
|
setSelectedFilters(prev => {
|
||||||
...prev,
|
const newFilters = { ...prev };
|
||||||
[type]: Array.isArray(value) ? value : [value]
|
|
||||||
}));
|
// Если значение пустое (пустой массив или пустая строка), удаляем фильтр
|
||||||
|
if (Array.isArray(value) && value.length === 0) {
|
||||||
|
delete newFilters[type];
|
||||||
|
} else if (!value || (typeof value === 'string' && value.trim() === '')) {
|
||||||
|
delete newFilters[type];
|
||||||
|
} else {
|
||||||
|
// Иначе устанавливаем значение
|
||||||
|
newFilters[type] = Array.isArray(value) ? value : [value];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Обновляем URL
|
||||||
|
updateUrlWithFilters(newFilters, searchQuery);
|
||||||
|
|
||||||
|
return newFilters;
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Обработчик изменения поискового запроса
|
||||||
|
const handleSearchChange = useCallback((value: string) => {
|
||||||
|
setSearchQuery(value);
|
||||||
|
updateUrlWithFilters(selectedFilters, value);
|
||||||
|
}, [selectedFilters, updateUrlWithFilters]);
|
||||||
|
|
||||||
// Функция для сброса всех фильтров
|
// Функция для сброса всех фильтров
|
||||||
const handleResetFilters = useCallback(() => {
|
const handleResetFilters = useCallback(() => {
|
||||||
setSearchQuery('');
|
setSearchQuery('');
|
||||||
setSelectedFilters({});
|
setSelectedFilters({});
|
||||||
setShowAllBrands(false);
|
setShowAllBrands(false);
|
||||||
setPartsIndexPage(1); // Сбрасываем страницу PartsIndex на первую
|
setPartsIndexPage(1); // Сбрасываем страницу PartsIndex на первую
|
||||||
}, []);
|
|
||||||
|
// Очищаем URL от фильтров
|
||||||
|
updateUrlWithFilters({}, '');
|
||||||
|
}, [updateUrlWithFilters]);
|
||||||
|
|
||||||
// Фильтрация по поиску и фильтрам для PartsAPI
|
// Фильтрация по поиску и фильтрам для PartsAPI
|
||||||
const filteredArticles = useMemo(() => {
|
const filteredArticles = useMemo(() => {
|
||||||
@ -672,6 +752,10 @@ export default function Catalog() {
|
|||||||
// Если изменился поисковый запрос или фильтры, нужно перезагрузить данные с сервера
|
// Если изменился поисковый запрос или фильтры, нужно перезагрузить данные с сервера
|
||||||
if (searchQuery.trim() || Object.keys(selectedFilters).length > 0) {
|
if (searchQuery.trim() || Object.keys(selectedFilters).length > 0) {
|
||||||
console.log('🔍 Поисковый запрос или фильтры изменились, сбрасываем пагинацию');
|
console.log('🔍 Поисковый запрос или фильтры изменились, сбрасываем пагинацию');
|
||||||
|
|
||||||
|
// Устанавливаем флаг изменения фильтров
|
||||||
|
setIsFilterChanging(true);
|
||||||
|
|
||||||
setPartsIndexPage(1);
|
setPartsIndexPage(1);
|
||||||
setCurrentUserPage(1);
|
setCurrentUserPage(1);
|
||||||
setHasMoreEntities(true);
|
setHasMoreEntities(true);
|
||||||
@ -679,10 +763,36 @@ export default function Catalog() {
|
|||||||
setEntitiesWithOffers([]);
|
setEntitiesWithOffers([]);
|
||||||
setEntitiesCache(new Map());
|
setEntitiesCache(new Map());
|
||||||
|
|
||||||
// Перезагружаем данные с новыми параметрами фильтрации
|
// Вычисляем параметры фильтрации прямо здесь, чтобы избежать зависимости от useMemo
|
||||||
const apiParams = convertFiltersToPartsIndexParams;
|
let apiParams: Record<string, any> = {};
|
||||||
|
if (paramsData?.partsIndexCatalogParams?.list && Object.keys(selectedFilters).length > 0) {
|
||||||
|
paramsData.partsIndexCatalogParams.list.forEach((param: any) => {
|
||||||
|
const selectedValues = selectedFilters[param.name];
|
||||||
|
if (selectedValues && selectedValues.length > 0) {
|
||||||
|
// Находим соответствующие значения из API данных
|
||||||
|
const matchingValues = param.values.filter((value: any) =>
|
||||||
|
selectedValues.includes(value.title || value.value)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (matchingValues.length > 0) {
|
||||||
|
// Используем ID параметра из API и значения
|
||||||
|
apiParams[param.id] = matchingValues.map((v: any) => v.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const paramsString = Object.keys(apiParams).length > 0 ? JSON.stringify(apiParams) : undefined;
|
const paramsString = Object.keys(apiParams).length > 0 ? JSON.stringify(apiParams) : undefined;
|
||||||
|
|
||||||
|
console.log('🔄 Запуск refetch с новыми фильтрами:', {
|
||||||
|
searchQuery,
|
||||||
|
selectedFilters,
|
||||||
|
apiParams,
|
||||||
|
paramsString,
|
||||||
|
catalogId,
|
||||||
|
groupId
|
||||||
|
});
|
||||||
|
|
||||||
// Также обновляем параметры фильтрации
|
// Также обновляем параметры фильтрации
|
||||||
refetchParams({
|
refetchParams({
|
||||||
catalogId: catalogId as string,
|
catalogId: catalogId as string,
|
||||||
@ -690,6 +800,10 @@ export default function Catalog() {
|
|||||||
lang: 'ru',
|
lang: 'ru',
|
||||||
q: searchQuery || undefined,
|
q: searchQuery || undefined,
|
||||||
params: paramsString
|
params: paramsString
|
||||||
|
}).then(result => {
|
||||||
|
console.log('✅ refetchParams результат:', result);
|
||||||
|
}).catch(error => {
|
||||||
|
console.error('❌ refetchParams ошибка:', error);
|
||||||
});
|
});
|
||||||
|
|
||||||
refetchEntities({
|
refetchEntities({
|
||||||
@ -700,11 +814,20 @@ export default function Catalog() {
|
|||||||
page: 1,
|
page: 1,
|
||||||
q: searchQuery || undefined,
|
q: searchQuery || undefined,
|
||||||
params: paramsString
|
params: paramsString
|
||||||
|
}).then(result => {
|
||||||
|
console.log('✅ refetchEntities результат:', result.data?.partsIndexCatalogEntities?.list?.length || 0, 'товаров');
|
||||||
|
}).catch(error => {
|
||||||
|
console.error('❌ refetchEntities ошибка:', error);
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
// Если нет активных фильтров, сбрасываем флаг
|
||||||
|
if (isFilterChanging) {
|
||||||
|
setIsFilterChanging(false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
}, [isPartsIndexMode, searchQuery, JSON.stringify(selectedFilters), refetchEntities, refetchParams, convertFiltersToPartsIndexParams]);
|
}, [isPartsIndexMode, searchQuery, JSON.stringify(selectedFilters), paramsData]);
|
||||||
|
|
||||||
// Управляем показом пустого состояния с задержкой
|
// Управляем показом пустого состояния с задержкой
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@ -724,12 +847,18 @@ export default function Catalog() {
|
|||||||
} else if (isPartsIndexMode && !entitiesLoading && !entitiesError) {
|
} else if (isPartsIndexMode && !entitiesLoading && !entitiesError) {
|
||||||
// Для PartsIndex показываем пустое состояние если нет товаров И данные уже загружены
|
// Для PartsIndex показываем пустое состояние если нет товаров И данные уже загружены
|
||||||
const hasLoadedData = accumulatedEntities.length > 0 || Boolean(entitiesData?.partsIndexCatalogEntities?.list);
|
const hasLoadedData = accumulatedEntities.length > 0 || Boolean(entitiesData?.partsIndexCatalogEntities?.list);
|
||||||
setShowEmptyState(hasLoadedData && visibleEntities.length === 0);
|
|
||||||
console.log('📊 Определяем showEmptyState для PartsIndex:', {
|
// Показываем пустое состояние если данные загружены и нет видимых товаров
|
||||||
|
// (товары уже отфильтрованы на сервере, поэтому не нужно ждать загрузки цен)
|
||||||
|
const shouldShowEmpty = hasLoadedData && visibleEntities.length === 0;
|
||||||
|
setShowEmptyState(shouldShowEmpty);
|
||||||
|
|
||||||
|
console.log('📊 Определяем showEmptyState для PartsIndex (серверная фильтрация):', {
|
||||||
hasLoadedData,
|
hasLoadedData,
|
||||||
visibleEntitiesLength: visibleEntities.length,
|
visibleEntitiesLength: visibleEntities.length,
|
||||||
accumulatedEntitiesLength: accumulatedEntities.length,
|
accumulatedEntitiesLength: accumulatedEntities.length,
|
||||||
showEmptyState: hasLoadedData && visibleEntities.length === 0
|
shouldShowEmpty,
|
||||||
|
showEmptyState: shouldShowEmpty
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
setShowEmptyState(false);
|
setShowEmptyState(false);
|
||||||
@ -884,7 +1013,7 @@ export default function Catalog() {
|
|||||||
onFilterChange={handleDesktopFilterChange}
|
onFilterChange={handleDesktopFilterChange}
|
||||||
filterValues={selectedFilters}
|
filterValues={selectedFilters}
|
||||||
searchQuery={searchQuery}
|
searchQuery={searchQuery}
|
||||||
onSearchChange={setSearchQuery}
|
onSearchChange={handleSearchChange}
|
||||||
isLoading={filtersGenerating}
|
isLoading={filtersGenerating}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@ -895,7 +1024,7 @@ export default function Catalog() {
|
|||||||
onFilterChange={handleDesktopFilterChange}
|
onFilterChange={handleDesktopFilterChange}
|
||||||
filterValues={selectedFilters}
|
filterValues={selectedFilters}
|
||||||
searchQuery={searchQuery}
|
searchQuery={searchQuery}
|
||||||
onSearchChange={setSearchQuery}
|
onSearchChange={handleSearchChange}
|
||||||
isLoading={filtersLoading}
|
isLoading={filtersLoading}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@ -906,7 +1035,7 @@ export default function Catalog() {
|
|||||||
onFilterChange={handleDesktopFilterChange}
|
onFilterChange={handleDesktopFilterChange}
|
||||||
filterValues={selectedFilters}
|
filterValues={selectedFilters}
|
||||||
searchQuery={searchQuery}
|
searchQuery={searchQuery}
|
||||||
onSearchChange={setSearchQuery}
|
onSearchChange={handleSearchChange}
|
||||||
isLoading={filtersLoading}
|
isLoading={filtersLoading}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@ -916,7 +1045,7 @@ export default function Catalog() {
|
|||||||
onClose={() => setShowFiltersMobile(false)}
|
onClose={() => setShowFiltersMobile(false)}
|
||||||
filters={isPartsAPIMode ? dynamicFilters : catalogFilters}
|
filters={isPartsAPIMode ? dynamicFilters : catalogFilters}
|
||||||
searchQuery={searchQuery}
|
searchQuery={searchQuery}
|
||||||
onSearchChange={setSearchQuery}
|
onSearchChange={handleSearchChange}
|
||||||
filterValues={selectedFilters}
|
filterValues={selectedFilters}
|
||||||
onFilterChange={handleMobileFilterChange}
|
onFilterChange={handleMobileFilterChange}
|
||||||
/>
|
/>
|
||||||
@ -936,6 +1065,8 @@ export default function Catalog() {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{/* Сообщение об ошибке */}
|
{/* Сообщение об ошибке */}
|
||||||
{isPartsAPIMode && articlesError && (
|
{isPartsAPIMode && articlesError && (
|
||||||
<div className="flex justify-center items-center py-8">
|
<div className="flex justify-center items-center py-8">
|
||||||
@ -985,12 +1116,21 @@ export default function Catalog() {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{/* Показываем индикатор загрузки при изменении фильтров */}
|
||||||
|
{isPartsIndexMode && isFilterChanging && (
|
||||||
|
<div className="flex flex-col items-center justify-center py-12">
|
||||||
|
<LoadingSpinner />
|
||||||
|
<div className="text-gray-500 text-lg mt-4">Применяем фильтры...</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* Отображение товаров PartsIndex */}
|
{/* Отображение товаров PartsIndex */}
|
||||||
{isPartsIndexMode && (() => {
|
{isPartsIndexMode && !isFilterChanging && (() => {
|
||||||
console.log('🎯 Проверяем отображение PartsIndex товаров:', {
|
console.log('🎯 Проверяем отображение PartsIndex товаров:', {
|
||||||
isPartsIndexMode,
|
isPartsIndexMode,
|
||||||
visibleEntitiesLength: visibleEntities.length,
|
visibleEntitiesLength: visibleEntities.length,
|
||||||
visibleEntities: visibleEntities.map(e => ({ id: e.id, code: e.code, brand: e.brand.name }))
|
visibleEntities: visibleEntities.map(e => ({ id: e.id, code: e.code, brand: e.brand.name })),
|
||||||
|
isFilterChanging
|
||||||
});
|
});
|
||||||
return visibleEntities.length > 0;
|
return visibleEntities.length > 0;
|
||||||
})() && (
|
})() && (
|
||||||
@ -1001,16 +1141,20 @@ export default function Catalog() {
|
|||||||
const priceData = getPrice(productForPrice);
|
const priceData = getPrice(productForPrice);
|
||||||
const isLoadingPriceData = isLoadingPrice(productForPrice);
|
const isLoadingPriceData = isLoadingPrice(productForPrice);
|
||||||
|
|
||||||
// Определяем цену для отображения
|
// Определяем цену для отображения (все товары уже отфильтрованы на сервере)
|
||||||
let displayPrice = "Цена по запросу";
|
let displayPrice = "";
|
||||||
let displayCurrency = "RUB";
|
let displayCurrency = "RUB";
|
||||||
let priceElement;
|
let priceElement;
|
||||||
|
|
||||||
if (isLoadingPriceData) {
|
if (isLoadingPriceData) {
|
||||||
|
// Показываем скелетон загрузки вместо текста
|
||||||
priceElement = <PriceSkeleton />;
|
priceElement = <PriceSkeleton />;
|
||||||
} else if (priceData && priceData.price) {
|
} else if (priceData && priceData.price) {
|
||||||
displayPrice = `${priceData.price.toLocaleString('ru-RU')} ₽`;
|
displayPrice = `${priceData.price.toLocaleString('ru-RU')} ₽`;
|
||||||
displayCurrency = priceData.currency || "RUB";
|
displayCurrency = priceData.currency || "RUB";
|
||||||
|
} else {
|
||||||
|
// Если нет данных о цене, показываем скелетон (товар должен загрузиться)
|
||||||
|
priceElement = <PriceSkeleton />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -1021,7 +1165,7 @@ export default function Catalog() {
|
|||||||
articleNumber={entity.code}
|
articleNumber={entity.code}
|
||||||
brandName={entity.brand.name}
|
brandName={entity.brand.name}
|
||||||
image={entity.images?.[0] || ''}
|
image={entity.images?.[0] || ''}
|
||||||
price={isLoadingPriceData ? "" : displayPrice}
|
price={priceElement ? "" : displayPrice}
|
||||||
priceElement={priceElement}
|
priceElement={priceElement}
|
||||||
oldPrice=""
|
oldPrice=""
|
||||||
discount=""
|
discount=""
|
||||||
|
146
src/pages/confidentiality.tsx
Normal file
146
src/pages/confidentiality.tsx
Normal file
@ -0,0 +1,146 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import Head from 'next/head';
|
||||||
|
import CatalogSubscribe from "@/components/CatalogSubscribe";
|
||||||
|
import MobileMenuBottomSection from "@/components/MobileMenuBottomSection";
|
||||||
|
import NewsAndPromos from "@/components/index/NewsAndPromos";
|
||||||
|
import Footer from "@/components/Footer";
|
||||||
|
import IndexTopMenuNav from "@/components/index/IndexTopMenuNav";
|
||||||
|
import MetaTags from "@/components/MetaTags";
|
||||||
|
import { getMetaByPath } from "@/lib/meta-config";
|
||||||
|
import JsonLdScript from "@/components/JsonLdScript";
|
||||||
|
import { generateOrganizationSchema, generateWebSiteSchema, PROTEK_ORGANIZATION } from "@/lib/schema";
|
||||||
|
|
||||||
|
|
||||||
|
export default function Confidentiality() {
|
||||||
|
const metaData = getMetaByPath('/');
|
||||||
|
|
||||||
|
// Добавьте эти строки:
|
||||||
|
const organizationSchema = generateOrganizationSchema(PROTEK_ORGANIZATION);
|
||||||
|
const websiteSchema = generateWebSiteSchema(
|
||||||
|
"Protek - Автозапчасти и аксессуары",
|
||||||
|
"https://protek.ru",
|
||||||
|
"https://protek.ru/search"
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<MetaTags {...metaData} />
|
||||||
|
<JsonLdScript schema={organizationSchema} />
|
||||||
|
<JsonLdScript schema={websiteSchema} />
|
||||||
|
<section className="section-info">
|
||||||
|
<div className="w-layout-blockcontainer container info w-container">
|
||||||
|
<div className="w-layout-vflex flex-block-9">
|
||||||
|
<div className="w-layout-hflex flex-block-7">
|
||||||
|
<a href="#" className="link-block w-inline-block">
|
||||||
|
<div>Главная</div>
|
||||||
|
</a>
|
||||||
|
<div className="text-block-3">→</div>
|
||||||
|
<a href="#" className="link-block-2 w-inline-block">
|
||||||
|
<div>Политика конфиденциальности</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div className="w-layout-hflex flex-block-8">
|
||||||
|
<div className="w-layout-hflex flex-block-10">
|
||||||
|
<h1 className="heading">Политика конфиденциальности</h1>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<div className="flex relative gap-8 items-start self-stretch pt-10 pb-20 max-md:p-8 max-sm:gap-5 max-sm:p-5">
|
||||||
|
<div className="flex relative flex-col gap-8 items-start p-10 bg-white rounded-3xl flex-[1_0_0] max-w-[1580px] mx-auto max-md:p-8 max-sm:gap-5 max-sm:p-5">
|
||||||
|
<div className="flex relative flex-col gap-5 items-start self-stretch max-sm:gap-4">
|
||||||
|
<div
|
||||||
|
layer-name="Объявлен старт продаж электрических насосов"
|
||||||
|
className="relative self-stretch text-3xl font-bold leading-9 text-gray-950"
|
||||||
|
>
|
||||||
|
Объявлен старт продаж электрических насосов
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
layer-name="Бренд вывел на рынок сразу широкий ассортимент, уже на старте продаж - более 100 артикулов и включает в себя позиции для брендов-лидеров автомобильного рынка, например: артикул 77WPE080 для Mercedes-Benz S-CLASS (W221, C216), артикул 77WPE096 – Land Rover DISCOVERY V (L462) / Jaguar F-PACE (X761), артикул 77WPE014 – Audi Q5 (8RB) / Volkswagen TOUAREG (7P5, 7P6)."
|
||||||
|
className="relative self-stretch text-base leading-6 text-gray-600 max-sm:text-sm"
|
||||||
|
>
|
||||||
|
Бренд вывел на рынок сразу широкий ассортимент, уже на старте
|
||||||
|
продаж - более 100 артикулов и включает в себя позиции для
|
||||||
|
брендов-лидеров автомобильного рынка, например: артикул 77WPE080
|
||||||
|
для Mercedes-Benz S-CLASS (W221, C216), артикул 77WPE096 – Land
|
||||||
|
Rover DISCOVERY V (L462) / Jaguar F-PACE (X761), артикул 77WPE014
|
||||||
|
– Audi Q5 (8RB) / Volkswagen TOUAREG (7P5, 7P6).
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex relative flex-col gap-8 items-start self-stretch max-sm:gap-5">
|
||||||
|
<div
|
||||||
|
layer-name="Преимущества электрических насосов охлаждающей жидкости MasterKit Electro:"
|
||||||
|
className="relative self-stretch text-3xl font-medium leading-9 text-gray-950"
|
||||||
|
>
|
||||||
|
Преимущества электрических насосов охлаждающей жидкости MasterKit
|
||||||
|
Electro:
|
||||||
|
</div>
|
||||||
|
<div className="flex relative flex-col gap-3.5 items-start self-stretch">
|
||||||
|
<div className="flex relative gap-10 items-start w-full max-md:gap-5 max-sm:gap-4">
|
||||||
|
<div className="relative shrink-0 mt-2 w-2 h-2 bg-gray-600 rounded-full" />
|
||||||
|
<div
|
||||||
|
layer-name="Отличная производительность за счёт применения компонентов известных мировых брендов."
|
||||||
|
className="relative text-base leading-6 text-gray-600 flex-[1_0_0] max-sm:text-sm"
|
||||||
|
>
|
||||||
|
Отличная производительность за счёт применения компонентов
|
||||||
|
известных мировых брендов.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex relative gap-10 items-start w-full max-md:gap-5 max-sm:gap-4">
|
||||||
|
<div className="relative shrink-0 mt-2 w-2 h-2 bg-gray-600 rounded-full" />
|
||||||
|
<div
|
||||||
|
layer-name="Герметичность и устойчивость к коррозии"
|
||||||
|
className="relative text-base leading-6 text-gray-600 flex-[1_0_0] max-sm:text-sm"
|
||||||
|
>
|
||||||
|
Герметичность и устойчивость к коррозии
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex relative gap-10 items-start w-full max-md:gap-5 max-sm:gap-4">
|
||||||
|
<div className="relative shrink-0 mt-2 w-2 h-2 bg-gray-600 rounded-full" />
|
||||||
|
<div
|
||||||
|
layer-name="Высококачественные материалы компонентов, обеспечивающие долгий срок службы"
|
||||||
|
className="relative text-base leading-6 text-gray-600 flex-[1_0_0] max-sm:text-sm"
|
||||||
|
>
|
||||||
|
Высококачественные материалы компонентов, обеспечивающие
|
||||||
|
долгий срок службы
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex relative gap-10 items-start w-full max-md:gap-5 max-sm:gap-4">
|
||||||
|
<div className="relative shrink-0 mt-2 w-2 h-2 bg-gray-600 rounded-full" />
|
||||||
|
<div
|
||||||
|
layer-name="Широкий ассортимент – более 100 артикулов"
|
||||||
|
className="relative text-base leading-6 text-gray-600 flex-[1_0_0] max-sm:text-sm"
|
||||||
|
>
|
||||||
|
Широкий ассортимент – более 100 артикулов
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
layer-name="На электрические насосы системы охлаждения MasterKit Electro предоставляется гарантия 1 год или 30.000 км пробега, в зависимости от того, что наступит раньше. Все новинки уже внесены в каталог подбора продукции и доступны для заказа."
|
||||||
|
className="relative self-stretch text-base leading-6 text-gray-600 max-sm:text-sm"
|
||||||
|
>
|
||||||
|
На электрические насосы системы охлаждения MasterKit Electro
|
||||||
|
предоставляется гарантия 1 год или 30.000 км пробега, в
|
||||||
|
зависимости от того, что наступит раньше. Все новинки уже внесены
|
||||||
|
в каталог подбора продукции и доступны для заказа.
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
layer-name="ABig_Button"
|
||||||
|
data-component-name="ABig_Button"
|
||||||
|
data-variant-name="Button big=Default"
|
||||||
|
className="relative gap-2.5 px-10 py-6 text-lg font-medium leading-5 text-center text-white no-underline bg-red-600 rounded-xl transition-all cursor-pointer border-[none] duration-[0.2s] ease-[ease] w-fit max-sm:px-8 max-sm:py-5 max-sm:w-full hover:bg-red-700"
|
||||||
|
>
|
||||||
|
Перейти к товару
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<section className="section-3">
|
||||||
|
<CatalogSubscribe />
|
||||||
|
</section>
|
||||||
|
<Footer />
|
||||||
|
<MobileMenuBottomSection />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
@ -9,7 +9,7 @@ async function testPartsIndexAPI() {
|
|||||||
|
|
||||||
// Получаем каталоги
|
// Получаем каталоги
|
||||||
console.log('\n📦 Получаем список каталогов...');
|
console.log('\n📦 Получаем список каталогов...');
|
||||||
const catalogsResponse = await fetch('https://api.parts-index.com/v1/catalogs?lang=ru', {
|
const catalogsResponse = await fetch(process.env.PARTSAPI_URL+"/v1/catalogs?lang=ru", {
|
||||||
headers: {
|
headers: {
|
||||||
'Accept': 'application/json',
|
'Accept': 'application/json',
|
||||||
},
|
},
|
||||||
@ -31,7 +31,7 @@ async function testPartsIndexAPI() {
|
|||||||
console.log(`\n🎯 Получаем группы для каталога "${firstCatalog.name}"...`);
|
console.log(`\n🎯 Получаем группы для каталога "${firstCatalog.name}"...`);
|
||||||
|
|
||||||
const groupsResponse = await fetch(
|
const groupsResponse = await fetch(
|
||||||
`https://api.parts-index.com/v1/catalogs/${firstCatalog.id}/groups?lang=ru`,
|
`${process.env.PARTSAPI_URL}/v1/catalogs/${firstCatalog.id}/groups?lang=ru`,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
'Accept': 'application/json',
|
'Accept': 'application/json',
|
||||||
|
21
user_input.py
Normal file
21
user_input.py
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
def main():
|
||||||
|
while True:
|
||||||
|
print("\n" + "="*50)
|
||||||
|
user_input = input("Please provide feedback or next task (type 'stop' to exit): ").strip()
|
||||||
|
|
||||||
|
if user_input.lower() == 'stop':
|
||||||
|
print("Exiting task loop. Thank you!")
|
||||||
|
break
|
||||||
|
elif user_input.lower() == '':
|
||||||
|
print("Please provide some input or type 'stop' to exit.")
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
print(f"\nReceived input: {user_input}")
|
||||||
|
print("Processing your request...")
|
||||||
|
# Here the main process would handle the user's input
|
||||||
|
return user_input
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
result = main()
|
||||||
|
if result and result.lower() != 'stop':
|
||||||
|
print(f"Next task received: {result}")
|
Reference in New Issue
Block a user