@g1cloud/bluesea 5.0.0-beta.27 → 5.0.0-beta.28

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.
Files changed (34) hide show
  1. package/README.md +21 -0
  2. package/bin/install-claude-skill.mjs +74 -0
  3. package/dist/{BSAlertModal-DT2Wai4R.js → BSAlertModal-BpbJuAe1.js} +1 -1
  4. package/dist/{BSGridColumnSettingModal-CUMe_yWj.js → BSGridColumnSettingModal-8MqhRWkU.js} +1 -1
  5. package/dist/{BSRichTextMaximizedModal-CaaSAgmI.js → BSRichTextMaximizedModal-C86Skc5v.js} +1 -1
  6. package/dist/{BSYesNoModal-C91E2MSF.js → BSYesNoModal-CHbktVAj.js} +1 -1
  7. package/dist/{BSYoutubeInputModal-DSI-NoGb.js → BSYoutubeInputModal-JKnr4hGE.js} +1 -1
  8. package/dist/{ImageInsertModal-7u7YeHsI.js → ImageInsertModal-DQwkQJ8b.js} +2 -2
  9. package/dist/{ImageProperties.vue_vue_type_script_setup_true_lang-CSHlFWfd.js → ImageProperties.vue_vue_type_script_setup_true_lang-BsMcsXdh.js} +1 -1
  10. package/dist/{ImagePropertiesModal-HRPdVJRK.js → ImagePropertiesModal-X7blKqTy.js} +2 -2
  11. package/dist/{LinkPropertiesModal-x73isyqI.js → LinkPropertiesModal-DGiiTivW.js} +1 -1
  12. package/dist/{TableInsertModal-BfWLdDCa.js → TableInsertModal-CupFfnOG.js} +1 -1
  13. package/dist/{TablePropertiesModal-D1wSXQ3C.js → TablePropertiesModal-CfK9i7Q5.js} +1 -1
  14. package/dist/{VideoInsertModal-DQ-wO6_P.js → VideoInsertModal-BwRRgibx.js} +2 -2
  15. package/dist/{VideoProperties.vue_vue_type_script_setup_true_lang-D73f1tdh.js → VideoProperties.vue_vue_type_script_setup_true_lang-zEMpmzTZ.js} +1 -1
  16. package/dist/{VideoPropertiesModal-BYADjPfV.js → VideoPropertiesModal-Dn6AzhPy.js} +2 -2
  17. package/dist/{YoutubeInsertModal-DNq4v5Ll.js → YoutubeInsertModal-DCn5bhN5.js} +2 -2
  18. package/dist/{YoutubeProperties.vue_vue_type_script_setup_true_lang-CWIVZP3H.js → YoutubeProperties.vue_vue_type_script_setup_true_lang-B-YVlp4Y.js} +1 -1
  19. package/dist/{YoutubePropertiesModal-D2-7I2sg.js → YoutubePropertiesModal-Dg-n8cTv.js} +2 -2
  20. package/dist/bluesea.js +1 -1
  21. package/dist/bluesea.umd.cjs +9 -2
  22. package/dist/{index-BIvcVEog.js → index-e3O4IL4V.js} +25 -18
  23. package/dist/text/i18n.d.ts +2 -1
  24. package/package.json +6 -1
  25. package/skills/bluesea-ui/SKILL.md +312 -0
  26. package/skills/bluesea-ui/references/components.md +189 -0
  27. package/skills/bluesea-ui/references/grid.md +159 -0
  28. package/skills/bluesea-ui/references/i18n.md +126 -0
  29. package/skills/bluesea-ui/references/validation.md +176 -0
  30. package/text/bluesea_text_en.json +248 -964
  31. package/text/bluesea_text_fr.json +248 -964
  32. package/text/bluesea_text_ja.json +248 -964
  33. package/text/bluesea_text_ko.json +248 -964
  34. package/text/bluesea_text_zh.json +248 -964
