javascript

Структура Vue проекта

  • воскресенье, 10 мая 2026 г. в 00:00:11
https://habr.com/ru/articles/1033300/

Правильная файловая структура - скелет любого фронтенд-приложения. В Vue 3 нет строгих предписаний, как раскладывать файлы по папкам, кроме базового разделения components/views/. Но с ростом проекта хаотичное размещение кода быстро превращается в проблему. В этой статье разберём популярные подходы к организации Vue-проектов: от простейшего плоского до микрофронтендов.

Плоская структура: быстрота и минимализм

С этой схемы стартуют почти все небольшие проекты (пет-приложения). Суть в том, что все компоненты лежат в одной-двух папках, а утилиты и композаблы сгруппированы по назначению, без дополнительных уровней вложенности.

/src
|-- /components
|   |-- BaseButton.vue
|   |-- BaseModal.vue
|   |-- ProductCard.vue
|-- /composables
|   |-- useCart.js
|   |-- useProducts.js
|-- /utils
|   |-- formatters.js
|-- /layouts
|   |-- MainLayout.vue
|   |-- AuthLayout.vue
|-- /plugins
|   |-- analytics.js
|-- /views
|   |-- HomePage.vue
|   |-- ProductPage.vue
|-- /router
|   |-- index.js
|-- /store
|   |-- index.js
|-- /assets
|-- /tests
|-- App.vue
|-- main.js

Плюсы:

  • Минимум настройек

  • Быстрый старт

  • Низкий порог входа

Минусы:

  • Не масштабируется

  • Быстро возникает дублирование

  • Сложно искать файлы при росте >30 компонентов

Когда использовать: демо, обучающие проекты.

Атомарный дизайн (Atomic Design)

Представьте, что ваш интерфейс - это живой организм, собранный из мельчайших «клеток». В масштабных проектах на Vue такой взгляд помогает держать хаос под контролем. Методология Atomic Design предлагает разложить любой экран на пять осмысленных уровней - от неделимых частиц до полноценных рабочих страниц:

  • Атомы - фундаментальные кирпичики интерфейса, которые уже нельзя разбить на более мелкие части без потери смысла. В мире Vue это одиночные элементы: кнопка, поле ввода, иконка, заголовок. Они абстрактны, чисты и не зависят от контекста.

  • Молекулы - первые полезные конструкции, рождающиеся при объединении нескольких атомов. Так, текстовое поле с кнопкой и меткой превращается в строку поиска. Молекулы уже начинают решать конкретные микро-задачи, оставаясь при этом переиспользуемыми.

  • Организмы - самостоятельные, визуально законченные блоки интерфейса, собранные из молекул и/или атомов. Например, шапка сайта, в которой живут логотип (атом), строка поиска (молекула) и навигационное меню из кнопок-атомов. Такой компонент уже определяет характер целого участка макета.

  • Шаблоны - скелет страницы, где организмы расставляются по сетке, образуя абстрактную композицию. Здесь появляются отступы, колонки и логическая структура, но данные пока условные или отсутствуют - это как архитектурный макет здания без мебели и жильцов.

  • Страницы - конечный уровень, где шаблон наполняется реальным контентом, живыми данными из API, состояниями загрузки и ошибками. Именно здесь интерфейс становится тем, что видит пользователь, и именно на этом уровне видны все нюансы работы системы.

Такой подход превращает хаотичный набор компонентов в стройную экосистему, где каждому элементу находится своё место, а внесение изменений на любом уровне автоматически отражается по всей иерархии - но всегда предсказуемо и безболезненно.

/src
|-- /components
|   |-- /atoms
|   |   |-- UiBadge.vue
|   |   |-- UiAvatar.vue
|   |-- /molecules
|   |   |-- SearchBar.vue
|   |   |-- CartItem.vue
|   |-- /organisms
|   |   |-- ProductCard.vue
|   |-- /templates
|   |   |-- ProductListTemplate.vue
|-- /pages
|   |-- HomePage.vue
|-- /composables
|   |-- useProducts.js
|-- /utils
|   |-- priceHelpers.js
|-- /layouts
|   |-- DefaultLayout.vue
|-- /router
|   |-- index.js
|-- /store
|   |-- index.js
|-- App.vue
|-- main.js

Плюсы: атомы и молекулы отлично переиспользуются. Если дизайнеры меняют кнопку - достаточно поправить один компонент, и изменения распространятся по всему приложению.

Минусы: постоянная мысленная нагрузка «а это атом или молекула?» замедляет разработку. Часто возникает жёсткая связанность на уровне шаблонов, а бизнес-логика размазывается по слоям, что усложняет рефакторинг.

Модульная архитектура

