mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
Merge branch '2.x' into dev
This commit is contained in:
commit
037b6b8c10
@ -5,7 +5,7 @@ import {
|
||||
onDeactivated,
|
||||
onBeforeUnmount,
|
||||
} from 'vue';
|
||||
import { raf, cancelRaf } from '../utils';
|
||||
import { raf, cancelRaf, inBrowser } from '../utils';
|
||||
|
||||
export type CurrentTime = {
|
||||
days: number;
|
||||
@ -106,6 +106,12 @@ export function useCountDown(options: UseCountDownOptions) {
|
||||
};
|
||||
|
||||
const tick = () => {
|
||||
// should not start counting in server
|
||||
// see: https://github.com/youzan/vant/issues/7807
|
||||
if (!inBrowser) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (options.millisecond) {
|
||||
microTick();
|
||||
} else {
|
||||
|
@ -516,8 +516,8 @@ Use [ref](https://v3.vuejs.org/guide/component-template-refs.html) to get Form i
|
||||
| Name | Description | Attribute | Return value |
|
||||
| --- | --- | --- | --- |
|
||||
| submit | Submit form | - | - |
|
||||
| validate | Validate form | _name?: string_ | _Promise_ |
|
||||
| resetValidation | Reset validation | _name?: string_ | - |
|
||||
| validate | Validate form | _name?: string \| string[]_ | _Promise_ |
|
||||
| resetValidation | Reset validation | _name?: string \| string[]_ | - |
|
||||
| scrollToField `v2.8.3` | Scroll to field | _name: string, alignToTop: boolean_ | - |
|
||||
|
||||
### Slots
|
||||
|
@ -555,8 +555,8 @@ export default {
|
||||
| 方法名 | 说明 | 参数 | 返回值 |
|
||||
| --- | --- | --- | --- |
|
||||
| submit | 提交表单,与点击提交按钮的效果等价 | - | - |
|
||||
| validate | 验证表单,支持传入`name`来验证单个表单项 | _name?: string_ | _Promise_ |
|
||||
| resetValidation | 重置表单项的验证提示,支持传入`name`来重置单个表单项 | _name?: string_ | - |
|
||||
| validate | 验证表单,支持传入 `name` 来验证单个或部分表单项 | _name?: string \| string[]_ | _Promise_ |
|
||||
| resetValidation | 重置表单项的验证提示,支持传入 `name` 来重置单个或部分表单项 | _name?: string \| string[]_ | - |
|
||||
| scrollToField `v2.8.3` | 滚动到对应表单项的位置,默认滚动到顶部,第二个参数传 false 可滚动至底部 | _name: string, alignToTop: boolean_ | - |
|
||||
|
||||
### Slots
|
||||
|
@ -37,11 +37,19 @@ export default createComponent({
|
||||
setup(props, { emit, slots }) {
|
||||
const { children, linkChildren } = useChildren(FORM_KEY);
|
||||
|
||||
const validateSeq = () =>
|
||||
const getFieldsByNames = (names) => {
|
||||
if (names) {
|
||||
return children.filter((field) => names.indexOf(field.name) !== -1);
|
||||
}
|
||||
return children;
|
||||
};
|
||||
|
||||
const validateSeq = (names) =>
|
||||
new Promise((resolve, reject) => {
|
||||
const errors = [];
|
||||
const fields = getFieldsByNames(names);
|
||||
|
||||
children
|
||||
fields
|
||||
.reduce(
|
||||
(promise, field) =>
|
||||
promise.then(() => {
|
||||
@ -64,9 +72,10 @@ export default createComponent({
|
||||
});
|
||||
});
|
||||
|
||||
const validateAll = () =>
|
||||
const validateAll = (names) =>
|
||||
new Promise((resolve, reject) => {
|
||||
Promise.all(children.map((item) => item.validate())).then((errors) => {
|
||||
const fields = getFieldsByNames(names);
|
||||
Promise.all(fields.map((item) => item.validate())).then((errors) => {
|
||||
errors = errors.filter((item) => item);
|
||||
|
||||
if (errors.length) {
|
||||
@ -96,17 +105,20 @@ export default createComponent({
|
||||
};
|
||||
|
||||
const validate = (name) => {
|
||||
if (name) {
|
||||
if (name && !Array.isArray(name)) {
|
||||
return validateField(name);
|
||||
}
|
||||
return props.validateFirst ? validateSeq() : validateAll();
|
||||
return props.validateFirst ? validateSeq(name) : validateAll(name);
|
||||
};
|
||||
|
||||
const resetValidation = (name) => {
|
||||
children.forEach((item) => {
|
||||
if (!name || item.name === name) {
|
||||
item.resetValidation();
|
||||
}
|
||||
if (name && !Array.isArray(name)) {
|
||||
name = [name];
|
||||
}
|
||||
|
||||
const fields = getFieldsByNames(name);
|
||||
fields.forEach((item) => {
|
||||
item.resetValidation();
|
||||
});
|
||||
};
|
||||
|
||||
|
@ -37,7 +37,7 @@ test('validate method - validate all fields', (done) => {
|
||||
});
|
||||
});
|
||||
|
||||
test('validate method - validate one field and passed', (done) => {
|
||||
test('validate method - validate one field and failed', (done) => {
|
||||
mountSimpleRulesForm({
|
||||
mounted() {
|
||||
this.$refs.form.validate('A').catch((err) => {
|
||||
@ -48,7 +48,7 @@ test('validate method - validate one field and passed', (done) => {
|
||||
});
|
||||
});
|
||||
|
||||
test('validate method - validate one field and failed', (done) => {
|
||||
test('validate method - validate one field and passed', (done) => {
|
||||
mountForm({
|
||||
template: `
|
||||
<van-form ref="form" @failed="onFailed">
|
||||
@ -64,6 +64,25 @@ test('validate method - validate one field and failed', (done) => {
|
||||
});
|
||||
});
|
||||
|
||||
test('validate method - validate two fields and failed', (done) => {
|
||||
mountForm({
|
||||
template: `
|
||||
<van-form ref="form" @failed="onFailed">
|
||||
<van-field name="A" :rules="rulesA" value="123" />
|
||||
<van-field name="B" :rules="rulesB" value="" />
|
||||
<van-button native-type="submit" />
|
||||
</van-form>
|
||||
`,
|
||||
data: getSimpleRules,
|
||||
mounted() {
|
||||
this.$refs.form.validate(['A', 'B']).catch((error) => {
|
||||
expect(error).toEqual([{ message: 'B failed', name: 'B' }]);
|
||||
done();
|
||||
});
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
test('validate method - unexisted name', (done) => {
|
||||
mountSimpleRulesForm({
|
||||
mounted() {
|
||||
@ -85,6 +104,19 @@ test('resetValidation method - reset all fields', (done) => {
|
||||
});
|
||||
});
|
||||
|
||||
test('resetValidation method - reset two fields', (done) => {
|
||||
const wrapper = mountSimpleRulesForm({
|
||||
mounted() {
|
||||
this.$refs.form.validate().catch(() => {
|
||||
this.$refs.form.resetValidation(['A', 'B']);
|
||||
const errors = wrapper.findAll('.van-field__error-message');
|
||||
expect(errors.length).toEqual(0);
|
||||
done();
|
||||
});
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
test('resetValidation method - reset one field', (done) => {
|
||||
const wrapper = mountSimpleRulesForm({
|
||||
mounted() {
|
||||
|
@ -120,7 +120,7 @@ export default createComponent({
|
||||
};
|
||||
|
||||
const renderPlaceholder = () => {
|
||||
if (loading.value && props.showLoading && inBrowser) {
|
||||
if (loading.value && props.showLoading) {
|
||||
return <div class={bem('loading')}>{renderLoadingIcon()}</div>;
|
||||
}
|
||||
if (error.value && props.showError) {
|
||||
|
@ -125,7 +125,7 @@ test('filter value during user input', () => {
|
||||
input.element.value = '2';
|
||||
input.trigger('input');
|
||||
expect(input.element.value).toEqual('2');
|
||||
expect(wrapper.emitted('input')[1][0]).toEqual('2');
|
||||
expect(wrapper.emitted('input')[1][0]).toEqual(2);
|
||||
});
|
||||
|
||||
test('shoud watch value and format it', () => {
|
||||
@ -149,12 +149,11 @@ test('only allow interger', () => {
|
||||
});
|
||||
|
||||
const input = wrapper.find('input');
|
||||
input.element.value = '1.2';
|
||||
input.element.value = '2.2';
|
||||
input.trigger('input');
|
||||
input.trigger('blur');
|
||||
|
||||
expect(wrapper.emitted('input')[0][0]).toEqual('1');
|
||||
expect(wrapper.emitted('input')[1][0]).toEqual(1);
|
||||
expect(wrapper.emitted('input')[0][0]).toEqual(2);
|
||||
});
|
||||
|
||||
test('input invalid value and blur', () => {
|
||||
@ -246,8 +245,8 @@ test('async-change prop', () => {
|
||||
input.element.value = '3';
|
||||
input.trigger('input');
|
||||
|
||||
expect(wrapper.emitted('input')[1][0]).toEqual('3');
|
||||
expect(wrapper.emitted('change')[1][0]).toEqual('3');
|
||||
expect(wrapper.emitted('input')[1][0]).toEqual(3);
|
||||
expect(wrapper.emitted('change')[1][0]).toEqual(3);
|
||||
});
|
||||
|
||||
test('min value is 0', () => {
|
||||
@ -277,6 +276,16 @@ test('show-plus & show-minus props', () => {
|
||||
expect(wrapper.html()).toMatchSnapshot();
|
||||
});
|
||||
|
||||
test('should hide input when show-input prop is false', () => {
|
||||
const wrapper = mount(Stepper, {
|
||||
propsData: {
|
||||
showInput: false,
|
||||
},
|
||||
});
|
||||
|
||||
expect(wrapper.find('input').element.style.display).toEqual('none');
|
||||
});
|
||||
|
||||
test('decimal-length prop', () => {
|
||||
const wrapper = mount(Stepper, {
|
||||
props: {
|
||||
|
@ -21,7 +21,9 @@ export default createComponent({
|
||||
const style = {};
|
||||
const { vertical } = parent.props;
|
||||
|
||||
style[vertical ? 'height' : 'width'] = `${parent.size.value}px`;
|
||||
if (parent.size.value) {
|
||||
style[vertical ? 'height' : 'width'] = `${parent.size.value}px`;
|
||||
}
|
||||
|
||||
if (state.offset) {
|
||||
style.transform = `translate${vertical ? 'Y' : 'X'}(${state.offset}px)`;
|
||||
|
@ -110,13 +110,17 @@ export default createComponent({
|
||||
const trackStyle = computed(() => {
|
||||
const mainAxis = props.vertical ? 'height' : 'width';
|
||||
const crossAxis = props.vertical ? 'width' : 'height';
|
||||
|
||||
return {
|
||||
[mainAxis]: `${trackSize.value}px`,
|
||||
[crossAxis]: props[crossAxis] ? `${props[crossAxis]}px` : '',
|
||||
const style = {
|
||||
transitionDuration: `${state.swiping ? 0 : props.duration}ms`,
|
||||
transform: `translate${props.vertical ? 'Y' : 'X'}(${state.offset}px)`,
|
||||
};
|
||||
|
||||
if (size.value) {
|
||||
style[mainAxis] = `${trackSize.value}px`;
|
||||
style[crossAxis] = props[crossAxis] ? `${props[crossAxis]}px` : '';
|
||||
}
|
||||
|
||||
return style;
|
||||
});
|
||||
|
||||
const getTargetActive = (pace) => {
|
||||
|
4
types/form.d.ts
vendored
4
types/form.d.ts
vendored
@ -3,9 +3,9 @@ import { VanComponent } from './component';
|
||||
export class Form extends VanComponent {
|
||||
submit(): void;
|
||||
|
||||
validate(name?: string): Promise<void>;
|
||||
validate(name?: string | string[]): Promise<void>;
|
||||
|
||||
resetValidation(name?: string): void;
|
||||
resetValidation(name?: string | string[]): void;
|
||||
|
||||
scrollToField(name: string, options?: boolean | ScrollIntoViewOptions): void;
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user