Сегодня WordPress — один из лучших бесплатных инструментов на рынке. Как и многие IT-компании, мы начинали с разработки сайтов на коробочных решениях с низким порогом входа. Расширив экспертизу и собрав мощную команду, мы научились профессионально «готовить» WordPress и теперь грамотно используем его, чтобы облегчить жизнь себе и нашим клиентам. Коммерческая разработка присутствует там, где есть хорошие разработчики и мощный менеджмент, а инструменты могут быть любыми.В этой статье мы расскажем, как строим архитектуру наших приложений на WordPressЛюбая CMS дает нам админку и ядро, а из ядра выделяется набор функций, которые мы можем использовать для разработки приложений. WordPress дает нам отличный движок на PHP (новые версии постоянно поддерживаются) с хорошей популярной админкой и неограниченным количеством расширений, а всю клиентскую часть мы пишем самостоятельно.Мы подходим к этому максимально ответственно: продумываем архитектуру, применяем особые подходы к разработке и устанавливаем определенные правила:Первое — мы используем систему готовых блоков и компонентов. У WordPress есть редактор Gutenberg (рассказывали об этом в одной из статей), и практически все страницы формируются из блоков, как в Tilda. Этот же принцип переносим и в построение архитектуры: формируем готовые блоки, shared-компоненты, глобальные пейдж-темплейты, отдельно бизнес-фичи и т д.Второе — мы не используем JQL, пишем все на чистом JS с использованием объектно-ориентированного подхода к построению логики. Для создания сложных интерфейсов можем на отдельных страницах прикрутить Vue или React (который на WP доступен из коробки). К тому же в WP доступен REST API, и если у нас нет каких-то методов или эндпоинтов, мы можем легко их добавить.Структура папок и файлов внутри темы WordPressФайлы в корне темы:1. style.css - обязательный файл, содержит информацию о теме.В нем указываются такие данные, как:название темыверсия темыминимальная версия WordPressверсия PHPПодробнее тут2. theme.json - настройки темы (типографика, цветовая палитра)Демонстрация настроек:Пример файла настроек theme.json{ "version": 1, "settings": { "typography": { "customFontSize": false, "lineHeight": true, "fontSizes": [ { "slug": "h1", "size": "6rem", "name": "H1" }, { "slug": "h2", "size": "2.88rem", "name": "H2" }, { "slug": "h3", "size": "1.5rem", "name": "H3" }, { "slug": "text_46", "size": "2.88rem", "name": "Text 46" }, { "slug": "text_24", "size": "1.5rem", "name": "Text 24" }, { "slug": "text_16", "size": "1rem", "name": "Text 16" } ] }, "spacing": { "spacingSizes": [ { "size": "clamp(1.5rem, 5vw, 2rem)", "slug": "30", "name": "1" }, { "size": "clamp(1.8rem, 1.8rem + ((1vw - 0.48rem) * 2.885), 3rem)", "slug": "40", "name": "2" }, { "size": "clamp(2.5rem, 8vw, 6.5rem)", "slug": "50", "name": "3" } ], "blockGap": true, "customPadding": true, "customMargin": true, "units": [ "px", "rem", "%" ] }, "color": { "palette": [ { "slug": "black", "color": "#000000", "name": "Black" }, { "slug": "pink", "color": "#F2CFCE", "name": "Pink" }, { "slug": "white", "color": "#ffffff", "name": "White" }, { "slug": "grey", "color": "#A5A5A5", "name": "Grey" } ] } } }3. functions.php - входной файл для всех скриптов PHPТут выполняем объявления глобальных констант, подключение load-файла из папки includes (где лежат все основные PHP-скрипты), подключения load-файла из папки blocks и components4. header.php - шаблон шапки сайтаТут выполняется функция wp_head(), которая подключает ресурсы и выводит метатеги5. footer.php - шаблон подвала сайтаТут выполняется функция wp_footer(), которая подключает JS-скрипты темы и плагинов6. page.php - дефолтный шаблон страницыКак правило, тут выполняется подключение header.php и footer.php c помощью функций get_header() и get_footer() соответственно:get_header(); if ( have_posts() ) { while ( have_posts() ) { the_post(); the_content(); } } get_footer();7. package.json - зависимости приложения8. index.php - пустой индексный файлПапки:Theme_folder ├── acf-json │ ├── group_64c25ae72741a.json │ └── ... ├── assets │ ├── css │ ├── js │ │ ├── config │ │ ├── admin.js │ │ ├── editor.js │ │ └── index.js │ └── resource ├── blocks │ ├── block1 │ │ ├── block.json │ │ ├── functions.php │ │ ├── template.php │ │ ├── _index.scss │ │ └── _index.js │ └── load.php ├── components │ ├── component1 │ │ ├── [component-name].php │ │ ├── functions.php │ │ ├── _index.scss │ │ └── _index.js │ └── load.php ├── includes │ ├── [module-name].php │ └── load.php ├── page-templates ├── footer.php ├── functions.php ├── header.php ├── index.php ├── package.json ├── page.php ├── style.css └── theme.jsonacf-json — папка для автосохранения конфигов полей и страниц с опциями, созданных плагином Advanced Custom Fields. Это необходимо, если разработка ведется в команде, все изменения полей будут отслеживаться в гите;assets — тут лежат все стили, скрипты, шрифты и прочие файлы, отвечающие за Frontend;blocks — кастомные Gutenberg-блоки;components — переиспользуемые компоненты;includes — папка со всеми скриптами РНР;page-templates — шаблоны страниц.Gutenberg blocksВ этой папке хранятся сами блоки в подпапках и файл load.php, в котором находится функция регистрации всех блоков.Блок представлен в виде папки со следующими файлами:block.json — данные для блока, такие как name, title, и т. д.;template.php — файл с разметкой блока;functions.php — PHP-функции, используемые только в рамках этого блока;_index.scss — стили для блока;_index.js — скрипты для блока.Пример файла block.json:{ "name" : "mytheme/custom-block", "title" : "Custom Block", "icon": "admin-site", "description": "My awesome custom block.", "apiVersion": 3, "textdomain": "mytheme", "supports": { "align" : false, "mode" : false, "jsx" : true, "anchor": true }, "acf": { "mode": "preview", "renderTemplate": "./template.php" } }Файл load.php в корне папки blocks:add_action( 'init', 'mytheme_block_registration' ); function mytheme_block_registration() { foreach(glob(THEME_DIR . '/blocks/*', GLOB_ONLYDIR) as $dir){ register_block_type( $dir ); $dir_exploded = explode( '/', $dir ); $block = $dir_exploded[ count( $dir_exploded ) - 1 ]; //Подключение файла functions.php для каждого блока if ( file_exists( get_template_directory() . '/blocks/' . $block . '/functions.php' ) ) { include_once get_template_directory() . '/blocks/' . $block . '/functions.php'; } } }ComponentsПримерно такая-же концепция как с Gutenberg-блоками, набор компонентов представленных в виде папок со следующими файлами:[component-name].phpfunctions.php_index.scss_index.jsИ файл load.php, который подключает все PHP-скрипты всех компонентов:foreach(glob(THEME_DIR . '/components/*/functions.php') as $file){ require_once $file; }Используем компоненты с помощью встроенной функции get_template_part(), куда передаем параметром args входные данные:get_template_part( 'components/checkbox/checkbox', args: [ 'label' => 'Сheckbox label', 'name' => 'checkbox-name', ] );Внутри файла компонента обозначаем параметры по умолчанию, затем объединяем их с входными параметрами и распаковываем в переменные, на основе которых и строим разметку:// Fix PhpStorm inspection on undefined variable. if ( empty( $args ) ) { $args = []; } $defaults = [ 'label' => '', 'checked' => false, 'disabled' => false, 'custom_class' => '', 'id' => '', 'name' => '', 'value' => '', ]; // Fill args with defaults to avoid errors. $args = mytheme_parse_args( $args, $defaults ); // Unpack arguments to variables for better readability. // Should be in the same order as the keys in `$defaults` array. [ 'label' => $label, 'checked' => $checked, 'disabled' => $disabled, 'custom_class' => $custom_class, 'id' => $id, 'name' => $name, 'value' => $value, ] = $args;AssetsВсе Frontend-файлы, которые не попали ни в components ни в blocks:css — папка с общими стилями, глобальными лэйаутами, переменными и миксинами;js — папка с утилитарными JS-функциями и настройками webpack;resource — папка с прочими файлами, такими как картинки, шрифты, видео и т. д.В js/config лежат настройки webpack, они могут отличаться от проекта к проекту, но основной принцип такой, что у нас есть 3 точки входа:index.js — основная точка входа, используемая во фронт-енд части приложения;admin.js — специфические стили и скрипты для admin-части приложения;editor.js — специфические стили и скрипты для области редактирования.После компиляции создается по 3 CSS и JS-файла, которые мы подключаем в соответсвующих хуках WordPress:wp_enqueue_scripts — стили и скрипты для фронтенда;enqueue_block_editor_assets — стили и скрипты для редактора;admin_enqueue_scripts — стили и скрипты для админ-части сайта.В index.js нужно подключить все файлы блоков и компонентов, выглядит это примерно так:import '../css/style.scss'; function importAll(r) { r.keys().forEach(r); } importAll(require.context('./../../assets/resource/fonts/', true, /\.(woff|woff2|eot|ttf|otf)$/)); // Импортируем все файлы js и scss из каждой подпапки 'components' importAll(require.context('./../../components', true, /\.js$/)); importAll(require.context('./../../components', true, /\.scss$/)); // Импортируем все файлы js и scss из каждой подпапки 'blocks' importAll(require.context('./../../blocks', true, /\.js$/)); importAll(require.context('./../../blocks', true, /\.scss$/));Такой базовый сетап уже позволяет закрывать 90% задач в рамках разработки кастомной темы для WordPress.Использование REST APIДля асинхронных запросов мы чаще всего используем WP REST API. Это не единственный способ создать кастомный эндпоинт, но, как показывает практика, самый удобный. Все, что нам нужно сделать — это воспользоваться функцией register_rest_route в хуке rest_api_init и написать функцию-колбэк, которая будет обрабатывать этот маршрут.Пример создания эндпоинта для блока со списком новостей и фильтрацией по таксономиям мы рассматривали в предыдущей статье:add_action( 'rest_api_init', 'wwzrds_news_list_route' ); function wwzrds_news_list_route() { register_rest_route( 'wwzrds/v1', '/news-list', [ 'methods' => 'GET', 'callback' => 'wwzrds_news_list_route_callback', 'permission_callback' => '__return_true', ] ); } function wwzrds_news_list_route_callback( WP_REST_Request $request ) { $json_params = ! empty( $request['json'] ) ? $request['json'] : null; if ( ! $json_params ) { return new WP_REST_Response( [ 'error' => 'Params are empty', ], 400 ); } $query_args = json_decode( $json_params, true ); if ( ! $query_args ) { return new WP_REST_Response( [ 'error' => 'Params are empty', ], 400 ); } $term = ! empty( $request['term'] ) ? $request['term'] : null; $page = ! empty( $request['page'] ) ? $request['page'] : null; $tax = ! empty( $request['taxonomy'] ) ? $request['taxonomy'] : 'technology'; $query_args['tax_query'] = [ 'relation' => 'AND', ]; if ( $term && $term !== 'all' ) { $query_args['tax_query']['term'] = [ 'taxonomy' => $tax, 'terms' => [ $term ], 'field' => 'slug', ]; } if ( $page ) { $query_args['paged'] = $page; } if ( $query_args['post_type'] !== 'post' || $query_args['post_status'] !== 'publish' ) { return new WP_REST_Response( [ 'error' => 'Invalid data', ], 400 ); } $query = new WP_Query( $query_args ); $max_pages = $query->max_num_pages; $news = $query->posts; wp_reset_query(); $response = []; if ( ! empty( $news ) ) { $response['posts'] = []; foreach ( $news as $new ) { $response['posts'][] = wwzrds_get_template_string( 'template-parts/components/post-item', [ 'post_id' => $new->ID, 'attrs' => [ 'style' => 'opacity:1; transform: translateY(0)', ], 'taxonomy' => $tax, 'custom_class' => 'news-list__post', ] ); } $response['pagination'] = wwzrds_generate_pagination( $page, $max_pages, false ); } else { $response['empty'] = wwzrds_get_template_string( 'template-parts/blocks/block-news-list/empty-results' ); } return new WP_REST_Response( $response, 200 ); }wwzrds_generate_pagination — функция, которая формирует разметку пагинации на основе входных данных;wwzrds_get_template_string — функция-хелпер, которая возвращает шаблон как строку.Пример создания эндпоинта:add_action( 'rest_api_init', 'db_user_bank_endpoints' ); /** * Get user's bank * * @param WP_REST_Request $request Full details about the request. * @return array $args. **/ function db_user_bank_endpoints( $request ) { register_rest_route('wp/v2', 'users/bank', array( 'methods' => 'GET', 'callback' => 'db_rest_user_bank_handler', 'permission_callback' => function() { return is_user_logged_in(); }, )); } function db_rest_user_bank_handler( $request = null ) { $user = wp_get_current_user(); $error = new WP_Error(); if ( empty( $user ) ) { $error->add( 400, __( 'Invalid user.', 'wp-rest-run' ), ['status' => 400] ); return $error; } return new WP_REST_Response( db_calculate_user_store( $user->ID ), 123 ); }Пример интеграции React-приложенияНачиная с версии 5 WordPress включает в себя React и позволяет использовать его через пакет wp.element.Допустим, что мы хотим добавить отдельный React-виджет на сайт.Необходимо в папке темы создать отдельную папку для файлов виджета и внутри этой папки инициализировать package.json командой:npm initПосле этого устанавливаем WP Scripts командой:npm install @wordpress/scripts --save-devОткрываем package.json и добавляем следующую секцию:"scripts": { "test": "echo \"Error: no test specified\" && exit 1", "build": "wp-scripts build", "check-engines": "wp-scripts check-engines", "check-licenses": "wp-scripts check-licenses", "format": "wp-scripts format", "lint:css": "wp-scripts lint-style", "lint:js": "wp-scripts lint-js", "lint:md:docs": "wp-scripts lint-md-docs", "lint:md:js": "wp-scripts lint-md-js", "lint:pkg-json": "wp-scripts lint-pkg-json", "packages-update": "wp-scripts packages-update", "plugin-zip": "wp-scripts plugin-zip", "start": "wp-scripts start", "test:e2e": "wp-scripts test-e2e", "test:unit": "wp-scripts test-unit-js" },В этой же папке добавляем webpack.config.js:const defaults = require('@wordpress/scripts/config/webpack.config'); module.exports = { ...defaults, externals: { react: 'React', 'react-dom': 'ReactDOM', }, };В index.js будем использовать метод render не из ReactDom, а из wp.element, в остальном это будет обычное React-приложение:const { render } = wp.element; // we are using wp.element here! import App from './calculator/components/App'; // check if element exists before rendering const appElement = document.getElementById('calculator-app'); if (appElement) { render(<App />, appElement); }В App.js поместим основное react приложение. Если нам необходимо импортировать что-то из React или ReactDOM, то импортируем это из wp.element, например:const {useState, useEffect} wp.element;При подключении react приложения в WordPress указываем зависимость от wp-element:add_action('wp_enqueue_scripts', 'calc_react_app'); function calc_react_app() { $calc_script_url = THEME_URL . '/calculator/build/index.js'; $calc_script_path = THEME_DIR . '/calculator/build/index.js'; wp_enqueue_script( 'my_react_app', $calc_script_url, // This refer to the built React app ['wp-element'], //This dependency indicates that you need React at Frontend filemtime($calc_script_path) // This could be changed to the theme version for production ); }Добавим отдельный React-виджет на сайтВ папке темы создаем отдельную папку для файлов виджета и внутри этой папки инициализируем package.json командой:npm initПосле этого устанавливаем WP Scripts командой:npm install @wordpress/scripts --save-devОткрываем package.json и добавляем следующую секцию:"scripts": { "test": "echo \"Error: no test specified\" && exit 1", "build": "wp-scripts build", "check-engines": "wp-scripts check-engines", "check-licenses": "wp-scripts check-licenses", "format": "wp-scripts format", "lint:css": "wp-scripts lint-style", "lint:js": "wp-scripts lint-js", "lint:md:docs": "wp-scripts lint-md-docs", "lint:md:js": "wp-scripts lint-md-js", "lint:pkg-json": "wp-scripts lint-pkg-json", "packages-update": "wp-scripts packages-update", "plugin-zip": "wp-scripts plugin-zip", "start": "wp-scripts start", "test:e2e": "wp-scripts test-e2e", "test:unit": "wp-scripts test-unit-js" },В той же папке добавляем webpack.config.js:`const defaults = require('@wordpress/scripts/config/webpack.config');` `module.exports = { ...defaults, externals: { react: 'React', 'react-dom': 'ReactDOM', }, };`В index.js будем использовать метод render не из ReactDom, а из wp.element, в остальном это будет обычное React-приложение:`const { render } = wp.element; //we are using wp.element here! import App from './calculator/components/App'` `if (document.getElementById('calculator-app')) { //check if element exists before rendering render(<App />, document.getElementById('calculator-app')); }`В App.js поместим основное react приложение. Если нам необходимо импортировать что-то из React или ReactDOM то импортируем это из wp.element, например:При подключении react приложения в WordPress указываем зависимость от wp-element:Такой скелет приложения позволяет очень быстро создавать современные, технологичные и масштабируемые проекты, которые легко поддерживать. Для работы над темой WordPress вы можете использовать публичный репозиторий по ссылке: github.com/WeWizards/WP-start-templateКонечно, многие скажут, что надо писать фронт на Vue или React целиком, получать данные по REST API и т. д. — и мы частично согласны, но каждая задача требует индивидуального подхода, в том числе и в выборе наиболее подходящего инструмента для ее реализации. Подписывайтесь на соцсети We Wizards и читайте о том, как мы делаем Headless WordPress + Vue/React в следующих статьях.InstagramTelegramVKДзен