Шаблонизатор проектов
Простейший способ получить заготовку проекта это выполнить команду
npx @salutejs/create-app
После запуска появится визард, в котором можно указать имя проекта и прочие опции. После завершения работы скрипта в текущей рабоче директории терминала будет создана еще одна директория с именем указанным как имя проекта.
При запуске команды без аргументов будет выбран пакет @salutejs/canvas-example
как пакет-шаблон. Но можно указать в качестве шаблона любой другой пакет. При отсутствии дополнительной конфигурации пакет-шаблон будет просто скопирован в целевую папку.
npx @salutejs/create-app --templatePackage any-npm-package
Если ваш пакет требует авторизации в npm, можно задать токен доступа в env-переменную NPM_REGISTRY_TOKEN
.
@salutejs/create-app
создавался для шаблонов @salutejs
, но не привязан ни к какому конкретному шаблону. Поэтому достаточно просто создавать свои шаблоны. В простейшем виде шаблоном является любой npm пакет. В этом можно убедиться вызвав npx @salutejs/create-app --templatePackage lodash
. В этом случае @salutejs/create-app
просто скопирует все файлы пакета в целевую папку.
Но такие шаблоны не имеют особого смысла, они лишь упрощают копирование пакета. Поэтому существует специальный API, позволяющий более тонко настраивать шаблоны. Для использования этого API необходимо создать папку ./.create-app/
в корне пакета-шаблона и файл index.mjs
в этой папке. В простейшем виде это может быть пустой файл. Помимо этого все файлы прогоняются через mustache.
Все копируемые из шаблона файлы прогоняются через mustache. При этом mustache директивы переопределены таким образом, чтобы оставаться просто комментариями для исходного кода. Это позволяет оставить исходные файлы валидным JS кодом (префикс директивы {{
заменен на /* <|
, а постфикс }}
на |> */
), чтобы шаблон можно было запускать и тестировать без шага предварительной шаблонизации. Также можно задать и свои правила для любых файлов при помощи поля rules конфига.
Например, строка, обернутая в приведенные ниже директивы, будет присутствовать в конечном файле если включен featureToggle
с id
равным imageOptimization
или отсутствовать в противном случае.
/* <|#imageOptimization|> */
{ href: `${basePath}/image-optimization`, content: 'Оптимизация изображений' },
/* <|/imageOptimization|> */
На оригинальном синтаксисе mustache это выглядело бы так
{{#imageOptimization}}
{ href: `${basePath}/image-optimization`, content: 'Оптимизация изображений' },
{{/imageOptimization}}
Более тонкая настройка (например описание шаблона, правила включения и исключения файлов и так далее) осуществляется с помощью экспортирования переменных из файла конфига, их более подробное описание ниже, а развернутый пример можно посмотреть в этом репозитории.
Текстовое описание шаблона. Будет показано пользователю сразу после запуска @salutejs/create-app
.
export templateDescription = "Описание шаблона";
Если необходимо, чтобы пользователь мог выбрать из некого списка дополнительных возможностей (featureToggle
) шаблона, нужно экспортировать переменную featureToggles, которая представляет из себя массив объектов вида.
{
featureId: 'imageOptimization',
name: `Оптимизация изображений
Для более быстрой загрузки изображений next использует изображения
правильного формата и размера. В примере так же используется аватарница
sberdevices, что является целевой картиной для использования изображений`,
defaultValue: true,
hidden: false,
}
При этом сам по себе массив ничего не делает, лишь отвечает за показанный пользователю интерфейс и сбор ответов пользователя. На этапе выполнения визарда появится вопрос про изменение дефолтных опций, если пользователь ответит на него положительно, то тогда он сможет переключать состояния различных featureToggle
.
- featureId - произвоьлный уникальный идентификатор
featureToggle
- name - название
featureToggle
, для большей ясности можно добавить к нему дополнительное описание через символ переноса строки. Важно помнить про то, что ширина терминала пользователя может быть достаточно маленькой, желательно форматировать описание так, чтобы длина каждой отдельной сртоки была не больше 80 символов - defaultValue - должен ли
featureToggle
быть включенным или выключенным по умолчанию - hidden - если
true
, то скрывает от пользователяfeatureToggle
(нужен, если мы всегда хотим применять некую логику шаблониазции вне зависимости от выбора пользователя, но не применять ее заранее к исходным файлам)
Основная часть конфига. После того как пользователь выбрал необходимые ему featureToggle
, эта часть конфига позволяет @salutejs/create-app
на основе этих ответов принять решение, какие модификаторы (featureConfig
) необходимо применить к шаблону.
В простом случае достаточно соотвествия "один к одному", то есть одному featureToggle
соответсвует один featureConfig
. Но в реальных примерах это не всегда так, поэтому featureConfigMap
позволяет установить более сложные правила соответствия. Переменная featureConfigMap
представляет из себя объект, ключами которого являются уникальные идентификаторы featureConfig
, а значениями другой объект вида
{
'imageOptimizationConfigId' :{
config: (options) => getImageOptimizationConfig(options),
test: (selectedFeatureToggleIds) => selectedFeatureToggleIds.includes('imageOptimization'),
}
/* ... */
}
- imageOptimizationConfigId - в данном примере это произвольный уникальный идентификатор
featureConfig
- config - функция, принимающая набор некоторых опций и возвращающая готовый featureConfig (должна быть определена автором шаблона)
- options.templatePath - абсолютный путь до текущей папки пакета-шаблона, можно использовать как префикс для всех путей, которые хотелось бы не указвать как абсолютный (см. пример ниже).
- test - функция, которая по выбранным
featureToggle
определяет, нужно ли применить текущийfeatureConfig
к шаблону в состоянииon
или нет состоянииoff
. В простейшем соответствии "один к одному" можно проверить, что выбранныйfeatureToggle
есть в массивеselectedFeatureToggleIds
.
featureConfig
, который возвращается функцией config
, имеет вид:
{
off: {
sourceAddon: [
`!${templatePath}src/components/PageImageOptimization.tsx`,
],
jsonChanges: [
{
glob: `${templatePath}package.json`,
changes: {
remove: ['scripts.build:static'],
merge: {
scripts: {
build: 'next build && next export',
},
},
},
},
],
},
on: {
sourceAddon: [],
jsonChanges: [],
},
}
Основной упор в @salutejs/create-app
сделан на то, что пакет-шаблон может оставаться запускаемым пакетом и валидным с точки зрения JS. Поэтому в объекте конфига есть 2 способа повлиять на состав файлов проекта, который получится после шаблонизации: если конфиг включен (on
) и если конфиг выключен (off
).
- off - это та часть конфига, которая будет применена, если функция
test
вернулаfalse
.sourceAddon
- массивglob
-паттернов, который говорит какие файлы нужно добавить или исключить при применении этой части конфига. В примере выше из общего списка файлов убирается файлPageImageOptimization.tsx
.jsonChanges
- это поле необходимо только потому, что JSON как формат лишен комментариев и невозможно добавить внутрь JSON-файла директивы шаблонизации с помощью комментариев. Принимает массив объектов в которых- glob -
glob
-паттерн который выбирает файлы, к которым будут применены изменения из поляchanges
- changes - объект, описывающий, какие поля в JSON-файле надо удалить (поле
remove
), а какие изменить/добавить (полеmerge
). Полеremove
подразумевает массив строк, каждая из которых задает путь до поля. В примере удаляется полеbuild:static
из объектаscripts
вpackage.json
. Полеmerge
предоставляет объект для объединения с определенным в JSON объектом (lodash merge).
- glob -
- on - часть конфига, которая будет применена, если функция
test
вернулаtrue
(такие же поля как и дляoff
)
На практике шаблон часто представляет из себя максимальный пример со всеми файлами. И важнее убирать из него файлы, когда какая-то функциональность не нужна, чем наоборот.
В случае если нужно переопределить для каких-то файлов стиль комментария, служащим директивой шаблонизации (например для файлов без расширения), необходимо экспортировать переменную rules, представляющую из себя массив объектов вида
{
test: (file) => ['.env.production', '.env.development', '.gitignore']
.includes(file.basename),
tags: ['# <|', '|> #'],
},
- test - функция, принимающая в себя файл (
file
) в терминах vinyl-fs и возвращающаяtrue
, для файлов, к которым нужно применять указанные тэги как префикс/постфикс директивы шаблонизации. - tags - префикс (первый элемент) и постфикс (второй элемнет) директив шаблониазции