mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
Merge branch 'dev' into next
This commit is contained in:
commit
582582e480
@ -21,7 +21,7 @@
|
|||||||
"@vant/eslint-config": "workspace:*",
|
"@vant/eslint-config": "workspace:*",
|
||||||
"@vant/stylelint-config": "workspace:*",
|
"@vant/stylelint-config": "workspace:*",
|
||||||
"eslint": "^8.2.0",
|
"eslint": "^8.2.0",
|
||||||
"husky": "^7.0.4",
|
"husky": "^8.0.1",
|
||||||
"lint-staged": "^12.1.2",
|
"lint-staged": "^12.1.2",
|
||||||
"prettier": "^2.5.0",
|
"prettier": "^2.5.0",
|
||||||
"rimraf": "^3.0.2",
|
"rimraf": "^3.0.2",
|
||||||
|
@ -1,5 +1,11 @@
|
|||||||
# 更新日志
|
# 更新日志
|
||||||
|
|
||||||
|
## v4.0.2
|
||||||
|
|
||||||
|
`2022-05-14`
|
||||||
|
|
||||||
|
- 修复编译 script setup 错误的问题
|
||||||
|
|
||||||
## v4.0.1
|
## v4.0.1
|
||||||
|
|
||||||
`2022-03-03`
|
`2022-03-03`
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vant/cli",
|
"name": "@vant/cli",
|
||||||
"version": "4.0.1",
|
"version": "4.0.2",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "lib/index.js",
|
"main": "lib/index.js",
|
||||||
"typings": "lib/index.d.ts",
|
"typings": "lib/index.d.ts",
|
||||||
@ -70,7 +70,7 @@
|
|||||||
"gh-pages": "^3.2.3",
|
"gh-pages": "^3.2.3",
|
||||||
"hash-sum": "^2.0.0",
|
"hash-sum": "^2.0.0",
|
||||||
"highlight.js": "^11.3.1",
|
"highlight.js": "^11.3.1",
|
||||||
"husky": "^7.0.4",
|
"husky": "^8.0.1",
|
||||||
"jest": "^27.3.1",
|
"jest": "^27.3.1",
|
||||||
"jest-canvas-mock": "^2.3.1",
|
"jest-canvas-mock": "^2.3.1",
|
||||||
"jest-serializer-html": "^7.1.0",
|
"jest-serializer-html": "^7.1.0",
|
||||||
|
@ -2,7 +2,7 @@ import fse from 'fs-extra';
|
|||||||
import babel from '@babel/core';
|
import babel from '@babel/core';
|
||||||
import esbuild, { type Format } from 'esbuild';
|
import esbuild, { type Format } from 'esbuild';
|
||||||
import { sep } from 'path';
|
import { sep } from 'path';
|
||||||
import { isJsx, replaceExt } from '../common/index.js';
|
import { isJsx, replaceExt, getVantConfig } from '../common/index.js';
|
||||||
import { replaceCSSImportExt } from '../common/css.js';
|
import { replaceCSSImportExt } from '../common/css.js';
|
||||||
import { replaceScriptImportExt } from './get-deps.js';
|
import { replaceScriptImportExt } from './get-deps.js';
|
||||||
|
|
||||||
@ -50,7 +50,9 @@ export async function compileScript(
|
|||||||
|
|
||||||
({ code } = esbuildResult);
|
({ code } = esbuildResult);
|
||||||
|
|
||||||
const jsFilePath = replaceExt(filePath, '.js');
|
const extensionMap = getVantConfig().build?.extensions;
|
||||||
|
const extension = extensionMap?.[format] || '.js';
|
||||||
|
const jsFilePath = replaceExt(filePath, extension);
|
||||||
removeSync(filePath);
|
removeSync(filePath);
|
||||||
outputFileSync(jsFilePath, code);
|
outputFileSync(jsFilePath, code);
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,12 @@
|
|||||||
import fse from 'fs-extra';
|
import fse from 'fs-extra';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import hash from 'hash-sum';
|
import hash from 'hash-sum';
|
||||||
import { parse, SFCBlock, compileTemplate } from 'vue/compiler-sfc';
|
import {
|
||||||
|
parse,
|
||||||
|
SFCBlock,
|
||||||
|
compileTemplate,
|
||||||
|
compileScript,
|
||||||
|
} from 'vue/compiler-sfc';
|
||||||
import { replaceExt } from '../common/index.js';
|
import { replaceExt } from '../common/index.js';
|
||||||
|
|
||||||
const { remove, readFileSync, outputFile } = fse;
|
const { remove, readFileSync, outputFile } = fse;
|
||||||
@ -73,8 +78,9 @@ export async function compileSfc(filePath: string): Promise<any> {
|
|||||||
const scopeId = hasScoped ? `data-v-${hash(source)}` : '';
|
const scopeId = hasScoped ? `data-v-${hash(source)}` : '';
|
||||||
|
|
||||||
// compile js part
|
// compile js part
|
||||||
if (descriptor.script) {
|
if (descriptor.script || descriptor.scriptSetup) {
|
||||||
const lang = descriptor.script.lang || 'js';
|
const lang =
|
||||||
|
descriptor.script?.lang || descriptor.scriptSetup?.lang || 'js';
|
||||||
const scriptFilePath = replaceExt(filePath, `.${lang}`);
|
const scriptFilePath = replaceExt(filePath, `.${lang}`);
|
||||||
|
|
||||||
tasks.push(
|
tasks.push(
|
||||||
@ -86,7 +92,14 @@ export async function compileSfc(filePath: string): Promise<any> {
|
|||||||
script += '// @ts-nocheck\n';
|
script += '// @ts-nocheck\n';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (descriptor.scriptSetup) {
|
||||||
|
script += compileScript(descriptor, {
|
||||||
|
id: scopeId,
|
||||||
|
}).content;
|
||||||
|
} else {
|
||||||
script += descriptor.script!.content;
|
script += descriptor.script!.content;
|
||||||
|
}
|
||||||
|
|
||||||
script = injectStyle(script, styles, filePath);
|
script = injectStyle(script, styles, filePath);
|
||||||
script = script.replace(EXPORT, `const ${VUEIDS} =`);
|
script = script.replace(EXPORT, `const ${VUEIDS} =`);
|
||||||
|
|
||||||
|
@ -14,7 +14,9 @@ export function getViteConfigForPackage({
|
|||||||
}): InlineConfig {
|
}): InlineConfig {
|
||||||
setBuildTarget('package');
|
setBuildTarget('package');
|
||||||
|
|
||||||
const { name } = getVantConfig();
|
const { name, build } = getVantConfig();
|
||||||
|
const entryExtension = build?.extensions?.esm || '.js';
|
||||||
|
const entry = join(ES_DIR, `index${entryExtension}`);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
root: CWD,
|
root: CWD,
|
||||||
@ -24,7 +26,7 @@ export function getViteConfigForPackage({
|
|||||||
build: {
|
build: {
|
||||||
lib: {
|
lib: {
|
||||||
name,
|
name,
|
||||||
entry: join(ES_DIR, 'index.js'),
|
entry,
|
||||||
formats,
|
formats,
|
||||||
fileName: (format: string) => {
|
fileName: (format: string) => {
|
||||||
const suffix = format === 'umd' ? '' : `.${format}`;
|
const suffix = format === 'umd' ? '' : `.${format}`;
|
||||||
|
@ -51,6 +51,7 @@ module.exports = {
|
|||||||
'vue/require-v-for-key': 'off',
|
'vue/require-v-for-key': 'off',
|
||||||
'vue/require-default-prop': 'off',
|
'vue/require-default-prop': 'off',
|
||||||
'vue/no-unused-components': 'off',
|
'vue/no-unused-components': 'off',
|
||||||
|
'vue/multi-word-component-names': 'off',
|
||||||
'vue/return-in-computed-property': 'off',
|
'vue/return-in-computed-property': 'off',
|
||||||
// typescript-eslint
|
// typescript-eslint
|
||||||
'@typescript-eslint/camelcase': 'off',
|
'@typescript-eslint/camelcase': 'off',
|
||||||
|
@ -6,9 +6,9 @@ exports[`should render content slot correctly 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
exports[`should render nothing when content is empty string 1`] = ``;
|
exports[`should render nothing when content is empty string 1`] = `""`;
|
||||||
|
|
||||||
exports[`should render nothing when content is undefined 1`] = ``;
|
exports[`should render nothing when content is undefined 1`] = `""`;
|
||||||
|
|
||||||
exports[`should render nothing when content is zero 1`] = `
|
exports[`should render nothing when content is zero 1`] = `
|
||||||
<div class="van-badge van-badge--top-right">
|
<div class="van-badge van-badge--top-right">
|
||||||
|
@ -59,6 +59,7 @@ import type {
|
|||||||
FieldFormatTrigger,
|
FieldFormatTrigger,
|
||||||
FieldValidateError,
|
FieldValidateError,
|
||||||
FieldAutosizeConfig,
|
FieldAutosizeConfig,
|
||||||
|
FieldValidationStatus,
|
||||||
FieldValidateTrigger,
|
FieldValidateTrigger,
|
||||||
FieldFormSharedProps,
|
FieldFormSharedProps,
|
||||||
} from './types';
|
} from './types';
|
||||||
@ -135,8 +136,8 @@ export default defineComponent({
|
|||||||
setup(props, { emit, slots }) {
|
setup(props, { emit, slots }) {
|
||||||
const id = useId();
|
const id = useId();
|
||||||
const state = reactive({
|
const state = reactive({
|
||||||
|
status: 'unvalidated' as FieldValidationStatus,
|
||||||
focused: false,
|
focused: false,
|
||||||
validateFailed: false,
|
|
||||||
validateMessage: '',
|
validateMessage: '',
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -181,7 +182,7 @@ export default defineComponent({
|
|||||||
rules.reduce(
|
rules.reduce(
|
||||||
(promise, rule) =>
|
(promise, rule) =>
|
||||||
promise.then(() => {
|
promise.then(() => {
|
||||||
if (state.validateFailed) {
|
if (state.status === 'failed') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -192,7 +193,7 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!runSyncRule(value, rule)) {
|
if (!runSyncRule(value, rule)) {
|
||||||
state.validateFailed = true;
|
state.status = 'failed';
|
||||||
state.validateMessage = getRuleMessage(value, rule);
|
state.validateMessage = getRuleMessage(value, rule);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -200,10 +201,10 @@ export default defineComponent({
|
|||||||
if (rule.validator) {
|
if (rule.validator) {
|
||||||
return runRuleValidator(value, rule).then((result) => {
|
return runRuleValidator(value, rule).then((result) => {
|
||||||
if (result && typeof result === 'string') {
|
if (result && typeof result === 'string') {
|
||||||
state.validateFailed = true;
|
state.status = 'failed';
|
||||||
state.validateMessage = result;
|
state.validateMessage = result;
|
||||||
} else if (result === false) {
|
} else if (result === false) {
|
||||||
state.validateFailed = true;
|
state.status = 'failed';
|
||||||
state.validateMessage = getRuleMessage(value, rule);
|
state.validateMessage = getRuleMessage(value, rule);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -213,10 +214,8 @@ export default defineComponent({
|
|||||||
);
|
);
|
||||||
|
|
||||||
const resetValidation = () => {
|
const resetValidation = () => {
|
||||||
if (state.validateFailed) {
|
state.status = 'unvalidated';
|
||||||
state.validateFailed = false;
|
|
||||||
state.validateMessage = '';
|
state.validateMessage = '';
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const validate = (rules = props.rules) =>
|
const validate = (rules = props.rules) =>
|
||||||
@ -224,12 +223,13 @@ export default defineComponent({
|
|||||||
resetValidation();
|
resetValidation();
|
||||||
if (rules) {
|
if (rules) {
|
||||||
runRules(rules).then(() => {
|
runRules(rules).then(() => {
|
||||||
if (state.validateFailed) {
|
if (state.status === 'failed') {
|
||||||
resolve({
|
resolve({
|
||||||
name: props.name,
|
name: props.name,
|
||||||
message: state.validateMessage,
|
message: state.validateMessage,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
|
state.status = 'passed';
|
||||||
resolve();
|
resolve();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -351,7 +351,7 @@ export default defineComponent({
|
|||||||
if (typeof props.error === 'boolean') {
|
if (typeof props.error === 'boolean') {
|
||||||
return props.error;
|
return props.error;
|
||||||
}
|
}
|
||||||
if (form && form.props.showError && state.validateFailed) {
|
if (form && form.props.showError && state.status === 'failed') {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -383,6 +383,8 @@ export default defineComponent({
|
|||||||
|
|
||||||
const getInputId = () => props.id || `${id}-input`;
|
const getInputId = () => props.id || `${id}-input`;
|
||||||
|
|
||||||
|
const getValidationStatus = () => state.status;
|
||||||
|
|
||||||
const renderInput = () => {
|
const renderInput = () => {
|
||||||
const controlClass = bem('control', [
|
const controlClass = bem('control', [
|
||||||
getProp('inputAlign'),
|
getProp('inputAlign'),
|
||||||
@ -407,7 +409,6 @@ export default defineComponent({
|
|||||||
name: props.name,
|
name: props.name,
|
||||||
rows: props.rows !== undefined ? +props.rows : undefined,
|
rows: props.rows !== undefined ? +props.rows : undefined,
|
||||||
class: controlClass,
|
class: controlClass,
|
||||||
value: props.modelValue,
|
|
||||||
disabled: getProp('disabled'),
|
disabled: getProp('disabled'),
|
||||||
readonly: getProp('readonly'),
|
readonly: getProp('readonly'),
|
||||||
autofocus: props.autofocus,
|
autofocus: props.autofocus,
|
||||||
@ -531,6 +532,7 @@ export default defineComponent({
|
|||||||
validate,
|
validate,
|
||||||
formValue,
|
formValue,
|
||||||
resetValidation,
|
resetValidation,
|
||||||
|
getValidationStatus,
|
||||||
});
|
});
|
||||||
|
|
||||||
provide(CUSTOM_FIELD_INJECTION_KEY, {
|
provide(CUSTOM_FIELD_INJECTION_KEY, {
|
||||||
|
@ -327,6 +327,7 @@ import type {
|
|||||||
FieldValidateError,
|
FieldValidateError,
|
||||||
FieldAutosizeConfig,
|
FieldAutosizeConfig,
|
||||||
FieldValidateTrigger,
|
FieldValidateTrigger,
|
||||||
|
FieldValidationStatus,
|
||||||
} from 'vant';
|
} from 'vant';
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -346,6 +346,7 @@ import type {
|
|||||||
FieldValidateError,
|
FieldValidateError,
|
||||||
FieldAutosizeConfig,
|
FieldAutosizeConfig,
|
||||||
FieldValidateTrigger,
|
FieldValidateTrigger,
|
||||||
|
FieldValidationStatus,
|
||||||
} from 'vant';
|
} from 'vant';
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -17,6 +17,7 @@ export type {
|
|||||||
FieldValidateError,
|
FieldValidateError,
|
||||||
FieldAutosizeConfig,
|
FieldAutosizeConfig,
|
||||||
FieldValidateTrigger,
|
FieldValidateTrigger,
|
||||||
|
FieldValidationStatus,
|
||||||
} from './types';
|
} from './types';
|
||||||
|
|
||||||
declare module 'vue' {
|
declare module 'vue' {
|
||||||
|
@ -66,6 +66,8 @@ export type FieldRule = {
|
|||||||
formatter?: FiledRuleFormatter;
|
formatter?: FiledRuleFormatter;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type FieldValidationStatus = 'passed' | 'failed' | 'unvalidated';
|
||||||
|
|
||||||
// Shared props of Field and Form
|
// Shared props of Field and Form
|
||||||
export type FieldFormSharedProps =
|
export type FieldFormSharedProps =
|
||||||
| 'colon'
|
| 'colon'
|
||||||
@ -83,6 +85,7 @@ export type FieldExpose = {
|
|||||||
rules?: FieldRule[] | undefined
|
rules?: FieldRule[] | undefined
|
||||||
) => Promise<void | FieldValidateError>;
|
) => Promise<void | FieldValidateError>;
|
||||||
resetValidation: () => void;
|
resetValidation: () => void;
|
||||||
|
getValidationStatus: () => FieldValidationStatus;
|
||||||
/** @private */
|
/** @private */
|
||||||
formValue: ComputedRef<unknown>;
|
formValue: ComputedRef<unknown>;
|
||||||
};
|
};
|
||||||
|
@ -18,6 +18,7 @@ import type {
|
|||||||
FieldTextAlign,
|
FieldTextAlign,
|
||||||
FieldValidateError,
|
FieldValidateError,
|
||||||
FieldValidateTrigger,
|
FieldValidateTrigger,
|
||||||
|
FieldValidationStatus,
|
||||||
} from '../field/types';
|
} from '../field/types';
|
||||||
import type { FormExpose } from './types';
|
import type { FormExpose } from './types';
|
||||||
|
|
||||||
@ -141,6 +142,12 @@ export default defineComponent({
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getValidationStatus = () =>
|
||||||
|
children.reduce<Record<string, FieldValidationStatus>>((form, field) => {
|
||||||
|
form[field.name] = field.getValidationStatus();
|
||||||
|
return form;
|
||||||
|
}, {});
|
||||||
|
|
||||||
const scrollToField = (
|
const scrollToField = (
|
||||||
name: string,
|
name: string,
|
||||||
options?: boolean | ScrollIntoViewOptions
|
options?: boolean | ScrollIntoViewOptions
|
||||||
@ -186,6 +193,7 @@ export default defineComponent({
|
|||||||
getValues,
|
getValues,
|
||||||
scrollToField,
|
scrollToField,
|
||||||
resetValidation,
|
resetValidation,
|
||||||
|
getValidationStatus,
|
||||||
});
|
});
|
||||||
|
|
||||||
return () => (
|
return () => (
|
||||||
|
@ -538,9 +538,10 @@ Use [ref](https://v3.vuejs.org/guide/component-template-refs.html) to get Form i
|
|||||||
| Name | Description | Attribute | Return value |
|
| Name | Description | Attribute | Return value |
|
||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| submit | Submit form | - | - |
|
| submit | Submit form | - | - |
|
||||||
| validate | Validate form | _name?: string \| string[]_ | _Promise_ |
|
|
||||||
| getValues `v3.4.8` | Get current form values | - | _Record<string, unknown>_ |
|
| getValues `v3.4.8` | Get current form values | - | _Record<string, unknown>_ |
|
||||||
|
| validate | Validate form | _name?: string \| string[]_ | _Promise\<void\>_ |
|
||||||
| resetValidation | Reset validation | _name?: string \| string[]_ | - |
|
| resetValidation | Reset validation | _name?: string \| string[]_ | - |
|
||||||
|
| getValidationStatus `v3.5.0` | Get validation status of all fields,status can be `passed`、`failed`、`unvalidated` | - | _Record\<string, FieldValidationStatus\>_ |
|
||||||
| scrollToField | Scroll to field | _name: string, alignToTop: boolean_ | - |
|
| scrollToField | Scroll to field | _name: string, alignToTop: boolean_ | - |
|
||||||
|
|
||||||
### Types
|
### Types
|
||||||
|
@ -576,9 +576,10 @@ export default {
|
|||||||
| 方法名 | 说明 | 参数 | 返回值 |
|
| 方法名 | 说明 | 参数 | 返回值 |
|
||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| submit | 提交表单,与点击提交按钮的效果等价 | - | - |
|
| submit | 提交表单,与点击提交按钮的效果等价 | - | - |
|
||||||
| validate | 验证表单,支持传入 `name` 来验证单个或部分表单项 | _name?: string \| string[]_ | _Promise_ |
|
|
||||||
| getValues `v3.4.8` | 获取所有表单项当前的值 | - | _Record<string, unknown>_ |
|
| getValues `v3.4.8` | 获取所有表单项当前的值 | - | _Record<string, unknown>_ |
|
||||||
| resetValidation | 重置表单项的验证提示,支持传入 `name` 来重置单个或部分表单项 | _name?: string \| string[]_ | - |
|
| validate | 验证表单,支持传入一个或多个 `name` 来验证单个或部分表单项,不传入 `name` 时,会验证所有表单项 | _name?: string \| string[]_ | _Promise\<void\>_ |
|
||||||
|
| resetValidation | 重置表单项的验证提示,支持传入一个或多个 `name` 来重置单个或部分表单项,不传入 `name` 时,会重置所有表单项 | _name?: string \| string[]_ | - |
|
||||||
|
| getValidationStatus `v3.5.0` | 获取所有表单项的校验状态,状态包括 `passed`、`failed`、`unvalidated` | - | _Record\<string, FieldValidationStatus\>_ |
|
||||||
| scrollToField | 滚动到对应表单项的位置,默认滚动到顶部,第二个参数传 false 可滚动至底部 | _name: string, alignToTop: boolean_ | - |
|
| scrollToField | 滚动到对应表单项的位置,默认滚动到顶部,第二个参数传 false 可滚动至底部 | _name: string, alignToTop: boolean_ | - |
|
||||||
|
|
||||||
### 类型定义
|
### 类型定义
|
||||||
|
@ -459,7 +459,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
|
@ -153,3 +153,37 @@ test('getValues method should return all current values', () => {
|
|||||||
|
|
||||||
expect(formRef.value?.getValues()).toEqual({ A: '123', B: '456' });
|
expect(formRef.value?.getValues()).toEqual({ A: '123', B: '456' });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('getValidationStatus method should the status of all fields', async () => {
|
||||||
|
const formRef = ref<FormInstance>();
|
||||||
|
const rules = getSimpleRules();
|
||||||
|
mount({
|
||||||
|
render() {
|
||||||
|
return (
|
||||||
|
<Form ref={formRef}>
|
||||||
|
<Field name="A" rules={rules.rulesA} modelValue="123" />
|
||||||
|
<Field name="B" rules={rules.rulesB} modelValue="456" />
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(formRef.value?.getValidationStatus()).toEqual({
|
||||||
|
A: 'unvalidated',
|
||||||
|
B: 'unvalidated',
|
||||||
|
});
|
||||||
|
|
||||||
|
await formRef.value?.validate();
|
||||||
|
|
||||||
|
expect(formRef.value?.getValidationStatus()).toEqual({
|
||||||
|
A: 'passed',
|
||||||
|
B: 'passed',
|
||||||
|
});
|
||||||
|
|
||||||
|
formRef.value?.resetValidation();
|
||||||
|
|
||||||
|
expect(formRef.value?.getValidationStatus()).toEqual({
|
||||||
|
A: 'unvalidated',
|
||||||
|
B: 'unvalidated',
|
||||||
|
});
|
||||||
|
});
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import type { ComponentPublicInstance } from 'vue';
|
import type { ComponentPublicInstance } from 'vue';
|
||||||
import type { FormProps } from './Form';
|
import type { FormProps } from './Form';
|
||||||
|
import type { FieldValidationStatus } from '../field';
|
||||||
|
|
||||||
export type FormExpose = {
|
export type FormExpose = {
|
||||||
submit: () => void;
|
submit: () => void;
|
||||||
@ -10,6 +11,7 @@ export type FormExpose = {
|
|||||||
options?: boolean | ScrollIntoViewOptions | undefined
|
options?: boolean | ScrollIntoViewOptions | undefined
|
||||||
) => void;
|
) => void;
|
||||||
resetValidation: (name?: string | string[] | undefined) => void;
|
resetValidation: (name?: string | string[] | undefined) => void;
|
||||||
|
getValidationStatus: () => Record<string, FieldValidationStatus>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type FormProvide = {
|
export type FormProvide = {
|
||||||
|
@ -41,13 +41,16 @@ Current supported languages:
|
|||||||
|
|
||||||
| Language | Filename | Version |
|
| Language | Filename | Version |
|
||||||
| ------------------------ | ------------ | -------- |
|
| ------------------------ | ------------ | -------- |
|
||||||
|
| Bulgarian | bg-BG | `v3.5.0` |
|
||||||
| Bangla (Bangladesh) | bn-BD | `v3.4.5` |
|
| Bangla (Bangladesh) | bn-BD | `v3.4.5` |
|
||||||
| Danish | da-DK | `v3.4.8` |
|
| Danish | da-DK | `v3.4.8` |
|
||||||
| German | de-DE | - |
|
| German | de-DE | - |
|
||||||
| German (formal) | de-DE-formal | - |
|
| German (formal) | de-DE-formal | - |
|
||||||
|
| Greek | el-GR | `v3.5.0` |
|
||||||
| English | en-US | - |
|
| English | en-US | - |
|
||||||
| Spanish (Spain) | es-ES | - |
|
| Spanish (Spain) | es-ES | - |
|
||||||
| French | fr-FR | - |
|
| French | fr-FR | - |
|
||||||
|
| Hebrew | he-IL | `v3.5.0` |
|
||||||
| Hindi | hi-IN | `v3.4.3` |
|
| Hindi | hi-IN | `v3.4.3` |
|
||||||
| Indonesian | id-ID | `v3.4.5` |
|
| Indonesian | id-ID | `v3.4.5` |
|
||||||
| Icelandic | is-IS | `v3.4.7` |
|
| Icelandic | is-IS | `v3.4.7` |
|
||||||
|
@ -42,13 +42,16 @@ Locale.add(messages);
|
|||||||
|
|
||||||
| 语言 | 文件名 | 版本 |
|
| 语言 | 文件名 | 版本 |
|
||||||
| -------------------- | ------------ | -------- |
|
| -------------------- | ------------ | -------- |
|
||||||
|
| 保加利亚语 | bg-BG | `v3.5.0` |
|
||||||
| 孟加拉语(孟加拉国) | bn-BD | `v3.4.5` |
|
| 孟加拉语(孟加拉国) | bn-BD | `v3.4.5` |
|
||||||
| 丹麦语 | da-DK | `v3.4.8` |
|
| 丹麦语 | da-DK | `v3.4.8` |
|
||||||
| 德语 | de-DE | - |
|
| 德语 | de-DE | - |
|
||||||
| 德语(正式) | de-DE-formal | - |
|
| 德语(正式) | de-DE-formal | - |
|
||||||
|
| 希腊语 | el-GR | `v3.5.0` |
|
||||||
| 英语 | en-US | - |
|
| 英语 | en-US | - |
|
||||||
| 西班牙语 | es-ES | - |
|
| 西班牙语 | es-ES | - |
|
||||||
| 法语 | fr-FR | - |
|
| 法语 | fr-FR | - |
|
||||||
|
| 希伯来语 | he-IL | `v3.5.0` |
|
||||||
| 印地语 | hi-IN | `v3.4.3` |
|
| 印地语 | hi-IN | `v3.4.3` |
|
||||||
| 印度尼西亚语 | id-ID | `v3.4.5` |
|
| 印度尼西亚语 | id-ID | `v3.4.5` |
|
||||||
| 冰岛语 | is-IS | `v3.4.7` |
|
| 冰岛语 | is-IS | `v3.4.7` |
|
||||||
|
71
packages/vant/src/locale/lang/bg-BG.ts
Normal file
71
packages/vant/src/locale/lang/bg-BG.ts
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
export default {
|
||||||
|
name: 'Име',
|
||||||
|
tel: 'Телефон',
|
||||||
|
save: 'Запазване',
|
||||||
|
confirm: 'Потвърди',
|
||||||
|
cancel: 'Отказ',
|
||||||
|
delete: 'Изтриване',
|
||||||
|
loading: 'Зареждане...',
|
||||||
|
noCoupon: 'Без купони',
|
||||||
|
nameEmpty: 'Моля, попълнете името',
|
||||||
|
addContact: 'Добавяне на контакт',
|
||||||
|
telInvalid: 'Неправилно формиран телефонен номер',
|
||||||
|
vanCalendar: {
|
||||||
|
end: 'Край',
|
||||||
|
start: 'Старт',
|
||||||
|
title: 'Календар',
|
||||||
|
weekdays: [
|
||||||
|
'неделя',
|
||||||
|
'понеделник',
|
||||||
|
'вторник',
|
||||||
|
'сряда',
|
||||||
|
'четвъртък',
|
||||||
|
'петък',
|
||||||
|
'събота',
|
||||||
|
],
|
||||||
|
monthTitle: (year: number, month: number) => `${year}/${month}`,
|
||||||
|
rangePrompt: (maxRange: number) => `Изберете не повече от ${maxRange} дни`,
|
||||||
|
},
|
||||||
|
vanCascader: {
|
||||||
|
select: 'Избор',
|
||||||
|
},
|
||||||
|
vanPagination: {
|
||||||
|
prev: 'Предишна',
|
||||||
|
next: 'Напред',
|
||||||
|
},
|
||||||
|
vanPullRefresh: {
|
||||||
|
pulling: 'Издърпайте за опресняване...',
|
||||||
|
loosing: 'Разхлабен за опресняване...',
|
||||||
|
},
|
||||||
|
vanSubmitBar: {
|
||||||
|
label: 'Общо:',
|
||||||
|
},
|
||||||
|
vanCoupon: {
|
||||||
|
unlimited: 'Неограничен',
|
||||||
|
discount: (discount: number) => `${discount * 10}% отстъпка`,
|
||||||
|
condition: (condition: number) => `Поне ${condition}`,
|
||||||
|
},
|
||||||
|
vanCouponCell: {
|
||||||
|
title: 'Купон',
|
||||||
|
count: (count: number) => `Имате ${count} купони`,
|
||||||
|
},
|
||||||
|
vanCouponList: {
|
||||||
|
exchange: 'Размяна',
|
||||||
|
close: 'Затвори',
|
||||||
|
enable: 'Налично',
|
||||||
|
disabled: 'Недостъпно',
|
||||||
|
placeholder: 'Код на купон',
|
||||||
|
},
|
||||||
|
vanAddressEdit: {
|
||||||
|
area: 'Площ',
|
||||||
|
postal: 'Пощенски',
|
||||||
|
areaEmpty: 'Моля, изберете зона за получаване',
|
||||||
|
addressEmpty: 'Адресът не може да бъде празен',
|
||||||
|
postalEmpty: 'Грешен пощенски код',
|
||||||
|
addressDetail: 'Адрес',
|
||||||
|
defaultAddress: 'Задаване като адрес по подразбиране',
|
||||||
|
},
|
||||||
|
vanAddressList: {
|
||||||
|
add: 'Добавяне на нов адрес',
|
||||||
|
},
|
||||||
|
};
|
72
packages/vant/src/locale/lang/el-GR.ts
Normal file
72
packages/vant/src/locale/lang/el-GR.ts
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
export default {
|
||||||
|
name: 'Όνομα',
|
||||||
|
tel: 'Τηλέφωνο',
|
||||||
|
save: 'Αποθήκευση',
|
||||||
|
confirm: 'Επιβεβαίωση',
|
||||||
|
cancel: 'Ακύρωση',
|
||||||
|
delete: 'Διαγραφή',
|
||||||
|
loading: 'Φόρτωση...',
|
||||||
|
noCoupon: 'Χωρίς κουπόνια',
|
||||||
|
nameEmpty: 'Παρακαλώ συμπληρώστε το όνομα',
|
||||||
|
addContact: 'Προσθήκη επαφής',
|
||||||
|
telInvalid: 'Αριθμός τηλεφώνου με εσφαλμένη μορφή',
|
||||||
|
vanCalendar: {
|
||||||
|
end: 'Τέλος',
|
||||||
|
start: 'Έναρξη',
|
||||||
|
title: 'Ημερολόγιο',
|
||||||
|
weekdays: [
|
||||||
|
'Κυριακή',
|
||||||
|
'Δευτέρα',
|
||||||
|
'Τρίτη',
|
||||||
|
'Τετάρτη',
|
||||||
|
'Πέμπτη',
|
||||||
|
'Παρασκευή',
|
||||||
|
'Σάββατο',
|
||||||
|
],
|
||||||
|
monthTitle: (year: number, month: number) => `${year}/${month}`,
|
||||||
|
rangePrompt: (maxRange: number) =>
|
||||||
|
`Επιλέξτε όχι περισσότερες από ${maxRange} ημέρες`,
|
||||||
|
},
|
||||||
|
vanCascader: {
|
||||||
|
select: 'Επιλογή',
|
||||||
|
},
|
||||||
|
vanPagination: {
|
||||||
|
prev: 'Προηγούμενο',
|
||||||
|
next: 'Επόμενο',
|
||||||
|
},
|
||||||
|
vanPullRefresh: {
|
||||||
|
pulling: 'Τραβήξτε για ανανέωση...',
|
||||||
|
loosing: 'Χαλαρά για ανανέωση...',
|
||||||
|
},
|
||||||
|
vanSubmitBar: {
|
||||||
|
label: 'Σύνολο:',
|
||||||
|
},
|
||||||
|
vanCoupon: {
|
||||||
|
unlimited: 'Απεριόριστο',
|
||||||
|
discount: (discount: number) => `${discount * 10}% έκπτωση`,
|
||||||
|
condition: (condition: number) => `Τουλάχιστον ${condition}`,
|
||||||
|
},
|
||||||
|
vanCouponCell: {
|
||||||
|
title: 'Κουπόνι',
|
||||||
|
count: (count: number) => `Έχετε ${count} κουπόνια`,
|
||||||
|
},
|
||||||
|
vanCouponList: {
|
||||||
|
exchange: 'Ανταλλαγή',
|
||||||
|
close: 'Κλείσιμο',
|
||||||
|
enable: 'Διαθέσιμο',
|
||||||
|
disabled: 'Μη διαθέσιμο',
|
||||||
|
placeholder: 'Κωδικός κουπονιού',
|
||||||
|
},
|
||||||
|
vanAddressEdit: {
|
||||||
|
area: 'Περιοχή',
|
||||||
|
postal: 'Ταχυδρομείο',
|
||||||
|
areaEmpty: 'Παρακαλώ επιλέξτε μια περιοχή λήψης',
|
||||||
|
addressEmpty: 'Η διεύθυνση δεν μπορεί να είναι κενή',
|
||||||
|
postalEmpty: 'Λάθος ταχυδρομικός κώδικας',
|
||||||
|
addressDetail: 'Διεύθυνση',
|
||||||
|
defaultAddress: 'Ορισμός ως προεπιλεγμένη διεύθυνση',
|
||||||
|
},
|
||||||
|
vanAddressList: {
|
||||||
|
add: 'Προσθήκη νέας διεύθυνσης',
|
||||||
|
},
|
||||||
|
};
|
63
packages/vant/src/locale/lang/he-IL.ts
Normal file
63
packages/vant/src/locale/lang/he-IL.ts
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
export default {
|
||||||
|
name: 'שם',
|
||||||
|
tel: 'טלפון',
|
||||||
|
save: 'שמור',
|
||||||
|
confirm: 'אישור',
|
||||||
|
cancel: 'ביטול',
|
||||||
|
delete: 'מחיקה',
|
||||||
|
loading: 'טוען...',
|
||||||
|
noCoupon: 'אין קופונים',
|
||||||
|
nameEmpty: 'אנא מלא את השדה',
|
||||||
|
addContact: 'הוסף איש-קשר',
|
||||||
|
telInvalid: 'מספר טלפון שגוי',
|
||||||
|
vanCalendar: {
|
||||||
|
end: 'סוף',
|
||||||
|
start: 'התחלה',
|
||||||
|
title: 'לוח שנה',
|
||||||
|
weekdays: ['ראשון', 'שני', 'שלישי', 'רביעי', 'חמישי', 'שישי', 'שבת'],
|
||||||
|
monthTitle: (year: number, month: number) => `${year}/${month}`,
|
||||||
|
rangePrompt: (maxRange: number) => ` בחר לא יותר מ ${maxRange} ימים `,
|
||||||
|
},
|
||||||
|
vanCascader: {
|
||||||
|
select: 'בחר',
|
||||||
|
},
|
||||||
|
vanPagination: {
|
||||||
|
prev: 'הקודם',
|
||||||
|
next: 'הבא',
|
||||||
|
},
|
||||||
|
vanPullRefresh: {
|
||||||
|
pulling: 'גרור כדי לרענן',
|
||||||
|
loosing: 'שחרר כדי לרענן',
|
||||||
|
},
|
||||||
|
vanSubmitBar: {
|
||||||
|
label: 'סך הכל:',
|
||||||
|
},
|
||||||
|
vanCoupon: {
|
||||||
|
unlimited: 'ללא הגבלה',
|
||||||
|
discount: (discount: number) => `${discount * 10}% הנחה`,
|
||||||
|
condition: (condition: number) => ` לפחות ${condition}`,
|
||||||
|
},
|
||||||
|
vanCouponCell: {
|
||||||
|
title: 'קופון',
|
||||||
|
count: (count: number) => ` יש לך ${count} קופונים `,
|
||||||
|
},
|
||||||
|
vanCouponList: {
|
||||||
|
exchange: 'החלפה',
|
||||||
|
close: 'סגירה',
|
||||||
|
enable: 'זמינים',
|
||||||
|
disabled: 'לא זמינים',
|
||||||
|
placeholder: 'קוד קופון',
|
||||||
|
},
|
||||||
|
vanAddressEdit: {
|
||||||
|
area: 'איזור',
|
||||||
|
postal: 'מיקוד',
|
||||||
|
areaEmpty: 'אנא בחר איזור קבלה',
|
||||||
|
addressEmpty: 'יש למלא כתובת',
|
||||||
|
postalEmpty: 'טעות במיקוד',
|
||||||
|
addressDetail: 'כתובת',
|
||||||
|
defaultAddress: 'הגדר ככתובת ברירת מחדש',
|
||||||
|
},
|
||||||
|
vanAddressList: {
|
||||||
|
add: 'הוספת כתובת חדשה',
|
||||||
|
},
|
||||||
|
};
|
@ -238,3 +238,28 @@ test('should not render mask and frame when options is empty', async () => {
|
|||||||
expect(wrapper.find('.van-picker__mask').exists()).toBeTruthy();
|
expect(wrapper.find('.van-picker__mask').exists()).toBeTruthy();
|
||||||
expect(wrapper.find('.van-picker__frame').exists()).toBeTruthy();
|
expect(wrapper.find('.van-picker__frame').exists()).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('columns-field-names responsiveness', async () => {
|
||||||
|
const columnsOne = [
|
||||||
|
{ type: 1, name: 'Ios' },
|
||||||
|
{ type: 2, name: 'Android' },
|
||||||
|
];
|
||||||
|
const columnsTwo = [
|
||||||
|
{ type: 1, serverName: 'server1' },
|
||||||
|
{ type: 2, serverName: 'server2' },
|
||||||
|
];
|
||||||
|
const wrapper = mount(Picker, {
|
||||||
|
props: {
|
||||||
|
columns: columnsOne,
|
||||||
|
columnsFieldNames: {
|
||||||
|
text: 'name',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(wrapper.findAll('.van-ellipsis')[0].text()).toEqual('Ios');
|
||||||
|
await wrapper.setProps({
|
||||||
|
columns: columnsTwo,
|
||||||
|
columnsFieldNames: { text: 'serverName' },
|
||||||
|
});
|
||||||
|
expect(wrapper.findAll('.van-ellipsis')[0].text()).toEqual('server1');
|
||||||
|
});
|
||||||
|
@ -143,8 +143,8 @@ export default {
|
|||||||
| show-plus | Whether to show plus button | _boolean_ | `true` |
|
| show-plus | Whether to show plus button | _boolean_ | `true` |
|
||||||
| show-minus | Whether to show minus button | _boolean_ | `true` |
|
| show-minus | Whether to show minus button | _boolean_ | `true` |
|
||||||
| show-input | Whether to show input | _boolean_ | `true` |
|
| show-input | Whether to show input | _boolean_ | `true` |
|
||||||
| long-press | Whether to allow long press | _boolean_ | `true` |
|
| long-press | Whether to enable the long press gesture, when enabled you can long press the increase and decrease buttons | _boolean_ | `true` |
|
||||||
| allow-empty | Whether to allow the input to be empty | _boolean_ | `false` |
|
| allow-empty | Whether to allow the input value to be empty, set to `true` to allow an empty string to be passed in | _boolean_ | `false` |
|
||||||
|
|
||||||
### Events
|
### Events
|
||||||
|
|
||||||
|
@ -163,8 +163,8 @@ export default {
|
|||||||
| show-plus | 是否显示增加按钮 | _boolean_ | `true` |
|
| show-plus | 是否显示增加按钮 | _boolean_ | `true` |
|
||||||
| show-minus | 是否显示减少按钮 | _boolean_ | `true` |
|
| show-minus | 是否显示减少按钮 | _boolean_ | `true` |
|
||||||
| show-input | 是否显示输入框 | _boolean_ | `true` |
|
| show-input | 是否显示输入框 | _boolean_ | `true` |
|
||||||
| long-press | 是否开启长按手势 | _boolean_ | `true` |
|
| long-press | 是否开启长按手势,开启后可以长按增加和减少按钮 | _boolean_ | `true` |
|
||||||
| allow-empty | 是否允许输入的值为空 | _boolean_ | `false` |
|
| allow-empty | 是否允许输入的值为空,设置为 `true` 后允许传入空字符串 | _boolean_ | `false` |
|
||||||
|
|
||||||
### Events
|
### Events
|
||||||
|
|
||||||
|
@ -59,6 +59,37 @@ export default {
|
|||||||
<van-switch v-model="checked" active-color="#ee0a24" inactive-color="#dcdee0" />
|
<van-switch v-model="checked" active-color="#ee0a24" inactive-color="#dcdee0" />
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Custom Node
|
||||||
|
|
||||||
|
Using `node` slot to custom the content of the node.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-switch v-model="checked">
|
||||||
|
<div class="icon-wrapper">
|
||||||
|
<van-icon :name="checked ? 'success' : 'cross'" />
|
||||||
|
</div>
|
||||||
|
</van-switch>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.icon-wrapper {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wrapper .van-icon-success {
|
||||||
|
line-height: 32px;
|
||||||
|
color: var(--van-blue);
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wrapper .van-icon-cross {
|
||||||
|
line-height: 32px;
|
||||||
|
color: var(--van-gray-5);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
```
|
||||||
|
|
||||||
### Async Control
|
### Async Control
|
||||||
|
|
||||||
```html
|
```html
|
||||||
@ -121,6 +152,12 @@ export default {
|
|||||||
| change | Emitted when check status changed | _value: any_ |
|
| change | Emitted when check status changed | _value: any_ |
|
||||||
| click | Emitted when component is clicked | _event: MouseEvent_ |
|
| click | Emitted when component is clicked | _event: MouseEvent_ |
|
||||||
|
|
||||||
|
### Slots
|
||||||
|
|
||||||
|
| Name | Description | SlotProps |
|
||||||
|
| ------------- | -------------------------- | --------- |
|
||||||
|
| node `v3.5.0` | Custom the content of node | - |
|
||||||
|
|
||||||
### Types
|
### Types
|
||||||
|
|
||||||
The component exports the following type definitions:
|
The component exports the following type definitions:
|
||||||
|
@ -69,6 +69,37 @@ export default {
|
|||||||
<van-switch v-model="checked" active-color="#ee0a24" inactive-color="#dcdee0" />
|
<van-switch v-model="checked" active-color="#ee0a24" inactive-color="#dcdee0" />
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### 自定义按钮
|
||||||
|
|
||||||
|
通过 `node` 插槽自定义按钮的内容。
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-switch v-model="checked">
|
||||||
|
<div class="icon-wrapper">
|
||||||
|
<van-icon :name="checked ? 'success' : 'cross'" />
|
||||||
|
</div>
|
||||||
|
</van-switch>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.icon-wrapper {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wrapper .van-icon-success {
|
||||||
|
line-height: 32px;
|
||||||
|
color: var(--van-blue);
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wrapper .van-icon-cross {
|
||||||
|
line-height: 32px;
|
||||||
|
color: var(--van-gray-5);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
```
|
||||||
|
|
||||||
### 异步控制
|
### 异步控制
|
||||||
|
|
||||||
需要异步控制开关时,可以使用 `modelValue` 属性和 `update:model-value` 事件代替 `v-model`,并在事件回调函数中手动处理开关状态。
|
需要异步控制开关时,可以使用 `modelValue` 属性和 `update:model-value` 事件代替 `v-model`,并在事件回调函数中手动处理开关状态。
|
||||||
@ -133,6 +164,12 @@ export default {
|
|||||||
| change | 开关状态切换时触发 | _value: any_ |
|
| change | 开关状态切换时触发 | _value: any_ |
|
||||||
| click | 点击时触发 | _event: MouseEvent_ |
|
| click | 点击时触发 | _event: MouseEvent_ |
|
||||||
|
|
||||||
|
### Slots
|
||||||
|
|
||||||
|
| 名称 | 说明 | 参数 |
|
||||||
|
| ------------- | ---------------- | ---- |
|
||||||
|
| node `v3.5.0` | 自定义按钮的内容 | - |
|
||||||
|
|
||||||
### 类型定义
|
### 类型定义
|
||||||
|
|
||||||
组件导出以下类型定义:
|
组件导出以下类型定义:
|
||||||
|
@ -31,7 +31,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
emits: ['change', 'update:modelValue'],
|
emits: ['change', 'update:modelValue'],
|
||||||
|
|
||||||
setup(props, { emit }) {
|
setup(props, { emit, slots }) {
|
||||||
const isChecked = () => props.modelValue === props.activeValue;
|
const isChecked = () => props.modelValue === props.activeValue;
|
||||||
|
|
||||||
const onClick = () => {
|
const onClick = () => {
|
||||||
@ -47,6 +47,9 @@ export default defineComponent({
|
|||||||
const color = isChecked() ? props.activeColor : props.inactiveColor;
|
const color = isChecked() ? props.activeColor : props.inactiveColor;
|
||||||
return <Loading class={bem('loading')} color={color} />;
|
return <Loading class={bem('loading')} color={color} />;
|
||||||
}
|
}
|
||||||
|
if (slots.node) {
|
||||||
|
return slots.node();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
useCustomFieldValue(() => props.modelValue);
|
useCustomFieldValue(() => props.modelValue);
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import VanSwitch from '..';
|
import VanSwitch from '..';
|
||||||
import VanCell from '../../cell';
|
import VanCell from '../../cell';
|
||||||
|
import VanIcon from '../../icon';
|
||||||
import { ref } from 'vue';
|
import { ref } from 'vue';
|
||||||
import { useTranslate } from '../../../docs/site';
|
import { useTranslate } from '../../../docs/site';
|
||||||
import { Dialog } from '../../dialog';
|
import { Dialog } from '../../dialog';
|
||||||
@ -12,6 +13,7 @@ const t = useTranslate({
|
|||||||
message: '是否切换开关?',
|
message: '是否切换开关?',
|
||||||
withCell: '搭配单元格使用',
|
withCell: '搭配单元格使用',
|
||||||
customSize: '自定义大小',
|
customSize: '自定义大小',
|
||||||
|
customNode: '自定义按钮',
|
||||||
customColor: '自定义颜色',
|
customColor: '自定义颜色',
|
||||||
asyncControl: '异步控制',
|
asyncControl: '异步控制',
|
||||||
},
|
},
|
||||||
@ -21,6 +23,7 @@ const t = useTranslate({
|
|||||||
message: 'Are you sure to toggle switch?',
|
message: 'Are you sure to toggle switch?',
|
||||||
withCell: 'Inside a Cell',
|
withCell: 'Inside a Cell',
|
||||||
customSize: 'Custom Size',
|
customSize: 'Custom Size',
|
||||||
|
customNode: 'Custom Node',
|
||||||
customColor: 'Custom Color',
|
customColor: 'Custom Color',
|
||||||
asyncControl: 'Async Control',
|
asyncControl: 'Async Control',
|
||||||
},
|
},
|
||||||
@ -67,6 +70,16 @@ const onUpdateValue = (checked: boolean) => {
|
|||||||
/>
|
/>
|
||||||
</demo-block>
|
</demo-block>
|
||||||
|
|
||||||
|
<demo-block :title="t('customNode')">
|
||||||
|
<van-switch v-model="checked3">
|
||||||
|
<template #node>
|
||||||
|
<div class="icon-wrapper">
|
||||||
|
<van-icon :name="checked3 ? 'success' : 'cross'" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</van-switch>
|
||||||
|
</demo-block>
|
||||||
|
|
||||||
<demo-block :title="t('asyncControl')">
|
<demo-block :title="t('asyncControl')">
|
||||||
<van-switch :model-value="checked4" @update:model-value="onUpdateValue" />
|
<van-switch :model-value="checked4" @update:model-value="onUpdateValue" />
|
||||||
</demo-block>
|
</demo-block>
|
||||||
@ -85,5 +98,24 @@ const onUpdateValue = (checked: boolean) => {
|
|||||||
.van-switch {
|
.van-switch {
|
||||||
margin-left: var(--van-padding-md);
|
margin-left: var(--van-padding-md);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.icon-wrapper {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 18px;
|
||||||
|
|
||||||
|
.van-icon {
|
||||||
|
line-height: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.van-icon-success {
|
||||||
|
color: var(--van-blue);
|
||||||
|
}
|
||||||
|
|
||||||
|
.van-icon-cross {
|
||||||
|
color: var(--van-gray-5);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@ -69,6 +69,20 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
<div role="switch"
|
||||||
|
class="van-switch van-switch--on"
|
||||||
|
tabindex="0"
|
||||||
|
aria-checked="true"
|
||||||
|
>
|
||||||
|
<div class="van-switch__node">
|
||||||
|
<div class="icon-wrapper">
|
||||||
|
<i class="van-badge__wrapper van-icon van-icon-success">
|
||||||
|
</i>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div role="switch"
|
<div role="switch"
|
||||||
class="van-switch van-switch--on"
|
class="van-switch van-switch--on"
|
||||||
|
@ -344,7 +344,8 @@ export default {
|
|||||||
|
|
||||||
| Name | Description | SlotProps |
|
| Name | Description | SlotProps |
|
||||||
| --- | --- | --- |
|
| --- | --- | --- |
|
||||||
| default | Custom icon | - |
|
| default | Custom upload area | - |
|
||||||
|
| preview-delete `v.3.5.0` | Custom delete icon | `item: FileListItem` |
|
||||||
| preview-cover | Custom content that covers the image preview | `item: FileListItem` |
|
| preview-cover | Custom content that covers the image preview | `item: FileListItem` |
|
||||||
|
|
||||||
### Parameters of before-read、after-read、before-delete
|
### Parameters of before-read、after-read、before-delete
|
||||||
|
@ -364,8 +364,9 @@ export default {
|
|||||||
### Slots
|
### Slots
|
||||||
|
|
||||||
| 名称 | 说明 | 参数 |
|
| 名称 | 说明 | 参数 |
|
||||||
| ------------- | ------------------------------ | -------------------- |
|
| --- | --- | --- |
|
||||||
| default | 自定义上传区域 | - |
|
| default | 自定义上传区域 | - |
|
||||||
|
| preview-delete `v3.5.0` | 自定义删除按钮 | - |
|
||||||
| preview-cover | 自定义覆盖在预览区域上方的内容 | _item: FileListItem_ |
|
| preview-cover | 自定义覆盖在预览区域上方的内容 | _item: FileListItem_ |
|
||||||
|
|
||||||
### 回调参数
|
### 回调参数
|
||||||
|
@ -280,7 +280,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<UploaderPreviewItem
|
<UploaderPreviewItem
|
||||||
v-slots={{ 'preview-cover': slots['preview-cover'] }}
|
v-slots={pick(slots, ['preview-cover', 'preview-delete'])}
|
||||||
item={item}
|
item={item}
|
||||||
index={index}
|
index={index}
|
||||||
onClick={() => emit('clickPreview', item, getDetail(index))}
|
onClick={() => emit('clickPreview', item, getDetail(index))}
|
||||||
|
@ -73,15 +73,20 @@ export default defineComponent({
|
|||||||
|
|
||||||
const renderDeleteIcon = () => {
|
const renderDeleteIcon = () => {
|
||||||
if (props.deletable && props.item.status !== 'uploading') {
|
if (props.deletable && props.item.status !== 'uploading') {
|
||||||
|
const slot = slots['preview-delete'];
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
role="button"
|
role="button"
|
||||||
class={bem('preview-delete')}
|
class={bem('preview-delete', { shadow: !slot })}
|
||||||
tabindex={0}
|
tabindex={0}
|
||||||
aria-label={t('delete')}
|
aria-label={t('delete')}
|
||||||
onClick={onDelete}
|
onClick={onDelete}
|
||||||
>
|
>
|
||||||
|
{slot ? (
|
||||||
|
slot()
|
||||||
|
) : (
|
||||||
<Icon name="cross" class={bem('preview-delete-icon')} />
|
<Icon name="cross" class={bem('preview-delete-icon')} />
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -106,10 +106,13 @@ body {
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0;
|
top: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
|
|
||||||
|
&--shadow {
|
||||||
width: var(--van-uploader-delete-icon-size);
|
width: var(--van-uploader-delete-icon-size);
|
||||||
height: var(--van-uploader-delete-icon-size);
|
height: var(--van-uploader-delete-icon-size);
|
||||||
background: var(--van-uploader-delete-background);
|
background: var(--van-uploader-delete-background);
|
||||||
border-radius: 0 0 0 12px;
|
border-radius: 0 0 0 12px;
|
||||||
|
}
|
||||||
|
|
||||||
&-icon {
|
&-icon {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
|
@ -30,7 +30,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -50,7 +50,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -126,7 +126,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -160,7 +160,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -195,7 +195,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -258,7 +258,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -294,7 +294,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -359,7 +359,7 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
|
@ -17,7 +17,7 @@ exports[`delete preview image 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -66,7 +66,7 @@ exports[`disable preview image 2`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -116,68 +116,7 @@ exports[`image-fit prop 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
|
||||||
aria-label="Delete"
|
|
||||||
>
|
|
||||||
<i class="van-badge__wrapper van-icon van-icon-cross van-uploader__preview-delete-icon">
|
|
||||||
</i>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="van-uploader__upload">
|
|
||||||
<i class="van-badge__wrapper van-icon van-icon-photograph van-uploader__upload-icon">
|
|
||||||
</i>
|
|
||||||
<input type="file"
|
|
||||||
class="van-uploader__input"
|
|
||||||
accept="image/*"
|
|
||||||
>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`preview-cover slot 1`] = `
|
|
||||||
<div class="van-uploader">
|
|
||||||
<div class="van-uploader__wrapper">
|
|
||||||
<div class="van-uploader__preview">
|
|
||||||
<div class="van-image van-uploader__preview-image">
|
|
||||||
<img src="https://cdn.jsdelivr.net/npm/@vant/assets/cat.jpeg"
|
|
||||||
class="van-image__img"
|
|
||||||
style="object-fit: cover;"
|
|
||||||
>
|
|
||||||
<div class="van-image__loading">
|
|
||||||
<i class="van-badge__wrapper van-icon van-icon-photo van-image__loading-icon">
|
|
||||||
</i>
|
|
||||||
</div>
|
|
||||||
<div class="van-uploader__preview-cover">
|
|
||||||
Custom Preview Cover
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div role="button"
|
|
||||||
class="van-uploader__preview-delete"
|
|
||||||
tabindex="0"
|
|
||||||
aria-label="Delete"
|
|
||||||
>
|
|
||||||
<i class="van-badge__wrapper van-icon van-icon-cross van-uploader__preview-delete-icon">
|
|
||||||
</i>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="van-uploader__preview">
|
|
||||||
<div class="van-image van-uploader__preview-image">
|
|
||||||
<img src="https://cdn.jsdelivr.net/npm/@vant/assets/cat.jpeg"
|
|
||||||
class="van-image__img"
|
|
||||||
style="object-fit: cover;"
|
|
||||||
>
|
|
||||||
<div class="van-image__loading">
|
|
||||||
<i class="van-badge__wrapper van-icon van-icon-photo van-image__loading-icon">
|
|
||||||
</i>
|
|
||||||
</div>
|
|
||||||
<div class="van-uploader__preview-cover">
|
|
||||||
Custom Preview Cover
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div role="button"
|
|
||||||
class="van-uploader__preview-delete"
|
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -212,7 +151,7 @@ exports[`render preview image 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -229,7 +168,7 @@ exports[`render preview image 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -246,7 +185,7 @@ exports[`render preview image 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div role="button"
|
<div role="button"
|
||||||
class="van-uploader__preview-delete"
|
class="van-uploader__preview-delete van-uploader__preview-delete--shadow"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Delete"
|
aria-label="Delete"
|
||||||
>
|
>
|
||||||
@ -277,6 +216,22 @@ exports[`should not render upload input when using readonly prop 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
exports[`should render preview-cover slot correctly 1`] = `
|
||||||
|
<div class="van-uploader__preview-cover">
|
||||||
|
Custom Preview Cover
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`should render preview-delete slot correctly 1`] = `
|
||||||
|
<div role="button"
|
||||||
|
class="van-uploader__preview-delete"
|
||||||
|
tabindex="0"
|
||||||
|
aria-label="Delete"
|
||||||
|
>
|
||||||
|
Custom Preview Delete
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
exports[`upload-icon prop 1`] = `
|
exports[`upload-icon prop 1`] = `
|
||||||
<div class="van-uploader">
|
<div class="van-uploader">
|
||||||
<div class="van-uploader__wrapper">
|
<div class="van-uploader__wrapper">
|
||||||
|
@ -565,7 +565,7 @@ test('closePreview event', async () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await later();
|
await later();
|
||||||
wrapper.find('.van-image').trigger('click');
|
await wrapper.find('.van-image').trigger('click');
|
||||||
|
|
||||||
const preview = document.querySelector<HTMLDivElement>('.van-image-preview');
|
const preview = document.querySelector<HTMLDivElement>('.van-image-preview');
|
||||||
const swipe = preview?.querySelector<HTMLDivElement>(
|
const swipe = preview?.querySelector<HTMLDivElement>(
|
||||||
@ -627,17 +627,32 @@ test('multiFile upload filter max-size file', async () => {
|
|||||||
expect(wrapper.emitted<[File]>('oversize')![0]).toBeTruthy();
|
expect(wrapper.emitted<[File]>('oversize')![0]).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('preview-cover slot', async () => {
|
test('should render preview-cover slot correctly', async () => {
|
||||||
const wrapper = mount(Uploader, {
|
const wrapper = mount(Uploader, {
|
||||||
props: {
|
props: {
|
||||||
modelValue: [{ url: IMAGE }, { url: IMAGE }],
|
modelValue: [{ url: IMAGE }],
|
||||||
},
|
},
|
||||||
slots: {
|
slots: {
|
||||||
'preview-cover': 'Custom Preview Cover',
|
'preview-cover': 'Custom Preview Cover',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
expect(wrapper.html()).toMatchSnapshot();
|
expect(wrapper.find('.van-uploader__preview-cover').html()).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should render preview-delete slot correctly', async () => {
|
||||||
|
const wrapper = mount(Uploader, {
|
||||||
|
props: {
|
||||||
|
modelValue: [{ url: IMAGE }],
|
||||||
|
},
|
||||||
|
slots: {
|
||||||
|
'preview-delete': 'Custom Preview Delete',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(
|
||||||
|
wrapper.find('.van-uploader__preview-delete').html()
|
||||||
|
).toMatchSnapshot();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should not render upload input when using readonly prop', async () => {
|
test('should not render upload input when using readonly prop', async () => {
|
||||||
|
5604
pnpm-lock.yaml
generated
5604
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user