Інтеграція Headless Commerce з React/Next.js Storefront

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.

Розробка та обслуговування будь-яких видів сайтів:

Інформаційні сайти або веб-програми
Сайти візитки, landing page, корпоративні сайти, онлайн каталоги, квіз, промо-сайти, блоги, ресурси новин, інформаційні портали, форуми, агрегатори
Сайти або веб-програми електронної комерції
Інтернет-магазини, B2B-портали, маркетплейси, онлайн-обмінники, кешбек-сайти, біржі, дропшиппінг-платформи, парсери товарів
Веб-програми для управління бізнес-процесами
CRM-системи, ERP-системи, корпоративні портали, системи управління виробництвом, парсери інформації
Сайти або веб-програми електронних послуг
Дошки оголошень, онлайн-школи, онлайн-кінотеатри, конструктори сайтів, портали надання електронних послуг, відеохостинги, тематичні портали

Це лише деякі з технічних типів сайтів, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Пропоновані послуги
Показано 1 з 1 послугУсі 2065 послуг
Інтеграція Headless Commerce з React/Next.js Storefront
Складна
~2-4 тижні
Часті питання

Наші компетенції:

Етапи розробки

Останні роботи

  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Розробка веб-додатків для компанії FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Розробка веб-сайту для компанії БЕЛФІНГРУП
    874
  • image_ecommerce_furnoro_435_0.webp
    Розробка інтернет магазину для компанії FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Розробка веб-додатків для компанії Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851

Інтеграція Headless Commerce з React/Next.js Storefront

Next.js — стандартний вибір для headless e-commerce витрини: SSG для SEO, ISR для актуальності даних, Server Components для скорочення JS-бандла, Edge Middleware для персоналізації. Інтеграція з будь-яким Commerce API вистроюється за єдиним паттерном.

Стек та залежності

{
    "dependencies": {
        "next": "14.x",
        "react": "18.x",
        "@tanstack/react-query": "^5.0",
        "zustand": "^4.0",
        "graphql-request": "^6.0",
        "next-auth": "^4.0",
        "@vercel/analytics": "^1.0"
    },
    "devDependencies": {
        "@graphql-codegen/cli": "^5.0",
        "@graphql-codegen/typescript": "^4.0",
        "typescript": "^5.0"
    }
}

Слой абстракції Commerce Client

Незалежно від backend (Bagisto, Shopify, Medusa), створюється єдиний інтерфейс:

// lib/commerce/types.ts
export interface Product {
    id: string;
    sku: string;
    slug: string;
    name: string;
    description: string;
    price: number;
    compareAtPrice?: number;
    images: ProductImage[];
    variants: ProductVariant[];
    categories: Category[];
}

export interface CommerceClient {
    getProduct(slug: string): Promise<Product>;
    getProducts(params: ProductsParams): Promise<PaginatedProducts>;
    getCategories(): Promise<Category[]>;
    createCart(): Promise<Cart>;
    addToCart(cartToken: string, item: CartItem): Promise<Cart>;
    checkout(cartToken: string, data: CheckoutData): Promise<Order>;
}
// lib/commerce/bagisto.ts
import { GraphQLClient } from 'graphql-request';
import type { CommerceClient, Product } from './types';
import { GET_PRODUCT, GET_PRODUCTS } from './queries';

export class BagistoClient implements CommerceClient {
    private client: GraphQLClient;

    constructor() {
        this->client = new GraphQLClient(
            process.env.NEXT_PUBLIC_BAGISTO_GRAPHQL_URL!,
            {
                headers: {
                    'Accept': 'application/json',
                },
            }
        );
    }

    async getProduct(slug: string): Promise<Product> {
        const { product } = await this.client.request(GET_PRODUCT, { slug });
        return this.normalizeProduct(product);
    }

    private normalizeProduct(raw: any): Product {
        return {
            id: String(raw.id),
            sku: raw.sku,
            slug: raw.urlKey,
            name: raw.name,
            description: raw.description,
            price: parseFloat(raw.priceHtml?.finalPrice ?? raw.price),
            images: raw.images?.map(img => ({
                url: img.path,
                altText: raw.name,
            })) ?? [],
            variants: raw.variants ?? [],
            categories: raw.categories ?? [],
        };
    }
}

Сторінки каталогу з ISR

// app/products/[slug]/page.tsx (App Router)
import { commerce } from '@/lib/commerce';
import { ProductGallery } from '@/components/product/Gallery';
import { AddToCartButton } from '@/components/cart/AddToCartButton';
import { VariantSelector } from '@/components/product/VariantSelector';

interface Props {
    params: { slug: string };
}

export async function generateStaticParams() {
    const slugs = await commerce.getAllProductSlugs();
    return slugs.map(slug => ({ slug }));
}

export const revalidate = 3600;

export default async function ProductPage({ params }: Props) {
    const product = await commerce.getProduct(params.slug);

    return (
        <div className="grid grid-cols-1 lg:grid-cols-2 gap-12">
            <ProductGallery images={product.images} />
            <div>
                <h1 className="text-3xl font-bold">{product.name}</h1>
                <div className="mt-4 text-2xl">{product.price} ₴</div>
                <VariantSelector variants={product.variants} />
                <AddToCartButton productId={product.id} />
            </div>
        </div>
    );
}

