@nitra/cursor 12.8.5 → 12.8.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +12 -0
- package/bin/n-cursor.js +5 -5
- package/package.json +1 -1
- package/rules/abie/js/http_route_base.mdc +25 -0
- package/rules/abie/js/ua_http_route.mdc +1 -1
- package/rules/abie/main.mdc +12 -0
- package/rules/adr/js/hooks.mdc +32 -0
- package/rules/adr/js/madr_format.mdc +96 -0
- package/rules/adr/js/settings_policy.mdc +34 -0
- package/rules/adr/main.mdc +13 -95
- package/rules/bun/js/bunfig.mdc +12 -0
- package/rules/bun/js/layout.mdc +60 -0
- package/rules/bun/js/lint.mdc +9 -0
- package/rules/bun/js/package_json.mdc +19 -0
- package/rules/bun/main.mdc +9 -61
- package/rules/capacitor/js/ios_spm.mdc +69 -0
- package/rules/capacitor/js/version.mdc +29 -0
- package/rules/capacitor/main.mdc +8 -22
- package/rules/changelog/js/agent-workflow.mdc +15 -0
- package/rules/changelog/js/changelog-format.mdc +33 -0
- package/rules/changelog/js/comparison-models.mdc +40 -0
- package/rules/changelog/main.mdc +4 -98
- package/rules/ci4/js/marksman_config.mdc +31 -0
- package/rules/ci4/js/vscode_extensions.mdc +33 -0
- package/rules/ci4/main.mdc +14 -14
- package/rules/docker/js/compile.mdc +44 -0
- package/rules/docker/js/hadolint.mdc +50 -0
- package/rules/docker/js/mirror.mdc +13 -0
- package/rules/docker/js/multistage.mdc +13 -0
- package/rules/docker/js/native-addon.mdc +43 -0
- package/rules/docker/js/nginx-tag.mdc +7 -0
- package/rules/docker/js/nginx-user.mdc +37 -0
- package/rules/docker/js/non-root.mdc +39 -0
- package/rules/docker/main.mdc +15 -196
- package/rules/ga/js/lint_toolchain.mdc +15 -0
- package/rules/ga/js/required_workflows.mdc +35 -0
- package/rules/ga/js/vscode.mdc +17 -0
- package/rules/ga/js/workflow_common.mdc +108 -0
- package/rules/ga/js/workflows.mdc +32 -0
- package/rules/ga/js/zizmor.mdc +7 -0
- package/rules/ga/main.mdc +17 -125
- package/rules/graphql/js/tooling.mdc +13 -0
- package/rules/graphql/js/vscode_extensions.mdc +13 -0
- package/rules/graphql/main.mdc +3 -22
- package/rules/hasura/js/internal_urls.mdc +27 -0
- package/rules/hasura/js/migrations.mdc +13 -0
- package/rules/hasura/js/svc_hl.mdc +17 -0
- package/rules/hasura/main.mdc +8 -30
- package/rules/image-avif/js/avif_generation.mdc +26 -0
- package/rules/image-avif/js/package_json_optout.mdc +21 -0
- package/rules/image-avif/main.mdc +7 -34
- package/rules/image-compress/js/package_json.mdc +7 -0
- package/rules/image-compress/js/package_setup.mdc +13 -0
- package/rules/image-compress/main.mdc +4 -12
- package/rules/js/docs/index.md +3 -3
- package/rules/js/js/dep-policy.mdc +17 -0
- package/rules/js/js/eslint-config.mdc +28 -0
- package/rules/js/js/extensions.mdc +8 -0
- package/rules/js/js/file-extensions.mdc +12 -0
- package/rules/js/js/for-in.mdc +26 -0
- package/rules/js/js/jscpd.mdc +42 -0
- package/rules/js/js/knip.mdc +15 -0
- package/rules/js/js/lint-js-workflow.mdc +58 -0
- package/rules/js/js/oxlintrc.mdc +20 -0
- package/rules/js/js/package-json.mdc +31 -0
- package/rules/js/js/tests.mdc +9 -0
- package/rules/js/js/utils-lib-structure.mdc +15 -0
- package/rules/js/main.mdc +21 -214
- package/rules/js-bun-db/js/bun-sql-migration.mdc +15 -0
- package/rules/js-bun-db/js/connection.mdc +42 -0
- package/rules/js-bun-db/js/pg-format-identifiers.mdc +102 -0
- package/rules/js-bun-db/js/pg-format-shim.mdc +99 -0
- package/rules/js-bun-db/js/pg-leftover.mdc +27 -0
- package/rules/js-bun-db/js/pg-listen-notify.mdc +51 -0
- package/rules/js-bun-db/js/query-safety.mdc +117 -0
- package/rules/js-bun-db/js/sql-array.mdc +88 -0
- package/rules/js-bun-db/js/unsafe.mdc +65 -0
- package/rules/js-bun-db/main.mdc +15 -605
- package/rules/js-bun-redis/js/imports.mdc +47 -0
- package/rules/js-bun-redis/js/package_json.mdc +44 -0
- package/rules/js-bun-redis/main.mdc +3 -11
- package/rules/js-mssql/js/mssql-in-list.mdc +38 -0
- package/rules/js-mssql/js/mssql-pool.mdc +56 -0
- package/rules/js-mssql/js/mssql-query-template.mdc +33 -0
- package/rules/js-mssql/js/mssql-tvp.mdc +75 -0
- package/rules/js-mssql/js/mssql-version.mdc +7 -0
- package/rules/js-mssql/main.mdc +10 -198
- package/rules/js-run/js/check-env.mdc +35 -0
- package/rules/js-run/js/conn-aliases.mdc +109 -0
- package/rules/js-run/js/jsconfig.mdc +20 -0
- package/rules/js-run/js/otel-configmap.mdc +6 -0
- package/rules/js-run/js/pino.mdc +6 -0
- package/rules/js-run/js/project-structure.mdc +11 -0
- package/rules/js-run/js/runtime.mdc +14 -0
- package/rules/js-run/js/scope.mdc +11 -0
- package/rules/js-run/js/settimeout.mdc +11 -0
- package/rules/js-run/js/temporal.mdc +5 -0
- package/rules/js-run/main.mdc +16 -218
- package/rules/k8s/js/configmap.mdc +41 -0
- package/rules/k8s/js/deployment_resources.mdc +49 -0
- package/rules/k8s/js/hasura_httproute.mdc +91 -0
- package/rules/k8s/js/hpa_apiversion.mdc +27 -0
- package/rules/k8s/js/ingress_gateway.mdc +16 -0
- package/rules/k8s/js/kustomize_structure.mdc +144 -0
- package/rules/k8s/js/lint_k8s.mdc +72 -0
- package/rules/k8s/js/multidoc_yaml.mdc +5 -0
- package/rules/k8s/js/network_policy.mdc +136 -0
- package/rules/k8s/js/schema_modeline.mdc +57 -0
- package/rules/k8s/js/service.mdc +44 -0
- package/rules/k8s/js/topology_hpa_pdb.mdc +181 -0
- package/rules/k8s/main.mdc +30 -843
- package/rules/nginx-default-tpl/js/dockerfile.mdc +36 -0
- package/rules/nginx-default-tpl/js/http-route.mdc +41 -0
- package/rules/nginx-default-tpl/js/ini-keys.mdc +21 -0
- package/rules/nginx-default-tpl/js/template-structure.mdc +86 -0
- package/rules/nginx-default-tpl/js/vscode.mdc +37 -0
- package/rules/nginx-default-tpl/main.mdc +6 -112
- package/rules/npm-module/js/docs/index.md +5 -5
- package/rules/npm-module/js/docs/rule_meta.md +6 -6
- package/rules/npm-module/js/docs/skill_meta.md +8 -8
- package/rules/npm-module/js/header_doc_pointer.mdc +18 -0
- package/rules/npm-module/js/package_structure.mdc +62 -0
- package/rules/npm-module/js/rule_meta.mdc +11 -0
- package/rules/npm-module/js/skill_meta.mdc +11 -0
- package/rules/npm-module/main.mdc +10 -55
- package/rules/php/js/lint_php_yml.mdc +12 -0
- package/rules/php/js/tooling.mdc +66 -0
- package/rules/php/main.mdc +7 -66
- package/rules/python/js/lint_python_yml.mdc +23 -0
- package/rules/python/js/pyproject_toml.mdc +32 -0
- package/rules/python/js/tooling.mdc +23 -0
- package/rules/python/main.mdc +9 -33
- package/rules/rego/js/rego-lint.mdc +31 -0
- package/rules/rego/js/vscode_extensions.mdc +11 -0
- package/rules/rego/js/vscode_settings.mdc +13 -0
- package/rules/rego/main.mdc +8 -24
- package/rules/rust/js/coverage.mdc +28 -0
- package/rules/rust/js/lint.mdc +22 -0
- package/rules/rust/js/tauri_composition.mdc +8 -0
- package/rules/rust/js/vscode_extensions.mdc +12 -0
- package/rules/rust/main.mdc +8 -38
- package/rules/security/js/rego_policies.mdc +15 -0
- package/rules/security/js/sample_secret.mdc +19 -0
- package/rules/security/js/trufflehog.mdc +21 -0
- package/rules/security/main.mdc +7 -35
- package/rules/style/js/admin-table.mdc +88 -0
- package/rules/style/js/colors.mdc +21 -0
- package/rules/style/js/gap.mdc +22 -0
- package/rules/style/js/quasar-fixes.mdc +32 -0
- package/rules/style/js/quasar.mdc +7 -0
- package/rules/style/js/tooling.mdc +85 -0
- package/rules/style/main.mdc +13 -253
- package/rules/tauri/js/cargo_mutants_config.mdc +39 -0
- package/rules/tauri/js/tool_surface.mdc +21 -0
- package/rules/tauri/js/tooling.mdc +25 -0
- package/rules/tauri/main.mdc +8 -78
- package/rules/test/js/cargo_mutants_config.mdc +18 -0
- package/rules/test/js/docs/index.md +7 -7
- package/rules/test/js/location.mdc +52 -0
- package/rules/test/js/no-console-store-restore.mdc +11 -0
- package/rules/test/js/no-process-chdir.mdc +15 -0
- package/rules/test/js/no-relative-fs-path.mdc +22 -0
- package/rules/test/js/sandbox-aware-test.mdc +28 -0
- package/rules/test/js/stryker_config.mdc +26 -0
- package/rules/test/js/vitest-config-pool-forks.mdc +33 -0
- package/rules/test/main.mdc +18 -184
- package/rules/text/js/ci-lint-text.mdc +15 -0
- package/rules/text/js/cspell.mdc +81 -0
- package/rules/text/js/dotenv-linter.mdc +16 -0
- package/rules/text/js/forbidden-prettier.mdc +13 -0
- package/rules/text/js/markdownlint.mdc +25 -0
- package/rules/text/js/oxfmt.mdc +35 -0
- package/rules/text/js/package-json.mdc +26 -0
- package/rules/text/js/shellcheck.mdc +18 -0
- package/rules/text/js/v8r.mdc +23 -0
- package/rules/text/js/vscode.mdc +86 -0
- package/rules/text/main.mdc +20 -237
- package/rules/vue/js/composition-api.mdc +82 -0
- package/rules/vue/js/nheader-layout.mdc +171 -0
- package/rules/vue/js/node-imports.mdc +25 -0
- package/rules/vue/js/quasar-ui.mdc +32 -0
- package/rules/vue/js/structure.mdc +101 -0
- package/rules/vue/js/testing.mdc +32 -0
- package/rules/vue/js/tfm-translations.mdc +26 -0
- package/rules/vue/js/vite-config.mdc +126 -0
- package/rules/vue/js/vite-env.mdc +55 -0
- package/rules/vue/js/vue-imports.mdc +25 -0
- package/rules/vue/main.mdc +16 -640
- package/scripts/auto-rules.mjs +6 -6
- package/scripts/auto-skills.mjs +3 -3
- package/scripts/docs/auto-rules.md +17 -31
- package/scripts/docs/auto-skills.md +18 -163
- package/scripts/docs/index.md +16 -16
- package/scripts/lib/docs/index.md +36 -36
- package/scripts/lib/docs/mirror-parity.md +7 -7
- package/scripts/lib/docs/rule-meta.md +12 -12
- package/scripts/lib/docs/skill-meta.md +9 -9
- package/scripts/lib/docs/worktree-notice.md +10 -8
- package/scripts/lib/rule-meta.mjs +6 -6
- package/scripts/lib/skill-meta.mjs +6 -6
- package/scripts/lib/worktree-notice.mjs +2 -2
- package/scripts/utils/docs/index.md +14 -14
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
## Структура папок
|
|
2
|
+
|
|
3
|
+
```javascript
|
|
4
|
+
const folderStructure = `
|
|
5
|
+
src/
|
|
6
|
+
components/
|
|
7
|
+
composables/
|
|
8
|
+
views/
|
|
9
|
+
router/
|
|
10
|
+
store/
|
|
11
|
+
assets/
|
|
12
|
+
public/
|
|
13
|
+
App.vue
|
|
14
|
+
main.mjs
|
|
15
|
+
`
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
### Найменування файлів
|
|
19
|
+
|
|
20
|
+
- **SFC:** імена файлів компонентів у **PascalCase** починаючи з букви N(`NMyWidget.vue`).
|
|
21
|
+
- **Інші JS-модулі:** узгоджено **kebab-case** (`date-utils.mjs`).
|
|
22
|
+
|
|
23
|
+
### Модулі та архітектура
|
|
24
|
+
|
|
25
|
+
- **ES modules** (`import`/`export`), один модуль — одна відповідальність, уникай **circular dependencies**.
|
|
26
|
+
- **Composition** замість успадкування; логіку для повторного використання винось у **composables** (замість зайвих HOC, де це доречно).
|
|
27
|
+
|
|
28
|
+
### Code splitting
|
|
29
|
+
|
|
30
|
+
- **Route-based:** lazy-імпорти в маршрутах (`() => import('...')`) і async components.
|
|
31
|
+
|
|
32
|
+
## Додаткові вказівки
|
|
33
|
+
|
|
34
|
+
```javascript
|
|
35
|
+
const additionalInstructions = `
|
|
36
|
+
1. Використовуй JavaScript
|
|
37
|
+
2. Коректно оголошуй props, emits, defineModel
|
|
38
|
+
3. За потреби використовуй компонент Teleport у Vue 3
|
|
39
|
+
4. Застосовуй Suspense для async components
|
|
40
|
+
5. Реалізуй належний error handling
|
|
41
|
+
6. Дотримуйся Vue 3 style guide і naming conventions
|
|
42
|
+
7. Використовуй Vite для швидкої розробки та збірки
|
|
43
|
+
`
|
|
44
|
+
```
|
|
45
|
+
|
|
46
|
+
### Детальні вказівки
|
|
47
|
+
|
|
48
|
+
1. **Vue router**: Завжди використовуй Vue router для маршрутизації в проекті версії 5 та його file-based routing.
|
|
49
|
+
2. **Компонент Teleport**: Використовуй Teleport у Vue 3, коли потрібно рендерити поза поточною ієрархією DOM.
|
|
50
|
+
3. **Suspense для async components**: Застосовуй Suspense для асинхронних компонентів і кращого UX.
|
|
51
|
+
4. **Error handling**: Реалізуй глобальний error handling механізмами Vue 3, щоб коректно перехоплювати та обробляти помилки.
|
|
52
|
+
5. **Style guide і naming**: Дотримуйся офіційного Vue 3 style guide і naming conventions для узгодженості кодової бази.
|
|
53
|
+
6. **Vue macros**: Використовуй Vue macros для більш ефективного розробки компонентів.
|
|
54
|
+
7. **Vue Reactivity Transform**: Використовуй Vue Reactivity Transform для змінних в компонентах.
|
|
55
|
+
|
|
56
|
+
## Коментарі в `<template>`
|
|
57
|
+
|
|
58
|
+
Додавай коментарі в `<template>` відповідно до логічного призначення блоку. Коментарі допомагають швидко орієнтуватися в розмітці.
|
|
59
|
+
|
|
60
|
+
```vue
|
|
61
|
+
<template>
|
|
62
|
+
<q-page>
|
|
63
|
+
<!-- ФІЛЬТРИ В ШАПЦІ -->
|
|
64
|
+
<teleport v-if="mounted" to="#header-filters">...</teleport>
|
|
65
|
+
|
|
66
|
+
<!-- ТАБЛИЦЯ -->
|
|
67
|
+
<q-table ... />
|
|
68
|
+
|
|
69
|
+
<!-- ДІАЛОГ РЕДАГУВАННЯ -->
|
|
70
|
+
<n-dialog v-model="editDialog">...</n-dialog>
|
|
71
|
+
</q-page>
|
|
72
|
+
</template>
|
|
73
|
+
```
|
|
74
|
+
|
|
75
|
+
## Pinia store для стану сторінки
|
|
76
|
+
|
|
77
|
+
Зберігай у Pinia store:
|
|
78
|
+
|
|
79
|
+
- вибрані значення фільтрів
|
|
80
|
+
- вибрані для відображення колонки (`NTableColumns`)
|
|
81
|
+
- кількість записів на сторінці (pagination)
|
|
82
|
+
|
|
83
|
+
Називай store за назвою сторінки або компонента — `customerPageStore`, `routePageStore` тощо. На сторінці звертайся до нього через змінну `pageStore`.
|
|
84
|
+
|
|
85
|
+
```javascript
|
|
86
|
+
// store/customerPage.mjs
|
|
87
|
+
export const useCustomerPageStore = defineStore('customerPage', {
|
|
88
|
+
state: () => ({
|
|
89
|
+
filterName: '',
|
|
90
|
+
filterStatus: [],
|
|
91
|
+
columns: [],
|
|
92
|
+
rowsPerPage: 20
|
|
93
|
+
})
|
|
94
|
+
})
|
|
95
|
+
```
|
|
96
|
+
|
|
97
|
+
```vue
|
|
98
|
+
<script setup>
|
|
99
|
+
const pageStore = useCustomerPageStore()
|
|
100
|
+
</script>
|
|
101
|
+
```
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
## Тестування Vue-компонентів
|
|
2
|
+
|
|
3
|
+
### Unit + Component / DOM
|
|
4
|
+
|
|
5
|
+
**Vitest** (`vitest`) + **Vue Test Utils** з **happy-dom** як DOM-середовищем. Це канон, узгоджений з `test.mdc` (Stryker з vitest-runner + `perTest`-аналіз покриття). `vitest.config.mjs` повторно використовує `vite.config.js` через `mergeConfig` і перемикає `environment` на `'happy-dom'`:
|
|
6
|
+
|
|
7
|
+
```js title="vitest.config.mjs"
|
|
8
|
+
import { defineConfig, mergeConfig } from 'vitest/config'
|
|
9
|
+
import viteConfig from './vite.config.js'
|
|
10
|
+
|
|
11
|
+
export default mergeConfig(viteConfig, defineConfig({
|
|
12
|
+
test: {
|
|
13
|
+
include: ['**/*.test.{js,mjs}', 'tests/**/*.test.{js,mjs}'],
|
|
14
|
+
environment: 'happy-dom',
|
|
15
|
+
coverage: { provider: 'v8', reporter: ['lcov', 'text-summary'] }
|
|
16
|
+
}
|
|
17
|
+
}))
|
|
18
|
+
```
|
|
19
|
+
|
|
20
|
+
`jsdom` не використовуй — happy-dom швидший і достатній для типових Vue-компонентних тестів.
|
|
21
|
+
|
|
22
|
+
У `package.json#scripts` тримай `"test": "vitest run"`; vitest devDeps (`vitest`, `@vitest/coverage-v8`, `@stryker-mutator/vitest-runner`) — у кореневому `devDependencies` (npm-module rule забороняє devDeps у published workspace-у).
|
|
23
|
+
|
|
24
|
+
### E2E
|
|
25
|
+
|
|
26
|
+
**Playwright** — змістовні сценарії користувацьких потоків.
|
|
27
|
+
|
|
28
|
+
Проекту повинен бути покритий тестами E2E за допомогою Playwright.
|
|
29
|
+
|
|
30
|
+
### CI/CD
|
|
31
|
+
|
|
32
|
+
У pipeline: **install**, **lint**, **test**, **vite build**; артефакти з продакшен-режиму.
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
## @nitra/tfm — переклади
|
|
2
|
+
|
|
3
|
+
Використовуй `@nitra/tfm` для всіх текстів. Переклади для всіх мов проекту оголошуй наприкінці `<script setup>` у функції `getTr()`. Змінна `lang` із `@nitra/tfm` — для визначення або зміни поточної мови застосунку.
|
|
4
|
+
|
|
5
|
+
```vue
|
|
6
|
+
<template>
|
|
7
|
+
{{ lang }}
|
|
8
|
+
{{ t`Анкеты` }}
|
|
9
|
+
{{ subtitle }}
|
|
10
|
+
</template>
|
|
11
|
+
<script setup>
|
|
12
|
+
import { lang, tf as tfm } from '@nitra/tfm'
|
|
13
|
+
const t = tfm.bind({ tr: getTr() })
|
|
14
|
+
const subtitle = $computed(() => t`Анкеты`)
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* LOCALIZATION
|
|
18
|
+
* @returns {object} translations
|
|
19
|
+
*/
|
|
20
|
+
function getTr() {
|
|
21
|
+
return {
|
|
22
|
+
Анкеты: { en: 'Surveys', ro: 'Sondaje', tr: 'Anketler' }
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
</script>
|
|
26
|
+
```
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
## Vite-конфігурація: VueMacros, AutoImport, Layouts
|
|
2
|
+
|
|
3
|
+
Потрібно використовувати **Vite версії 8 та вище** для frontend проекту на Vue.
|
|
4
|
+
|
|
5
|
+
Потрібно використовувати **`unplugin-auto-import`** для автоматичного імпортування компонентів, composables, utils та інших функцій і прибирати з файлів усередині Vite проектів відповідні ручні імпорти, зокрема рядки виду `import { … } from 'vue'` — API Vue (`ref`, `computed`, `watch` тощо) мають підставлятися через auto-import, а не дублюватися явним імпортом з модуля `vue`.
|
|
6
|
+
|
|
7
|
+
**Виняток — бібліотеки компонентів (`vue` у `peerDependencies`).** Увесь стек auto-import застосовується лише коли `vue` підключено як звичайну `dependencies`. Якщо ж пакет оголошує `vue` у `peerDependencies` — це проєкт-бібліотека компонентів: його джерела споживаються Vite-додатками і **не** проходять через `unplugin-auto-import` споживача. Тому до таких пакетів **не** застосовуються: заборона явних `import { ref, computed, … } from 'vue'` (вони обовʼязкові), вимога `'vue'` у `AutoImport.imports`, а також вимоги наявності `VueMacros` / `AutoImport` у `vite.config`. Решта перевірок (заборона `esbuild`, `process.env.npm_lifecycle_event` тощо) лишаються. Тригер винятку — `isVueComponentLibraryPkg` у `npm/rules/vue/js/packages.mjs`.
|
|
8
|
+
|
|
9
|
+
Потрібно використовувати **`vite-plugin-vue-layouts-next`** для автоматичного імпортування layout компонентів.
|
|
10
|
+
|
|
11
|
+
### Еталонний `vite.config.js`
|
|
12
|
+
|
|
13
|
+
```javascript title="vite.config.js"
|
|
14
|
+
import Vue from '@vitejs/plugin-vue'
|
|
15
|
+
import VueMacros from 'vue-macros/vite'
|
|
16
|
+
import { defineConfig } from 'vite'
|
|
17
|
+
import AutoImport from 'unplugin-auto-import/vite'
|
|
18
|
+
import Layouts from 'vite-plugin-vue-layouts-next'
|
|
19
|
+
|
|
20
|
+
export default defineConfig({
|
|
21
|
+
plugins: [
|
|
22
|
+
AutoImport({
|
|
23
|
+
imports: [
|
|
24
|
+
// presets
|
|
25
|
+
'vue',
|
|
26
|
+
'vue-router',
|
|
27
|
+
'quasar',
|
|
28
|
+
'pinia',
|
|
29
|
+
// custom
|
|
30
|
+
{
|
|
31
|
+
'@nitra/vite-boot/apollo': [
|
|
32
|
+
// named imports
|
|
33
|
+
'gql',
|
|
34
|
+
'useQuery',
|
|
35
|
+
'useMutation',
|
|
36
|
+
'useSubscription'
|
|
37
|
+
],
|
|
38
|
+
'@nitra/consola': [
|
|
39
|
+
// named imports
|
|
40
|
+
'createLogger' // import { createLogger } from '@nitra/consola'
|
|
41
|
+
]
|
|
42
|
+
}
|
|
43
|
+
]
|
|
44
|
+
}),
|
|
45
|
+
VueMacros({
|
|
46
|
+
plugins: {
|
|
47
|
+
vue: Vue()
|
|
48
|
+
}
|
|
49
|
+
}),
|
|
50
|
+
Layouts()
|
|
51
|
+
]
|
|
52
|
+
})
|
|
53
|
+
```
|
|
54
|
+
|
|
55
|
+
### Перевірка `npm_lifecycle_event` (Bun-сумісність)
|
|
56
|
+
|
|
57
|
+
У більшості проектів у файлі `vite.config.js` є конструкція виду:
|
|
58
|
+
|
|
59
|
+
```javascript
|
|
60
|
+
switch (process.env.npm_lifecycle_event) {
|
|
61
|
+
case 'start-remote-tr': {
|
|
62
|
+
```
|
|
63
|
+
|
|
64
|
+
Вона перестала працювати з новим Bun. Тепер її треба обрамити у функцію і передавати `mode`:
|
|
65
|
+
|
|
66
|
+
```javascript title="vite.config.js"
|
|
67
|
+
function getProxy(mode) {
|
|
68
|
+
const proxy = {}
|
|
69
|
+
|
|
70
|
+
switch (mode) {
|
|
71
|
+
case 'remote-tr': {
|
|
72
|
+
proxy['^/auth/.*'] = 'https://tr.efes.cloud'
|
|
73
|
+
proxy['/file-link/'] = 'https://tr.efes.cloud'
|
|
74
|
+
break
|
|
75
|
+
}
|
|
76
|
+
default: {
|
|
77
|
+
proxy['^/auth/.*'] = 'https://dev.efes.cloud'
|
|
78
|
+
proxy['/file-link/'] = 'https://dev.efes.cloud'
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
return proxy
|
|
82
|
+
}
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
```javascript title="vite.config.js"
|
|
86
|
+
export default defineConfig(({ mode, command }) => {
|
|
87
|
+
// ...
|
|
88
|
+
server: {
|
|
89
|
+
proxy: getProxy(mode)
|
|
90
|
+
}
|
|
91
|
+
})
|
|
92
|
+
```
|
|
93
|
+
|
|
94
|
+
### esbuild заборонено
|
|
95
|
+
|
|
96
|
+
У проєкті не має бути залежності `esbuild` і згадок `esbuild` у конфігах/коді. Якщо десь є налаштування або інструкції під `esbuild` — заміни на **rolldown**.
|
|
97
|
+
|
|
98
|
+
### Еталонний `package.json`
|
|
99
|
+
|
|
100
|
+
мінімальний повний `package.json` для Vite + Vue + `vue-macros` (версії підлаштуй під проєкт):
|
|
101
|
+
|
|
102
|
+
```json title="package.json"
|
|
103
|
+
{
|
|
104
|
+
"name": "my-vue-app",
|
|
105
|
+
"private": true,
|
|
106
|
+
"type": "module",
|
|
107
|
+
"dependencies": {
|
|
108
|
+
"vue": "^3.6.12"
|
|
109
|
+
},
|
|
110
|
+
"devDependencies": {
|
|
111
|
+
"vite": "^8.0.0",
|
|
112
|
+
"@vitejs/plugin-vue": "^6.0.0",
|
|
113
|
+
"vue-macros": "^3.1.2"
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
### VS Code розширення
|
|
119
|
+
|
|
120
|
+
У файлі `.vscode/extensions.json` потрібна рекомендація для Vue:
|
|
121
|
+
|
|
122
|
+
```json title=".vscode/extensions.json"
|
|
123
|
+
{
|
|
124
|
+
"recommendations": ["Vue.volar"]
|
|
125
|
+
}
|
|
126
|
+
```
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
## Vite client types (Volar, імпорти асетів)
|
|
2
|
+
|
|
3
|
+
Без типів **Vite** редактор (Volar / TypeScript) не знає, що імпорт статичного файлу (`import url from './hero.avif'`, `*.png`, `*.svg` тощо) відповідає модулю з `string` URL. Тоді у `.vue` з'являється помилка на кшталт **Cannot find module '…' or its corresponding type declarations**.
|
|
4
|
+
|
|
5
|
+
У **кожному** workspace-пакеті з **Vue + Vite** обов'язково:
|
|
6
|
+
|
|
7
|
+
1. **`src/vite-env.d.ts`** — рівно з посиланням на клієнтські типи Vite (одного рядка достатньо):
|
|
8
|
+
|
|
9
|
+
```ts title="src/vite-env.d.ts"
|
|
10
|
+
/// <reference types="vite/client" />
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
Так підтягуються декларації з `vite/client.d.ts` (`declare module '*.avif'`, `*.png`, …).
|
|
14
|
+
|
|
15
|
+
2. **Корінь пакета:** **`jsconfig.json`** із **`include`**, що охоплює `src` (наприклад `"include": ["src/**/*"]`), щоб мова служби бачила `vite-env.d.ts` і SFC.
|
|
16
|
+
|
|
17
|
+
Мінімальний приклад для JS-пакета:
|
|
18
|
+
|
|
19
|
+
```json title="jsconfig.json"
|
|
20
|
+
{
|
|
21
|
+
"compilerOptions": {
|
|
22
|
+
"target": "ESNext",
|
|
23
|
+
"module": "ESNext",
|
|
24
|
+
"moduleResolution": "bundler",
|
|
25
|
+
"lib": ["ESNext", "DOM", "DOM.Iterable"],
|
|
26
|
+
"jsx": "preserve",
|
|
27
|
+
"strict": true,
|
|
28
|
+
"noEmit": true,
|
|
29
|
+
"skipLibCheck": true,
|
|
30
|
+
"resolveJsonModule": true,
|
|
31
|
+
"isolatedModules": true,
|
|
32
|
+
"allowJs": true
|
|
33
|
+
},
|
|
34
|
+
"include": ["src/**/*"]
|
|
35
|
+
}
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
**Не** звужуй без потреби **`compilerOptions.types`** до `["vite/client"]`: це може відрізати інші пакети з `@types` і зламати інші підказки. Достатньо `/// <reference types="vite/client" />` у `vite-env.d.ts` і коректного `include`.
|
|
39
|
+
|
|
40
|
+
## Статичні файли — BASE_URL
|
|
41
|
+
|
|
42
|
+
Для підключення статичних файлів не використовуй відносні шляхи. Завжди будуй URL через `import.meta.env.BASE_URL`:
|
|
43
|
+
|
|
44
|
+
```vue
|
|
45
|
+
<!-- Погано -->
|
|
46
|
+
<img src="/logo.png" />
|
|
47
|
+
<img src="./assets/logo.png" />
|
|
48
|
+
|
|
49
|
+
<!-- Добре -->
|
|
50
|
+
<img :src="baseUrl + 'logo.png'" />
|
|
51
|
+
```
|
|
52
|
+
|
|
53
|
+
```javascript
|
|
54
|
+
const baseUrl = import.meta.env.BASE_URL
|
|
55
|
+
```
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
## Заборона явних value-імпортів з `vue`
|
|
2
|
+
|
|
3
|
+
Потрібно використовувати `unplugin-auto-import` для автоматичного імпортування Vue API. Коли `'vue'` додано до `AutoImport.imports` у `vite.config`, явні value-імпорти вигляду `import { ref, computed } from 'vue'` у файлах додатка **заборонені** — API Vue підставляється автоматично.
|
|
4
|
+
|
|
5
|
+
Дозволені лише:
|
|
6
|
+
- side-effect: `import 'vue'`
|
|
7
|
+
- type-only: `import type { … } from 'vue'` або `import { type Foo } from 'vue'`
|
|
8
|
+
|
|
9
|
+
```vue
|
|
10
|
+
<!-- ❌ заборонено (auto-import покриває) -->
|
|
11
|
+
<script setup>
|
|
12
|
+
import { ref, computed, watch, onMounted } from 'vue'
|
|
13
|
+
</script>
|
|
14
|
+
|
|
15
|
+
<!-- ✅ дозволено — не потрібен ручний імпорт -->
|
|
16
|
+
<script setup>
|
|
17
|
+
const count = $ref(0)
|
|
18
|
+
const doubled = $computed(() => count * 2)
|
|
19
|
+
onMounted(() => { /* ... */ })
|
|
20
|
+
</script>
|
|
21
|
+
```
|
|
22
|
+
|
|
23
|
+
**Виняток:** бібліотеки компонентів (`vue` у `peerDependencies`) — їхні джерела не проходять через `unplugin-auto-import` споживача, тому явні `import { ref } from 'vue'` там обовʼязкові. Визначається через `isVueComponentLibraryPkg` у `npm/rules/vue/js/packages.mjs`.
|
|
24
|
+
|
|
25
|
+
Перевірка виконується через **oxc-parser** (`module.staticImports`) у `npm/rules/vue/lib/vue-forbidden-imports.mjs`, функція `findForbiddenVueImportsInSourceFile`. Для `.vue` — лише вміст `<script>` тегів, `<template>` ігнорується.
|