Модульный подход группирует файлы не по технической роли, а по предметной области. Каждый функциональный блок (модуль) полностью инкапсулирует свою логику: компоненты, хранилище, API-клиенты, тесты и даже стили. Это похоже на разделение кода по микросервисам, но внутри одного репозитория.

/src
|-- /core                          // App Shell и общие ресурсы
|   |-- /components
|   |   |-- BaseButton.vue
|   |-- /models
|   |-- /store                     // Глобальное состояние (сессия, уведомления)
|   |-- /services                  // API-клиент, интерсепторы
|   |-- /views
|   |   |-- AdminLayout.vue
|   |-- /utils
|   |   |-- validators.js
|-- /modules
|   |-- /products                   // Модуль «Каталог товаров»
|   |   |-- /components
|   |   |   |-- ProductThumbnail.vue
|   |   |   |-- ProductCard.vue
|   |   |-- /models
|   |   |-- /store
|   |   |   |-- productStore.js
|   |   |-- /services
|   |   |   |-- productApi.js
|   |   |-- /views
|   |   |   |-- ProductDetailPage.vue
|   |   |-- /tests
|   |   |   |-- productTests.spec.js
|   |-- /cart                       // Модуль «Корзина»
|   |   |-- /components
|   |   |   |-- CartIcon.vue
|   |   |   |-- CartItem.vue
|   |   |-- /store
|   |   |   |-- cartStore.js
|   |   |-- /services
|   |   |-- /views
|   |   |   |-- CartPage.vue
|   |   |-- /tests
|   |-- /account                    // Модуль «Личный кабинет»
|   |   |-- /components
|   |   |   |-- ProfileCard.vue
|   |   |-- /store
|   |   |   |-- accountStore.js
|   |   |-- /services
|   |   |-- /views
|   |   |   |-- ProfilePage.vue
|-- /assets
|   |-- /images
|   |-- /styles
|-- /plugins
|   |-- translate.js
|-- App.vue
|-- main.js

Плюсы:

  • Слабая связанность. Модуль cart использует API корзины и внутреннее хранилище, ничего не зная о деталях модуля products. Вы можете переписать логику товаров, не затронув корзину.

  • Колокация ответственности. Разработчик, отвечающий за продуктовый каталог, видит все файлы в одной папке, что ускоряет навигацию и рефакторинг.

  • Масштабирование команд. Если над проектом работают несколько разработчиков, каждый может вести свой модуль практически без конфликтов слияния.

  • Миграция в микрофронтенды. Когда модуль разрастается до такого объёма, что его выгодно развернуть независимо, он уже изолирован по коду, остаётся лишь наладить его отдельную сборку и деплой.

Feature-Sliced Design (FSD)

FSD - более формализованный архитектурный подход, который разбивает приложение на слои по степени близости к бизнес-логике. Он пришёл из React-экосистемы, но применим и во Vue.

Слои FSD (снизу вверх):

  1. Shared - переиспользуемый код без бизнес-логики (UI-кит, утилиты, API-клиент)

  2. Entities - бизнес-сущности (пользователь, товар, заказ)

  3. Features - пользовательские сценарии (добавление в корзину, авторизация)

  4. Widgets - самостоятельные блоки (шапка, список товаров, профиль)

  5. Pages - страницы приложения

  6. App - глобальные настройки, провайдеры, стили

Правила архитектуры:

  • Код может импортироваться только снизу вверх

  • Запрещены импорты с того же или более высоких слоев

  • Shared можно использовать везде

  • Entities не знают о Features и выше