@@ -0,0 +1,126 @@
1
+ # i18n and MultiLangText
2
+
3
+ Bluesea treats every user-visible text prop as potentially multi-lingual. Understanding `MultiLangText` and the `i18n` helper up front prevents 90% of "caption is showing `{ key: 'xxx' }`" mistakes.
4
+
5
+ ## The `MultiLangText` union
6
+
7
+ ```ts
8
+ type LocaleName = string // e.g. 'ko', 'en', 'ja', 'ko-KR'
9
+
10
+ type MultiLangString = Record<LocaleName, string> // { ko: '저장', en: 'Save' }
11
+ type MultiLangMessage = { key: string; args?: unknown[]; locale?: LocaleName }
12
+
13
+ type MultiLangText =
14
+ | string
15
+ | MultiLangString
16
+ | MultiLangMessage
17
+ ```
18
+
19
+ Bluesea uses these rules to render:
20
+
21
+ 1. If it's a plain string, use it as-is.
22
+ 2. If it's an object with `key`, look the key up in the `I18NTexts` registry for the current locale and substitute `args`.
23
+ 3. Otherwise, treat it as per-locale map and pick `currentLocale` (with fallbacks).
24
+
25
+ You detect `MultiLangMessage` by the presence of a `key` property; `isMultiLangMessage(text)` does this check.
26
+
27
+ ## Registering texts
28
+
29
+ ```ts
30
+ import { i18n } from '@g1cloud/bluesea'
31
+
32
+ i18n.addTexts('ko', [
33
+ { key: 'btn.save', text: '저장' },
34
+ { key: 'err.required', text: '필수 입력 항목입니다.' },
35
+ { key: 'msg.saved', text: '{0} 개가 저장되었습니다.' }, // {0} {1} for args
36
+ { key: 'help.html', text: '<strong>주의</strong>', html: true },
37
+ ])
38
+
39
+ i18n.addTexts('en', [
40
+ { key: 'btn.save', text: 'Save' },
41
+ { key: 'err.required', text: 'This field is required.' },
42
+ { key: 'msg.saved', text: '{0} items saved.' },
43
+ ])
44
+ ```
45
+
46
+ ### Compact JSON format
47
+
48
+ For large catalogues, `addTexts` also accepts a compact object form so the JSON file stays small. Same call — the runtime detects the shape via `Array.isArray`:
49
+
50
+ ```ts
51
+ i18n.addTexts('ko', {
52
+ 'btn.save': '저장', // plain text
53
+ 'help.html': ['<strong>주의</strong>', 1], // array → html: true
54
+ })
55
+ ```
56
+
57
+ - value is a **string** → equivalent to `{ key, text, html: false }`
58
+ - value is `[text, 1]` (any truthy second element) → equivalent to `{ key, text, html: true }`
59
+
60
+ The shipped `@g1cloud/bluesea/text/bluesea_text_*.json` files use this compact format. Both formats can be mixed for the same locale; later calls overwrite earlier keys.
61
+
62
+ For real apps, store texts in `texts_ko.json` / `texts_en.json` and run the message generator (`pnpm generate-message` in bluesea-demo). The generator produces type-safe key constants.
63
+
64
+ Locale fallback chain: `currentLocale` → parent locale (`ko-KR` → `ko`) → `defaultLocale`. If no match, the key itself is returned — that is what "unresolved" text looks like in the UI.
65
+
66
+ ## Looking up at runtime
67
+
68
+ ```ts
69
+ import { t, interpretMultiLangText } from '@g1cloud/bluesea'
70
+
71
+ const msg = t({ key: 'msg.saved', args: [3] }) // → '3 items saved.' (in en)
72
+
73
+ // Generic interpreter that accepts any MultiLangText
74
+ const label = interpretMultiLangText({ ko: '저장', en: 'Save' })
75
+ ```
76
+
77
+ ## The `v-t` directive
78
+
79
+ Register once (`app.directive('t', vT)`), then use it anywhere you'd hard-code text. It sets `textContent` by default; pass a modifier to target a different attribute.
80
+
81
+ ```vue
82
+ <span v-t="{ key: 'btn.save' }" />
83
+ <input v-t.placeholder="{ key: 'ph.name' }" />
84
+ <img v-t.alt="{ key: 'alt.logo' }" />
85
+ ```
86
+
87
+ The directive is reactive — switching `blueseaConfig.currentLocale` updates all bound elements.
88
+
89
+ ## Switching locale
90
+
91
+ ```ts
92
+ import { blueseaConfig } from '@g1cloud/bluesea'
93
+ blueseaConfig.setCurrentLocale('en') // UI locale
94
+ blueseaConfig.setCurrentDataLocale('en') // BSMultiLang* pickers
95
+ ```
96
+
97
+ There is also `<BSLocaleSelect>` for a ready-made dropdown in the header.
98
+
99
+ ## Data locale vs UI locale
100
+
101
+ Bluesea separates two concerns:
102
+
103
+ - `currentLocale` / `locales` — which language the **chrome** renders in (labels, buttons, errors).
104
+ - `currentDataLocale` / `dataLocales` — which language the **data** is authored in, used by `BSMultiLang*` inputs (e.g. editing product name in `ko`, `en`, `ja`).
105
+
106
+ You can run the UI in Korean while editing English-only content, or vice versa. Most apps set both the same for end users but expose a separate data-locale toggle in admin tools.
107
+
108
+ ## `MultiLangString` in data models
109
+
110
+ When a DB field is multilingual, model it as `MultiLangString` and render it with `cellType: 'MULTI_LANG_STRING'` in grids or `<BSMultiLangTextInput>` in forms. Bluesea picks the `currentDataLocale` with the same fallback chain.
111
+
112
+ ```ts
113
+ type Product = {
114
+ id: string
115
+ name: MultiLangString // { ko: '양말', en: 'Socks', ja: '靴下' }
116
+ description: MultiLangString
117
+ }
118
+ ```
119
+
120
+ ## Common mistakes
121
+
122
+ - Hard-coding Korean strings in a library component and wondering why they don't switch locale — wrap them as `{ key: '...' }` and register for each locale.
123
+ - Using `interpretMultiLangText` inside a `computed` and expecting it not to re-run when locale changes — it does, because `blueseaConfig` is reactive.
124
+ - Passing a key that doesn't exist — the key itself leaks into the UI. Check your text files or run the message generator.
125
+ - Forgetting to register `v-t` globally and then using `v-t="..."` — the directive silently does nothing.
126
+ - Storing `{ key: '...' }` in the database. Keys are UI concerns; database data should be `MultiLangString` values.
@@ -0,0 +1,176 @@
1
+ # Validation in Bluesea
2
+
3
+ Bluesea's validation is deliberately unusual. Instead of wiring a validator per field, inputs *register themselves* on their DOM element. A `FormValidator` then walks the DOM tree inside a root element, finds every registered `FieldValidator`, runs them, and collects errors. This means you rarely touch `FieldValidator` directly — just give each input a `name` and scope a `FormValidator` to the form root.
4
+
5
+ ## Types at a glance
6
+
7
+ ```ts
8
+ type ValidationPhase = 'input' | 'change' | 'blur' | 'form'
9
+
10
+ type ValidationError = {
11
+ code: string
12
+ message: MultiLangText
13
+ }
14
+
15
+ type FieldValidationRule<T> = (
16
+ value: T,
17
+ phase: ValidationPhase,
18
+ fieldContext?: FieldContext<any>,
19
+ ) => Promise<ValidationError[] | undefined> | ValidationError[] | undefined
20
+
21
+ type FormValidationError = ValidationError & { name?: string }
22
+
23
+ type FormValidationRule = (phase?: ValidationPhase) => Promise<FormValidationError[] | undefined>
24
+
25
+ class ValidationFailedError {
26
+ constructor(public errors: FormValidationError[]) {}
27
+ }
28
+ ```
29
+
30
+ ## Standard form flow
31
+
32
+ ```vue
33
+ <template>
34
+ <div ref="formEl">
35
+ <BSTextInput v-model="form.name" name="name" required :max-length="50" />
36
+ <BSTextInput v-model="form.email" name="email" required reg-exp="^[^@]+@[^@]+$" />
37
+ <BSNumberInput v-model="form.age" name="age" :min-value="0" :max-value="150" />
38
+ <BSButton caption="저장" @click="save" />
39
+ </div>
40
+ </template>
41
+
42
+ <script setup lang="ts">
43
+ import { ref } from 'vue'
44
+ import { formValidator, isValidationFailedError, showNotification } from '@g1cloud/bluesea'
45
+
46
+ const formEl = ref<HTMLElement>()
47
+ const form = ref({ name: '', email: '', age: 0 })
48
+
49
+ const validator = formValidator({
50
+ element: formEl,
51
+ rules: [
52
+ // form-level cross-field rule
53
+ async () => form.value.age < 18 && form.value.name.includes('kid')
54
+ ? [{ code: 'ageMismatch', message: { key: 'err.ageMismatch' } }]
55
+ : undefined,
56
+ ],
57
+ })
58
+
59
+ async function save() {
60
+ try {
61
+ await validator.validate() // throws on failure
62
+ await api.save(form.value)
63
+ showNotification({ key: 'msg.saved' })
64
+ } catch (e) {
65
+ if (isValidationFailedError(e)) {
66
+ showNotification({ key: 'err.validationFailed' }, 'error')
67
+ return
68
+ }
69
+ throw e
70
+ }
71
+ }
72
+ </script>
73
+ ```
74
+
75
+ `validate()` throws `ValidationFailedError` with the full error list. `validationResult()` returns the errors without throwing, which is handy for custom UI (e.g. show a summary banner).
76
+
77
+ ## Built-in field rules per input
78
+
79
+ Built-in rules are configured via props on each input. Each has a companion `validationMessage*` prop to override the default message.
80
+
81
+ | Rule | Inputs | Props |
82
+ |---|---|---|
83
+ | required | All input components | `required`, `validationMessageRequired` |
84
+ | length | `BSTextInput`, `BSTextArea` | `minLength`, `maxLength`, `validationMessageMinLength`, `validationMessageMaxLength`, `validationMessageBetweenLength` |
85
+ | regexp | `BSTextInput` | `regExp`, `validationMessageRegExp` |
86
+ | numeric range | `BSNumberInput`, `BSPriceInput`, `BSPercentInput` | `minValue`, `maxValue`, `validationMessageMinValue`, `validationMessageMaxValue`, `validationMessageBetweenValue` |
87
+ | date order | `BSDateRange` | auto; override `validationMessageDateOrder` |
88
+ | date bounds | `BSDateInput`, `BSDateRange` | `minValue`, `maxValue` |
89
+ | file size / count | `BSFileUpload`, `BS*ImageUpload` | `maxFileSize`, `maxFileCount` |
90
+
91
+ Default error messages are i18n keys like `bs.error.validation.required`, `bs.error.validation.lengthMin`, so translations come "for free" if you registered Bluesea's text files.
92
+
93
+ ## Custom field rules (`extraValidationRules`)
94
+
95
+ When built-ins aren't enough, pass `:extra-validation-rules` — an array of functions. Each returns `ValidationError[]` (or empty / undefined for "ok"). Phase control lets you defer expensive checks.
96
+
97
+ ```ts
98
+ import type { FieldValidationRule } from '@g1cloud/bluesea'
99
+
100
+ const uniqueEmail: FieldValidationRule<string> = async (value, phase) => {
101
+ if (!value || phase === 'input') return // run on blur / form only
102
+ const exists = await api.users.exists(value)
103
+ return exists ? [{ code: 'duplicate', message: { key: 'err.emailTaken' } }] : undefined
104
+ }
105
+
106
+ // in template
107
+ <BSTextInput v-model="form.email" name="email" required
108
+ :extra-validation-rules="[uniqueEmail]" />
109
+ ```
110
+
111
+ Rules of thumb:
112
+ - Use `phase === 'form'` to skip expensive work on keystrokes.
113
+ - Return `undefined` for "pass"; return an empty array also passes, but `undefined` is clearer.
114
+ - Throw *inside* rules sparingly — the validator does not convert thrown errors into nice messages.
115
+
116
+ ## Form-level rules
117
+
118
+ Cross-field rules go in `formValidator({ rules: [...] })`. They receive the phase but no value; you read your reactive state directly.
119
+
120
+ ```ts
121
+ formValidator({
122
+ element: formEl,
123
+ rules: [
124
+ async () => form.value.password !== form.value.passwordConfirm
125
+ ? [{ name: 'passwordConfirm', code: 'mismatch', message: { key: 'err.pwMismatch' } }]
126
+ : undefined,
127
+ ],
128
+ })
129
+ ```
130
+
131
+ The `name` on the error aligns the failure with a specific input so `validator.getFieldValidator(name)` can jump to it.
132
+
133
+ ## Talking to individual FieldValidators
134
+
135
+ Rarely needed, but when you want to trigger validation imperatively:
136
+
137
+ ```ts
138
+ import { validateField, validateFields } from '@g1cloud/bluesea'
139
+
140
+ await validateField('email') // by name
141
+ await validateFields(['email', 'age'])
142
+ await validator.getFieldValidator('email')?.validate('blur')
143
+ ```
144
+
145
+ Clearing errors (e.g. on `cancel`):
146
+
147
+ ```ts
148
+ validator.clear()
149
+ ```
150
+
151
+ ## Async rules and the "disabled" gotcha
152
+
153
+ If a field is `disabled`, its validator short-circuits to valid. Pass `:force-validate-when-disabled="true"` only when you really need validation despite the disabled state (rare — usually disabled means "field is derived / not user-editable").
154
+
155
+ For conditionally-disabled rules, prefer returning `undefined` early rather than toggling `disabled`, because `disabled` also skips the built-in rules.
156
+
157
+ ## FieldContext
158
+
159
+ A `FieldContext<T>` (from `@/model/FieldContext`) is injected when a field is rendered inside a grid editor. It gives the rule access to `row`, sibling values, and the editingRows collection. Use it when one cell's validity depends on another cell in the same row.
160
+
161
+ ```ts
162
+ const priceMustExceedCost: FieldValidationRule<number> = (price, _phase, ctx) => {
163
+ const row = ctx?.row as { cost?: number } | undefined
164
+ return (row && price !== undefined && row.cost !== undefined && price < row.cost)
165
+ ? [{ code: 'priceLtCost', message: { key: 'err.priceLtCost' } }]
166
+ : undefined
167
+ }
168
+ ```
169
+
170
+ ## Common mistakes
171
+
172
+ - Forgetting `name` on an input — the validator still runs, but `error.name` is empty, so "jump to error" UX breaks.
173
+ - Wrapping the `ref` target with `v-if` that unmounts on error — the error clears because the validator is unregistered. Prefer `:disabled` or `:view-mode` to hide without unmounting.
174
+ - Calling `validator.validate()` from a plain synchronous handler — it's async. `await` it.
175
+ - Using `throw new Error(...)` in a rule. Return `[{ code, message }]` instead.
176
+ - Re-creating `formValidator` on every render; create it once in `<script setup>` at module scope.