5 интеграций в React: Контент + Дизайн + Разработка
- суббота, 22 июля 2023 г. в 00:00:15
Привет всем! Сколько раз вы уже сталкивались с запросами от бизнеса о необходимости ускорить разработку, внедряя интеграции со сторонними сервисами? И неужели не звучит заманчиво возможность менять дизайн и контент независимо, без необходимости привлекать разработчика?
На данный момент уже существуют привычные инструменты, позволяющие разным членам команды, таким как разработчики, дизайнеры и контент-менеджеры, эффективно взаимодействовать. Например, Tilda, Wordpress и множество других решений предоставляют удобные средства для работы с различными аспектами проекта. Однако возникает сложность, когда необходимо интегрировать результаты работы этих инструментов внутрь React приложения. Идеальным вариантом было бы обеспечить взаимодействие нашего приложения с контентом и возможность перехвата событий для оптимальной интеграции и гибкости.
В статье я продемонстрирую, как решить три задачи, связанные с интеграциями в React приложении:
Изменять контент страницы на React без необходимости привлечения разработчика;
Расширять функционал React приложения с помощью сторонних сервисов;
Создавать страницы с минимальными затратами на разработку.
Эти подходы позволят нам обеспечить гибкость и доступность нашего приложения, а также значительно ускорить процесс разработки. Давайте рассмотрим каждую из этих задач подробнее и выявим оптимальные пути их решения.
В данной статье я представлю различные способы интеграций, которые помогут экономить время разработчиков, дадут больше свободы дизайнерам и контент-менеджерам.
Обычно, от сторонних решений мы ожидаем следующего:
Легкость освоения: инструмент должен быть понятен и прост в освоении для всех сторон — как его внедрить и поддерживать, как управлять визуальной частью и контентом;
Гибкость: хотя невозможно создать универсальное решение для всех случаев, инструмент должен выполнять свою функцию и, в идеале, предоставлять возможности для расширения функционала;
Уровень взаимодействия: мы хотим предоставить пользователям насыщенные интерактивом страницы. Возможно ли динамическое отображение различных элементов в зависимости от состояния React приложения, таких как авторизация, и вызов функций из основного приложения?
Легкость интеграции: насколько прост и быстр процесс первого и последующих деплоев. Можно ли автоматизировать этот процесс?
Очевидность интеграции: некоторые инструменты предназначены для интеграции, в то время как другие — нет. Сколько усилий потребуется для отображения и взаимодействия с интегрированным контентом?
На примерах мы затронем следующие вопросы:
Как отображать результаты работы инструмента внутри нашего React приложения в специальном контейнере?
Как подписываться на события из React?
Какие возможности есть для Server Side Rendering?
Стоит отметить, что я приведу примеры решений на конкретных продуктах, таких как Tilda, Hubspot, Typeform, Contentful, Builder.io, но существуют их аналоги, работающие похожим образом. Это позволит нам наглядно рассмотреть различные подходы к интеграции и их применение в практике.
Если нашей целью является лишь вынос управления данными в админку, а все остальные компоненты приложения хотим “делать по-старинке” (путем написания кода и создания дизайна в редакторе), то оптимальным решением может быть использование Headless CMS. Такой подход позволяет хранить данные в SaaS решениях, получать их через SDK или напрямую и затем отрисовывать их на клиентской стороне с помощью собственных усилий.
Примерами таких программ являются Strapi и Contentful. В данной статье мы рассмотрим подход интеграции с использованием последнего. Одной из главных задач этого подхода является решение проблемы хардкода данных в верстку. При изменении требований со стороны бизнеса, необходимо лезть в код и выполнять деплой, что может занимать много времени.
В парадигме Headless CMS мы сначала настраиваем модель данных. Например, у нас может быть модель "Статья", которая имеет поля "Заголовок" и "Текст статьи". Этот этап работы может выполнять разработчик. Затем контент-менеджеры могут создавать конкретные экземпляры сущностей на основе этой модели, такие как "Статья 1", "Статья 2" и так далее.
Далее мы рассмотрим примеры реализации такого подхода, чтобы показать, как Contentful может помочь решить проблему хардкода данных и сделать процесс обновления и изменения контента более удобным и эффективным для команды.
В случае с Contentful, мы можем получать данные через стандартное Web API с помощью функции fetch
. После добавления ключей доступа, необходимо выполнить запрос к Contentful API и сохранить полученные данные локально для дальнейшего использования:
const [page, setPage] = useState(null);
useEffect(() => {
window
.fetch(
`https://graphql.contentful.com/content/v1/spaces/${process.env.REACT_APP_CONTENTFUL_SPACE}/`,
{
method: "POST",
headers: {
"Content-Type": "application/json",
// Authenticate the request
Authorization: `Bearer ${process.env.REACT_APP_CONTENTFUL_API_KEY}`,
},
// send the GraphQL query
body: JSON.stringify({ query }),
}
)
.then((response) => response.json())
.then(({ data, errors }) => {
if (errors) {
console.error(errors);
}
// rerender the entire component with new data
setPage(data.blogPostCollection.items[0]);
});
}, []);
После того, как мы получили данные из Contentful и сохранили их локально, дальнейшая работа с ними становится привычной и удобной в React компонентах. Мы можем использовать данные в компонентах так же, как и с любыми другими данными, и отрисовывать их на странице с помощью JSX
:
return (
<div className={"bg-white rounded-2xl p-6"}>
<h1 className={"text-6xl text-gray-900"}>{page.title}</h1>
{page.body.json.content.map((item) => (
<p className={"text-2xl m-5 p-2 text-gray-800"}>
{item.content[0].value}
</p>
))}
</div>
);
Взаимодействие с кнопками и обработка событий в React приложении с использованием данных из Contentful не отличается от привычного паттерна React. Мы можем легко добавить кнопку и передать обработчик onClick
в компоненте, чтобы реагировать на действия пользователя.
Тут все хорошо. На этапе рендера контента на сервере мы можем выполнять запросы к Contentful и отправлять клиенту уже готовые данные. В частности, если мы используем Next.js, то можем получать данные на уровне функций getServerSideProps
(для Server Side Rendering) или getStaticProps
(для Static Site Generation). Это позволяет нам предоставлять пользователю быстрый и оптимизированный контент с учетом данных, полученных из Contentful.
В этом аспекте также все идет гладко. Данные загружаются каждый раз при запросе, что позволяет избежать необходимости редеплоя приложения. Единственное, если вы используете подход с Static Site Generation, вам потребуется использовать вебхуки для наблюдения за изменениями контента. С помощью вебхуков вы будете автоматически получать уведомления о внесенных изменениях в Contentful, что позволит поддерживать актуальность контента на вашем сайте без необходимости ручного вмешательства.
Плюсы:
Контент-менеджеры могут обновлять контент на страницах без необходимости привлекать разработчиков, что значительно экономит время команды;
Изменения происходят в реальном времени. Это особенно критично, если деплой приложения занимает значительное время, так как позволяет мгновенно обновлять контент без перезагрузки всего приложения;
Возможность решать задачи, связанные с meta-тегами для страницы, такие как заголовок, описание, ключевые слова и другие метаданные.
Минусы:
При использовании большого количества полей в Contentful, необходимо заботиться о их поддержке и осмысленных названиях, особенно если на странице присутствуют множество заголовков и различных данных;
Несмотря на то, что мы вынесли управление контентом в админку, верстать всё равно необходимо вручную. Если дизайн изменяется, потребуется внесение изменений в код приложения. Также, если появляются новые переменные или блоки контента, их придется добавить в кодовую базу.
Здесь можно прочитать подробнее про интеграцию React и Contentful.
В заключение, подход с использованием Headless CMS, таких как Contentful, обеспечивает эффективное управление контентом, экономит время разработчиков и позволяет быстро обновлять контент на сайте. Однако, необходимо учитывать некоторые недостатки, такие как управление множеством полей и верстка, которые могут потребовать дополнительных усилий и внимания при использовании этого подхода.
Пример | Код
Если у нас есть сервис, который можно отобразить внутри iframe
, и у него не установлены заголовки безопасности типа x-frame-options: SAMEORIGIN
или Content-Security-Policy: frame-ancestors none
, то мы можем внедрить контент этого сервиса на свой сайт, отображая его внутри "окошка" iframe
. Некоторые сервисы, такие как Typeform, предлагают интеграцию через iframe
и предоставляют SDK для удобной работы с ними. Однако, для веб-сайтов, где отсутствует готовое SDK, если мы хотим добавить интерактивность и взаимодействие с родительским приложением, необходимо использовать механизм postMessage
.
Рассмотрим ситуацию, когда у нас уже имеется SDK для интеграции в наше React приложение другого веб-ресурса. На примере Typeform, предоставляющего SDK для удобной интеграции, мы рассмотрим этот подход.
Возможности отображения Typeform на сайте разнообразны, такие как слайдер, всплывающее окно и многое другое. Главное, что нам потребуется сделать - это установить пакет @typeform/embed-react
и добавить соответствующий виджет на страницу:
<Widget
id="iUesUyX7"
style={{ width: "100%" }}
className="my-form"
onQuestionChanged={() => {
alert("onQuestionChanged");
}}
/>
Чтобы влиять на контент внутри страницы, мы можем использовать механизм hidden fields в Typeform. Передавая данные через скрытые поля, мы можем взаимодействовать с содержимым формы и использовать их для дополнительной обработки и анализа.
Если нам необходимо реагировать на события, происходящие в Typeform, мы можем передать обработчики событий onSubmit
и onReady
в Widget
, как обычно мы это делаем для React компонентов. Однако, следует отметить, что при работе с SDK Typeform есть некоторые ограничения, и мы не можем добавить собственные компоненты напрямую в форму.
У нас возникают ограничения с использованием SSR, что приводит к невозможности индексации контента, отображенного через iframe
. Это ограничение распространяется на любое решение, использующее данную технологию.
Обратите внимание, что отсутствие поддержки Server Side Rendering может повлиять на SEO вашего сайта, поскольку контент, загружаемый через iframe
, недоступен для поисковых роботов и поисковых систем.
Так как мы получаем актуальное содержание в iframe
каждый раз при запросе, то несем ответственность только за отображение тега. Обновления контента происходят со стороны стороннего сервиса, что позволяет нам автоматически получать самую свежую информацию без необходимости вмешательства с нашей стороны. Это облегчает процесс поддержания актуальности контента на нашем сайте, поскольку изменения в источнике автоматически отражаются в iframe
.
Плюсы:
Если продукт поддерживает интеграцию, то работа через iframe
позволит решить бизнес-задачу буквально за несколько строчек кода;
Изменения содержимого страницы автоматически появляются в iframe
.
Минусы:
Не работает SSR;
Не все решения предоставляют SDK, что заставляет писать код-обертку. Это может добавить новые головные боли и увеличить затраты времени.
Если мы хотим отобразить сайт внутри iframe
и одновременно взаимодействовать с нашим React приложением, то нам придется использовать postMessage
. Важно отметить, что для этого необходимо иметь возможность добавить собственный HTML на страницу, желательно в <head>
.
Механизм postMessage
позволяет установить безопасное взаимодействие между родительским и встроенным в iframe
приложением. С помощью postMessage
мы можем отправлять сообщения и передавать данные между родительским и встроенным окном, даже если они находятся на разных доменах.
Мы добавим код обработки запросов от родительского приложения, а также код для отправки событий со страницы Hubspot. Поскольку Hubspot не предоставляет официального SDK для интеграции, нам приходится использовать альтернативный подход. Хотя данный способ не является официальным, он представляет собой один из доступных вариантов для взаимодействия с Hubspot через iframe
.
Со стороны React приложения мы добавим ссылку на Hubspot в iframe
и передадим frameRef
, через который будем взаимодействовать с сервисом:
<iframe
title={"hubspot"}
src={HUBSPOT_URL}
className={"w-full h-full rounded-2xl"}
ref={frameRef}
/>
В следующем примере кода мы реализовали обработку ссылок и отдельных кнопок в Hubspot, которые мы определили в hubspotConfig
. Это позволяет перехватывать и обрабатывать события со стороны сервиса. Кроме того, мы можем добавить конфигурацию для управления поведением страницы Hubspot, такую как включение или отключение определенного функционала:
const hubspotConfig = useMemo(
() => ({
handlers: [
{
type: "click-button",
name: CTA_BUTTON_NAME,
selector: "a[href='#custom']",
},
],
}),
[]
);
Затем мы подпишемся в useEffect
на событие message
из iframe
:
useEffect(() => {
const handler = (event) => {
if (!event.origin.includes("hubspot") || !event.data.type) { // #1
return;
}
switch (event.data.type) {
case "click-button": { // #2
if (
event.data.payload.buttonName === "link" &&
event.data.payload.href
) {
window.location.href = event.data.payload.href;
break;
}
alert(`Button: ${event.data.payload.buttonName}`);
break;
}
case "load": {
if (hubspotConfig && frameRef.current?.contentWindow) {
frameRef.current.contentWindow.postMessage(
createAction("set-config", { config: hubspotConfig }), // #3
"*"
);
}
break;
}
default: {
console.log("There is not event type");
}
}
};
window.addEventListener("message", handler, false);
return () => {
window.removeEventListener("message", handler);
};
}, [...])
Из интересного:
Необходимо фильтровать события, проверяя поле origin
, используя условие event.origin.includes("hubspot")
. Это важно, так как событие message
может приходить из различных iframe
и workers
, и нам нужно обрабатывать только те, которые исходят от Hubspot;
Когда мы получаем действие с типом click-button
, сформированное на стороне Hubspot, мы обрабатываем клик в React. Таким образом, мы можем реагировать на действия, происходящие внутри iframe
, и синхронизировать их с React приложением;
Мы отправляем действие set-config
только после загрузки iframe
(тип action load
- подробности ниже). Это позволяет нам сообщить Hubspot, какие кнопки мы хотим дополнительно обрабатывать.
В админке Hubspot для страницы мы добавляем в head
следующий скрипт:
(function () {
const createAction = (type, payload) => {
return {
type,
payload,
};
};
const sendMessage = (action) => {
window.parent.postMessage(action, "*");
};
window.addEventListener("DOMContentLoaded", () => { // #1
window.addEventListener( // #2
"message",
(event) => {
if (event.data.type === "set-config") { // #4
const config = event.data.payload.config;
config.handlers?.forEach((handler) => {
const element = document.querySelector(handler.selector);
switch (handler.type) {
case "click-button": {
element?.addEventListener("click", (e) => {
e.preventDefault();
sendMessage(
createAction("click-button", {
buttonName: handler.name,
})
);
});
break;
}
default: {
// eslint-disable-next-line no-console
console.log("There is not the event: " + handler.type);
}
}
});
const links = document.querySelectorAll("a");
links.forEach((link) => {
link.onclick = "event.preventDefault()";
link.addEventListener("click", (e) => {
e.preventDefault();
sendMessage(
createAction("click-button", {
buttonName: "link",
href: link.href,
})
);
});
});
}
},
false
);
sendMessage(
createAction("load", { // #3
title: document.title,
description:
document.querySelector('meta[name="description"]')?.content || "",
})
);
});
})();
Порядок действий:
iframe
загружается, и событие DOMContentLoaded
срабатывает;
Мы подписываемся на событие message
в родительском приложении;
Отправляем action
с типом load
для сообщения React приложению о готовности этого скрипта к приему config
. Здесь стоит отметить, что мы можем передать метаданные страницы, чтобы добавить их на уровне React приложения в раздел <head>
для SEO;
При получении set-config
мы проходим по всем ссылкам и кнопкам, указанным в config
, и добавляем обработчики click
. Теперь при нажатии на элемент мы отправляем родительскому приложению информацию об этом событии.
Из-за использования iframe
у нас возникают проблемы с SSR, и контент не может быть проиндексирован поисковыми системами.
Причина заключается в том, что содержимое iframe
загружается динамически на стороне клиента, а не на сервере. В результате, при обращении поисковой системы к странице, она не может увидеть содержимое iframe
, так как оно не доступно в HTML-разметке, которая передается с сервера.
Это ограничение может негативно сказаться на SEO и влиять на индексацию контента, что делает использование iframe
неподходящим при проектировании приложений с точки зрения доступности и поисковой оптимизации.
Подобно примеру с Typeform, в данном случае также невозможно предсказать, что может быть загружено внутри iframe
, так как содержимое может меняться. Контент в iframe
будет всегда актуальным, и мы должны взаимодействовать с ним на основе информации, полученной через postMessage
и общий config
.
Однако, важно отметить, что необходимо своевременно обновлять "контракт" между родительским React приложением и iframe
. Если внутри iframe
происходят изменения, которые могут повлиять на обмен сообщениями или использование общего config
, необходимо обеспечить синхронизацию и соответствующие обновления в обоих приложениях. Это позволит гарантировать правильное функционирование интеграции и сохранение актуальной информации.
Плюсы:
Возможность реализовать функциональность любой сложности, используя возможности как нашего React приложения, так и стороннего сервиса;
Не требуется привлекать разработчика для изменения содержимого страницы.
Минусы:
Отсутствие поддержки SSR, что может негативно сказаться на SEO;
Необходимость добавить код-обертку для обмена сообщениями через postMessage
. Этот подход может быть сложен для людей, не знакомых с данным функционалом, и может вызывать трудности при интеграции и отладке.
Подход с использованием iframe
и postMessage
обладает определенными преимуществами и ограничениями, и выбор данного подхода зависит от требований проекта и опыта разработчиков, которые будут работать с ним.
Иногда возникает ситуация, когда у нас имеется готовая верстка, экспортированная с другого сайта, и мы желаем ее интегрировать в наше React приложение. Однако, в данном случае использование iframe
или прокси для взаимодействия со сторонним сервисом не подходит.
Если наш сайт имеет в HTTP-заголовках x-frame-options: SAMEORIGIN|DENY
или Content-Security-Policy: frame-ancestors none
, то нам будет невозможно отобразить его внутри iframe
. В таком случае, придется решить задачу иначе - путем копирования разметки, включая все стили и скрипты, и затем попытаться вставить ее в наше React приложение.
Для демонстрации данного способа интеграции мы воспользуемся платформой Tilda. Этот сервис предоставляет возможность экспорта файлов, которые затем можно разместить на собственном хостинге. Важно отметить, что официально запрещено изменять контент, который мы экспортировали, но в данном примере мы модифицируем его исключительно в образовательных целях.
Чтобы начать, необходимо скачать zip-архив из настроек вашего сайта. После успешного экспорта файлов, надо переместить их в соответствующую директорию для статических ресурсов.
В данном примере Tilda содержит значительное количество скриптов, что может усложнить задачу интеграции, особенно при работе с изображениями. Проблема возникает из-за использования "магических" вызовов функций, отвечающих за lazy loading изображений. К сожалению, такой подход не совместим с React routing, так как они предполагают запуск при инициализации страницы, а не при динамическом переключении урлов. Эта проблема решается небольшим изменением скриптов.
Так же стоит позаботится о смене всех урлов для статики(картинки, скрипты, стили) с домена Tilda на свой собственный. Добавить ссылки в index.html
на assets
(js, css). js/lazyload-1.3.min.js
превращается в %PUBLIC_URL%/tilda/js/lazyload-1.3.min.js
.
В коде нам необходимо поместить наш html
в строковую переменную:
export const TILDA_HTML = `
<div
id="allrecords"
data-tilda-export="yes"
class="t-records"
data-hook="blocks-collection-content-node"
data-tilda-project-id="4457918"
data-tilda-page-id="33161614"
data-tilda-formskey="599a4bf5b358fcc8f5ce82f8419bbb21"
data-tilda-lazy="yes"
>
...
</div>
`
И вставить его в React компоненте, используя dangerouslySetInnerHTML
:
return (
<div
dangerouslySetInnerHTML={{ __html: TILDA_HTML }}
className={"w-full bg-white rounded-2xl overflow-hidden"}
/>
);
Для добавления обработчиков событий, мы можем использовать стандартное DOM API:
useEffect(() => {
const tildaButton = document.getElementById("tilda-button");
const onClick = () => {
alert("onClick tilda button");
};
tildaButton.addEventListener("click", onClick);
return () => {
tildaButton.removeEventListener("click", onClick);
};
}, []);
Предварительно надо добавить в разметку id или нужные классы.
Здесь все прекрасно. Мы осуществляем рендеринг html
внутри нашего компонента. Это позволяет web crawler'ам легко считывать контент на странице и индексировать его.
Каждый раз, когда происходят изменения дизайна в конструкторе, требуется ручное вмешательство для проведения всей процедуры интеграции. Необходимо скачивать файлы, размещать их в нужных директориях и менять все ссылки. Это затрудняет автоматизацию данного процесса, что может быть неудобным при частых изменениях.
Плюс:
Возможность быстрой интеграции готового лендинга в свое React приложение. Оценочное время переноса - 2 часа работы.
Минусы:
Отсутствие возможности автоматизировать процесс интеграции, требует ручного вмешательства;
Неясное поведение скриптов, что может вызывать различные проблемы;
Официальное ограничение на изменение экспортированного контента с подобных ресурсов.
Несмотря на простоту подхода, он полезен, когда у нас есть ограниченное время и требуется быстро перенести лендинг из конструктора в React приложение. Он позволяет достичь быстрых результатов, особенно при срочных проектах.
Однако, в долгосрочной перспективе, такой подход может стать сложным для поддержки и масштабирования. После переноса лендинга, любые последующие изменения в его дизайне или контенте могут потребовать повторной ручной интеграции, что снижает эффективность и гибкость разработки.
Если мы хотим использовать React компоненты в привычном формате, предоставить дизайнеру возможность работать с ними в визуальном редакторе, а контент менеджеру редактировать данные отдельно от дизайна, мы можем обратить внимание на конструкторы сайтов. Существует множество интересных решений для дизайнеров и контент менеджеров, таких как Framer или Webflow. Они обладают мощными инструментами визуального редактора и CMS, либо позволяют импортировать данные извне. Однако, сегодня я рассмотрю Builder.io, так как этот инструмент позволяет работать с React компонентами.
Работа с данными и дизайном выходит за рамки данной статьи. Давайте перейдем к сразу к коду:
const [content, setContent] = useState(null);
useEffect(() => {
builder
.get(MODEL_NAME, {
userAttributes: {
urlPath: "/builder",
},
})
.promise()
.then(setContent);
const clickHandler = () => {
alert("Click button!");
};
document.addEventListener("click-button", clickHandler);
return () => {
document.removeEventListener("click-button", clickHandler);
};
}, []);
return (
<div className={"w-full h-full"}>
<BuilderComponent model={MODEL_NAME} content={content} />
</div>
);
Для интеграции этого инструмента, нам нужно отрисовать компонент BuilderComponent
и передать в него данные, которые мы получили из builder.get
. Это все, что требуется от разработчика для успешной интеграции.
Дополнительно, мы регистрируем событие click-button
, которое будет вызываться при клике на кнопку. Конфигурация события производится в админке Builder.io. Это позволяет нам использовать событие для перехода на другие страницы или выполнения действий при нажатии на кнопку.
Кроме того, у нас есть возможность регистрировать собственные кастомные компоненты, которые станут доступны из админки Builder.io. Это позволит нам создавать и использовать настраиваемые компоненты, которые соответствуют уникальным потребностям проекта:
const CustomTicker = ({ title }) => {
// ...
return (
<div>
{title}: {count}
</div>
);
};
Builder.registerComponent(CustomTicker, {
name: "Custom Ticker",
inputs: [{ name: "title", type: "text", defaultValue: "Ticks" }],
});
После того, как мы зарегистрировали компонент, его можно легко добавить на страницу из левого меню Builder.io. Это предоставляет возможность удобно выбирать и размещать настраиваемые компоненты на странице для дальнейшего редактирования и отображения в React приложении.
При работе с собственными компонентами, мы получаем доступ ко всему функционалу нашего приложения. Например, можно взаимодействовать с Context API для управления состоянием и передачи данных между компонентами.
Кроме того, мы можем легко прописывать собственные обработчики непосредственно в визуальном редакторе. Это позволяет подключать Headless CMS и использовать данные в наших скриптах, что дает большую гибкость и возможности при разработке и настройке компонентов.
SSR работает отлично, если мы получаем данные на сервере. В результате, мы создаем HTML, который практически неотличим от "родной" кодовой базы. Разработчики могут использовать обучающие материалы, чтобы успешно интегрировать Next.js и Builder.io, что предоставляет дополнительные возможности и преимущества при разработке и оптимизации нашего приложения.
Мы получаем контент на этапе запроса, что гарантирует актуальность информации всегда. Нет необходимости в редеплое, если мы хотим внести изменения на странице, созданной с помощью Builder.io. Это обеспечивает быструю и удобную возможность обновления контента без необходимости полного перезапуска приложения.
Плюсы:
Легкая интеграция для React разработчика. Нам просто необходимо добавить компонент на страницу и зарегистрировать свои собственные;
Возможность импортирования макетов из Figma. Однако, есть некоторые нюансы и требуется адаптация готовых макетов для экспорта;
Повышение скорости разработки. Обучив дизайнеров и контент-менеджеров данному инструменту, можно создавать страницы без необходимости вмешательства разработчика.
Минусы:
Визуальный редактор уступает другим инструментам для дизайнеров. Понимание CSS требуется, так как местами необходимо писать вставки на этом языке;
Порог входа. Наличие множества абстракций может потребовать несколько дней для изучения концепции и понимания задумки авторов.
В общем и целом, этот инструмент позволяет делегировать часть работы разработчика другим членам команды. Основное преимущество заключается в возможности четкого разделения ответственности между участниками команды и снижении взаимозависимости.
При подведении субъективных итогов по решению задачи “создания страниц”, хочется основываться на личных ощущениях от опыта интеграции. Понимая, что эти инструменты изначально предназначены для разных задач, все же есть пересечения по возможностям в рамках работы внутри React приложения:
Подход | Iframe(Hubspot) | Экспорт статики(Tilda) | Конструктор(Builder.io) |
---|---|---|---|
Обработка событий | + | + | + |
Routing | + | + | + |
Визуальный редактор | + | + | + |
Автоматизация деплоя новых версий | + | - | + |
Отделение данных от дизайна | + | - | + |
Индексация контента(SSR) | - | + | + |
SDK | - | - | + |
Документация по интеграции | - | - | + |
Безусловно, выбор инструмента зависит от конкретной задачи. Если ваша цель - обеспечить доставку контента, то подойдет Headless CMS. Если вам нужно добавить определенный функционал, например, формы, то использование iframe
с SDK может быть подходящим решением. Целью этой статьи было вдохновить на эксперименты с различными подходами и поощрить стремление решать бизнес-задачи с помощью наиболее подходящих инструментов.
Успехов в ваших экспериментах и разработке!