// Метаданні для SEO
export async function generateMetadata({ params }: Props) {
    const product = await commerce.getProduct(params.slug);
    return {
        title: product.name,
        description: product.description.slice(0, 160),
        openGraph: {
            images: [product.images[0]?.url],
        },
    };
}

Управління станом кошика

Zustand для клієнтського стану кошика з персистентністю:

// stores/cart.ts
import { create } from 'zustand';
import { persist } from 'zustand/middleware';

interface CartState {
    cartToken: string | null;
    items: CartItem[];
    total: number;
    addItem: (productId: string, variantId?: string, qty?: number) => Promise<void>;
    removeItem: (lineId: string) => Promise<void>;
    clearCart: () => void;
}

export const useCartStore = create<CartState>()(
    persist(
        (set, get) => ({
            cartToken: null,
            items: [],
            total: 0,

            addItem: async (productId, variantId, qty = 1) => {
                let { cartToken } = get();

                if (!cartToken) {
                    const cart = await commerce.createCart();
                    cartToken = cart.token;
                    set({ cartToken });
                }

                const updatedCart = await commerce.addToCart(cartToken, {
                    productId,
                    variantId,
                    quantity: qty,
                });

                set({
                    items: updatedCart.items,
                    total: updatedCart.total,
                });
            },

            removeItem: async (lineId) => {
                const { cartToken } = get();
                if (!cartToken) return;

                const updatedCart = await commerce.removeFromCart(cartToken, lineId);
                set({ items: updatedCart.items, total: updatedCart.total });
            },

            clearCart: () => set({ cartToken: null, items: [], total: 0 }),
        }),
        { name: 'cart-storage', partialize: (state) => ({ cartToken: state.cartToken }) }
    )
);

Checkout-поток

// app/checkout/page.tsx
'use client';

import { useForm } from 'react-hook-form';
import { zodResolver } from '@hookform/resolvers/zod';
import { checkoutSchema, CheckoutFormData } from '@/lib/validations/checkout';
import { useCartStore } from '@/stores/cart';

export default function CheckoutPage() {
    const { cartToken, clearCart } = useCartStore();
    const { register, handleSubmit, formState: { errors } } = useForm<CheckoutFormData>({
        resolver: zodResolver(checkoutSchema),
    });

    const onSubmit = async (data: CheckoutFormData) => {
        if (!cartToken) return;

        // Збереження адреси
        await commerce.saveShippingAddress(cartToken, data.shipping);

        // Отримання методів доставки
        const shippingMethods = await commerce.getShippingMethods(cartToken);

        // Вибір методу та перехід до оплати
        await commerce.saveShippingMethod(cartToken, shippingMethods[0].id);

        // Редирект на платіжний шлюз
        const order = await commerce.placeOrder(cartToken, data.payment);
        clearCart();
        router.push(`/orders/${order.id}/success`);
    };

    return (
        <form onSubmit={handleSubmit(onSubmit)}>
            {/* поля форми */}
        </form>
    );
}

Пошук з Algolia або Typesense

// components/search/SearchResults.tsx
import { InstantSearch, SearchBox, Hits, Configure } from 'react-instantsearch';
import { typesenseInstantsearchAdapter } from 'typesense-instantsearch-adapter';

const searchClient = new TypesenseInstantSearchAdapter({
    server: {
        apiKey: process.env.NEXT_PUBLIC_TYPESENSE_API_KEY!,
        nodes: [{ host: 'search.example.com', port: 443, protocol: 'https' }],
    },
    additionalSearchParameters: { query_by: 'name,sku,description' },
}).searchClient;

export function ProductSearch() {
    return (
        <InstantSearch searchClient={searchClient} indexName="products">
            <Configure hitsPerPage={24} />
            <SearchBox placeholder="Пошук товарів..." />
            <Hits hitComponent={ProductHit} />
        </InstantSearch>
    );
}

SEO: структуровані дані

// components/product/ProductSchema.tsx
export function ProductSchema({ product }: { product: Product }) {
    const schema = {
        '@context': 'https://schema.org',
        '@type': 'Product',
        name: product.name,
        sku: product.sku,
        offers: {
            '@type': 'Offer',
            price: product.price,
            priceCurrency: 'UAH',
            availability: 'https://schema.org/InStock',
        },
        image: product.images.map(i => i.url),
    };

    return (
        <script
            type="application/ld+json"
            dangerouslySetInnerHTML={{ __html: JSON.stringify(schema) }}
        />
    );
}

Продуктивність: Core Web Vitals

Техніка LCP CLS INP
ISR для сторінок товарів +
next/image з blur placeholder + +
Предзавантаження шрифтів + +
Server Components для каталогу +
Skeleton-заглушки для кошика +
Prefetch для hover-станів +

Терміни розробки витрини

Компонент Термін
Каталог + сторінка товара 2-3 тижні
Кошик + checkout 1-2 тижні
Особистий кабінет 1 тиждень
Пошук + фільтри 1-2 тижні
Інтеграції (аналітика, пікселі) 3-5 днів
Разом 5-9 тижнів