Интеграция CMS Webflow для сайта

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.
Разработка и обслуживание любых видов сайтов:
Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Интеграция CMS Webflow для сайта
Средняя
от 1 рабочего дня до 3 рабочих дней
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • 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

Интеграция CMS Webflow для сайта

Webflow — визуальный редактор сайтов с встроенной CMS. В отличие от Tilda, предоставляет REST API для работы с коллекциями контента: создание, чтение, обновление записей программно. Это открывает несколько сценариев: синхронизация данных из внешних систем, headless использование Webflow CMS как backend, или кастомный фронтенд поверх Webflow-данных.

Webflow Data API v2

С 2023 года Webflow перешёл на API v2. Аутентификация через OAuth2 (для приложений) или Site API Token (для интеграций):

// lib/webflow.ts
const WEBFLOW_API_TOKEN = process.env.WEBFLOW_API_TOKEN!
const SITE_ID = process.env.WEBFLOW_SITE_ID!

const BASE_URL = 'https://api.webflow.com/v2'

async function webflowFetch<T>(
  path: string,
  options: RequestInit = {}
): Promise<T> {
  const res = await fetch(`${BASE_URL}${path}`, {
    ...options,
    headers: {
      Authorization: `Bearer ${WEBFLOW_API_TOKEN}`,
      'Content-Type': 'application/json',
      ...options.headers,
    },
    next: { tags: ['webflow'] },
  })
  if (!res.ok) {
    const err = await res.json()
    throw new Error(`Webflow API error: ${err.message}`)
  }
  return res.json()
}

// Получить список коллекций сайта
export async function getCollections() {
  return webflowFetch<{ collections: WebflowCollection[] }>(
    `/sites/${SITE_ID}/collections`
  )
}

// Элементы коллекции с пагинацией
export async function getCollectionItems(
  collectionId: string,
  params: { limit?: number; offset?: number; live?: boolean } = {}
) {
  const query = new URLSearchParams({
    limit: String(params.limit ?? 100),
    offset: String(params.offset ?? 0),
    ...(params.live ? { live: 'true' } : {}),
  })
  return webflowFetch<{ items: WebflowItem[]; pagination: WebflowPagination }>(
    `/collections/${collectionId}/items?${query}`
  )
}

Типы данных Webflow CMS

Webflow хранит поля в fieldData. Поля создаются в дизайнере, каждому присваивается slug:

interface BlogPost {
  id: string
  cmsLocaleId: string
  lastPublished: string
  lastUpdated: string
  createdOn: string
  isArchived: boolean
  isDraft: boolean
  fieldData: {
    name: string           // обязательное поле Name
    slug: string           // обязательное поле Slug
    'post-body': string    // Rich Text → HTML
    'post-summary': string
    'main-image': { url: string; alt: string }
    'author': string       // reference → id другого элемента
    'publish-date': string
    'tags': string[]       // multi-reference
  }
}

Интеграция с Next.js

// app/blog/[slug]/page.tsx
import { getCollectionItems } from '@/lib/webflow'

const BLOG_COLLECTION_ID = process.env.WEBFLOW_BLOG_COLLECTION_ID!

export async function generateStaticParams() {
  const { items } = await getCollectionItems(BLOG_COLLECTION_ID)
  return items
    .filter(item => !item.isDraft && !item.isArchived)
    .map(item => ({ slug: item.fieldData.slug }))
}

export default async function PostPage({ params }: { params: { slug: string } }) {
  const { items } = await getCollectionItems(BLOG_COLLECTION_ID)
  const post = items.find(i => i.fieldData.slug === params.slug)
  if (!post) notFound()

  return (
    <article>
      <h1>{post.fieldData.name}</h1>
      <div
        className="prose"
        dangerouslySetInnerHTML={{ __html: post.fieldData['post-body'] }}
      />
    </article>
  )
}

export const revalidate = 3600

Для больших коллекций — постраничная загрузка:

export async function getAllItems(collectionId: string) {
  const allItems = []
  let offset = 0
  const limit = 100

  while (true) {
    const { items, pagination } = await getCollectionItems(collectionId, { offset, limit })
    allItems.push(...items)
    if (offset + limit >= pagination.total) break
    offset += limit
  }

  return allItems
}

Запись данных в Webflow CMS

Типичный случай — форма на сайте сохраняет лид прямо в Webflow CMS коллекцию:

export async function createCollectionItem(
  collectionId: string,
  fieldData: Record<string, unknown>,
  options: { live?: boolean } = {}
) {
  return webflowFetch(`/collections/${collectionId}/items`, {
    method: 'POST',
    body: JSON.stringify({
      isArchived: false,
      isDraft: !options.live,
      fieldData,
    }),
  })
}

// Использование
await createCollectionItem(LEADS_COLLECTION_ID, {
  name: formData.name,
  email: formData.email,
  message: formData.message,
  source: 'contact-form',
}, { live: false }) // черновик, менеджер увидит в CMS

Webhook от Webflow

// app/api/webflow-webhook/route.ts
import { revalidateTag } from 'next/cache'
import crypto from 'crypto'

export async function POST(request: Request) {
  const signature = request.headers.get('x-webflow-signature')
  const body = await request.text()

  // Верификация подписи
  const expected = crypto
    .createHmac('sha256', process.env.WEBFLOW_WEBHOOK_SECRET!)
    .update(body)
    .digest('hex')

  if (signature !== expected) {
    return new Response('Unauthorized', { status: 401 })
  }

  const payload = JSON.parse(body)
  // triggerType: collection_item_created, collection_item_changed, etc.
  if (payload.triggerType.startsWith('collection_item')) {
    revalidateTag('webflow')
  }

  return new Response('OK')
}

Webhooks настраиваются в Webflow: Site Settings → Integrations → Webhooks.

Синхронизация из внешней системы

Распространённый случай — продуктовый каталог в ERP, отображение на сайте через Webflow CMS:

// scripts/sync-products.ts
import { createCollectionItem, updateCollectionItem, getCollectionItems } from '@/lib/webflow'

async function syncProducts(erpProducts: ERPProduct[]) {
  const { items: existing } = await getCollectionItems(PRODUCTS_COLLECTION_ID)
  const existingMap = new Map(existing.map(i => [i.fieldData['sku'], i.id]))

  for (const product of erpProducts) {
    const fieldData = {
      name: product.name,
      slug: product.sku.toLowerCase(),
      'product-sku': product.sku,
      'price': product.price,
      'in-stock': product.stock > 0,
      'description': product.description,
    }

    if (existingMap.has(product.sku)) {
      await updateCollectionItem(PRODUCTS_COLLECTION_ID, existingMap.get(product.sku)!, fieldData)
    } else {
      await createCollectionItem(PRODUCTS_COLLECTION_ID, fieldData, { live: true })
    }

    // API rate limit: 60 req/min
    await new Promise(r => setTimeout(r, 1100))
  }
}

Ограничения API

  • Rate limit: 60 запросов в минуту на токен
  • CMS коллекции: лимит зависит от тарифа (от 2000 до 20 000 элементов)
  • Rich Text поле возвращает HTML, не структурированный AST
  • Нет транзакций — массовые операции нужно строить с учётом частичных сбоев

Сроки

Чтение данных из Webflow в Next.js с ISR: 2–3 дня. Двунаправленная синхронизация с внешней системой: 5–7 дней.