/src
|-- /app
|   |-- App.vue
|   |-- main.js
|   |-- providers/
|   |   |-- RouterProvider.vue
|   |   |-- ThemeProvider.vue
|   |-- styles/
|   |   |-- index.css
|-- /pages
|   |-- /home
|   |   |-- HomePage.vue
|   |   |-- components/
|   |   |-- model/
|   |   |-- lib/
|   |-- /catalog
|   |   |-- CatalogPage.vue
|   |   |-- components/
|   |   |-- model/
|   |   |-- lib/
|-- /widgets
|   |-- /header
|   |   |-- Header.vue
|   |   |-- components/
|   |   |-- model/
|   |-- /footer
|   |   |-- Footer.vue
|   |   |-- components/
|   |-- /product-list
|   |   |-- ProductList.vue
|   |   |-- components/
|   |   |-- model/
|-- /features
|   |-- /auth
|   |   |-- /login
|   |   |   |-- LoginForm.vue
|   |   |   |-- model/
|   |   |   |-- lib/
|   |   |-- /registration
|   |   |   |-- RegistrationForm.vue
|   |   |   |-- model/
|   |   |-- /logout
|   |       |-- LogoutButton.vue
|   |-- /product
|   |   |-- /add-to-cart
|   |   |   |-- AddToCartButton.vue
|   |   |   |-- model/
|-- /entities
|   |-- /product
|   |   |-- ui/
|   |   |   |-- ProductCard.vue
|   |   |   |-- ProductImage.vue
|   |   |   |-- ProductPrice.vue
|   |   |-- model/
|   |   |   |-- types.ts
|   |   |   |-- selectors.js
|   |   |   |-- api.js
|   |   |-- lib/
|   |       |-- helpers.js
|   |-- /user
|   |   |-- ui/
|   |   |   |-- UserAvatar.vue
|   |   |   |-- UserInfo.vue
|   |   |-- model/
|   |   |   |-- types.ts
|   |   |   |-- selectors.js
|   |   |   |-- api.js
|-- /shared
|   |-- /ui
|   |   |-- /button
|   |   |   |-- Button.vue
|   |   |   |-- Button.spec.js
|   |   |-- /input
|   |   |   |-- Input.vue
|   |-- /lib
|   |   |-- api/
|   |   |   |-- apiClient.js
|   |   |   |-- endpoints.js
|   |   |-- utils/
|   |   |   |-- formatters.js
|   |   |   |-- validators.js
|   |   |   |-- constants.js
|   |   |-- hooks/
|   |       |-- useDebounce.js
|   |       |-- useLocalStorage.js
|   |-- /config
|   |   |-- theme.js
|   |-- /assets
|       |-- /images
|       |-- /fonts
|       |-- /icons
|-- /tests
|   |-- mocks/
|-- router.js
|-- store.js
|-- main.js

Слои строго регламентируют импорты: элементы с нижнего слоя не могут обращаться к элементам верхнего. Это защищает от циклических зависимостей, но требует уверенного знания методологии всей командой. Для средних проектов и небольших команд такой уровень формализации часто избыточен - та же инкапсуляция достигается проще через модульную структуру.

Микрофронтенды

Микрофронтенды переносят философию микросервисов на клиентскую часть. Каждый раздел магазина - каталог, корзина, личный кабинет - разрабатывается и деплоится независимо, возможно, разными командами и на разных стеках.

Основные элементы:

  • Оболочка (Shell): координация маршрутизации, базовый каркас.

  • Независимые мини-приложения: Product Listing, Checkout, User Account.

Это серьёзный инструмент, где над фичами работают десятки разработчиков. Для подавляющего большинства приложений сложность деплоя, версионирования и отладки микрофронтендов не окупается.

Плюсы:

  • Независимое развертывание

  • Разные технологии в разных модулях

  • Масштабирование команд

  • Изоляция ошибок

Минусы:

  • Сложность настройки

  • Дублирование зависимостей

  • Сложность отладки

Что же выбрать для Vue 3-проекта?

Представь, что проект Vue - это большой шкаф с вещами. В начале вещей мало, их можно сложить как попало. Но когда проект растёт, нужна система хранения, чтобы быстро находить то, что относится к «корзине», «каталогу», «авторизации». Модульная архитектура - это способ разложить код по папкам-модулям так, чтобы в каждой лежало всё, что касается отдельной бизнес-задачи. Вот почему.

  1. Она органично вписывается в экосистему Vue. Composition API со своими композаблами идеально сочетается с изолированными модулями. В одном модуле вы держите и презентационные компоненты, и логический слой (useProductuseCartuseAuth), и стор (Pinia-хранилища, если необходим модульный стейт). Никаких дополнительных абстракций не требуется.

  2. Инкрементальное внедрение. Вы можете начать с плоской структуры и безболезненно выделить модули по мере кристаллизации бизнес-требований. Уже работающий код просто переезжает в /modules/…, без переписывания всей архитектуры.

  3. Совместимость с другими подходами. Если команда дизайна настаивает на атомарном дизайне, реализуйте его внутри core/components/ и, скажем, внутри modules/products/components/. Если позже потребуется жёсткий FSD для части приложения - вы вольны организовать так отдельный модуль.

  4. Живой пример. Представьте, что в модуле cart понадобилось добавить промокоды. Вы создаёте подпапку modules/cart/composables/usePromocode.js, компонент PromocodeInput.vue и тест. Вся функциональность замкнута в одной ветке, не затрагивая каталог или профиль. Через месяц, когда корзина разовьётся до отдельного микрофронтенда, вы просто настраиваете для папки modules/cart отдельный Webpack/Vite-entry. Код менять не придётся.

P.S. И напоследок, чтобы ни у кого не возникло желания накинуться с критикой: это не истина в последней инстанции. Это исключительно мои мысли, основанные на моём опыте, наблюдениях и чувствах. Я не претендую на то, чтобы переубедить всех. Моё мнение - всего лишь одно из многих, но оно имеет полное право на существование, как и ваше. Если у вас иначе - замечательно, делитесь в комментариях конструктивно. Но без негатива, пожалуйста)