Merge branch 'dev' into next

This commit is contained in:
chenjiahan 2022-08-19 22:14:47 +08:00
commit 292ac6b55e
24 changed files with 167 additions and 52 deletions

View File

@ -4,6 +4,8 @@ on:
push: push:
branches: [dev, 2.x, gh-pages] branches: [dev, 2.x, gh-pages]
workflow_dispatch:
jobs: jobs:
build: build:
runs-on: ubuntu-latest runs-on: ubuntu-latest

View File

@ -1,6 +1,15 @@
name: CI name: CI
on: [push] on:
push:
branches:
- '**'
pull_request:
branches:
- dev
workflow_dispatch:
jobs: jobs:
lint: lint:

View File

@ -172,6 +172,8 @@ module.exports = {
}; };
``` ```
> Tips: 在配置 postcss-pxtorem 时,同样应避免 ignore node_modules 目录,否则会导致 Vant 样式无法被编译。
#### 其他设计稿尺寸 #### 其他设计稿尺寸
如果设计稿的尺寸不是 375而是 750 或其他大小,可以将 `rootValue` 配置调整为: 如果设计稿的尺寸不是 375而是 750 或其他大小,可以将 `rootValue` 配置调整为:

View File

@ -48,7 +48,11 @@ export default defineComponent({
return true; return true;
} }
const { content, showZero } = props; const { content, showZero } = props;
return isDef(content) && content !== '' && (showZero || content !== 0); return (
isDef(content) &&
content !== '' &&
(showZero || (content !== 0 && content !== '0'))
);
}; };
const renderContent = () => { const renderContent = () => {

View File

@ -148,7 +148,7 @@ app.use(Badge);
| dot | 是否展示为小红点 | _boolean_ | `false` | | dot | 是否展示为小红点 | _boolean_ | `false` |
| max | 最大值,超过最大值会显示 `{max}+`,仅当 content 为数字时有效 | _number \| string_ | - | | max | 最大值,超过最大值会显示 `{max}+`,仅当 content 为数字时有效 | _number \| string_ | - |
| offset `v3.0.5` | 设置徽标的偏移量,数组的两项分别对应水平和垂直方向的偏移量,默认单位为 `px` | _[number \| string, number \| string]_ | - | | offset `v3.0.5` | 设置徽标的偏移量,数组的两项分别对应水平和垂直方向的偏移量,默认单位为 `px` | _[number \| string, number \| string]_ | - |
| show-zero `v3.0.10` | 当 content 为数字 0 时,是否展示徽标 | _boolean_ | `true` | | show-zero `v3.0.10` | 当 content 为数字 0 或字符串 '0' 时,是否展示徽标 | _boolean_ | `true` |
| position `v3.2.7` | 徽标位置,可选值为 `top-left` `bottom-left` `bottom-right` | _string_ | `top-right` | | position `v3.2.7` | 徽标位置,可选值为 `top-left` `bottom-left` `bottom-right` | _string_ | `top-right` |
### Slots ### Slots
@ -184,17 +184,3 @@ import type { BadgeProps, BadgePosition } from 'vant';
| --van-badge-dot-color | _var(--van-danger-color)_ | - | | --van-badge-dot-color | _var(--van-danger-color)_ | - |
| --van-badge-dot-size | _8px_ | - | | --van-badge-dot-size | _8px_ | - |
| --van-badge-font | _-apple-system-font, Helvetica Neue, Arial, sans-serif_ | - | | --van-badge-font | _-apple-system-font, Helvetica Neue, Arial, sans-serif_ | - |
## 常见问题
### 设置 show-zero 属性为 false 不生效?
注意 `show-zero` 属性仅对数字类型的 `0` 有效,对字符串类型的 `'0'` 无效。
```html
<!-- 正确写法,不显示 0 -->
<van-badge :content="0" :show-zero="false" />
<!-- 错误写法,显示 0 -->
<van-badge content="0" :show-zero="false" />
```

View File

@ -185,10 +185,6 @@ export default defineComponent({
const months: Date[] = []; const months: Date[] = [];
const cursor = new Date(props.minDate); const cursor = new Date(props.minDate);
if (props.lazyRender && !props.show && props.poppable) {
return months;
}
cursor.setDate(1); cursor.setDate(1);
do { do {
@ -299,7 +295,9 @@ export default defineComponent({
props.type === 'single' props.type === 'single'
? (currentDate.value as Date) ? (currentDate.value as Date)
: (currentDate.value as Date[])[0]; : (currentDate.value as Date[])[0];
scrollToDate(targetDate); if (isDate(targetDate)) {
scrollToDate(targetDate);
}
} else { } else {
raf(onScroll); raf(onScroll);
} }

View File

@ -0,0 +1,24 @@
/**
* The z-index of Popup components.
* Will affect this components:
* - ActionSheet
* - Calendar
* - Dialog
* - DropdownItem
* - ImagePreview
* - Notify
* - Popup
* - Popover
* - ShareSheet
* - Toast
*/
let globalZIndex = 2000;
/** the global z-index is automatically incremented after reading */
export const useGlobalZIndex = () => ++globalZIndex;
/** reset the global z-index */
export const setGlobalZIndex = (val: number) => {
globalZIndex = val;
};

View File

@ -1,5 +1,6 @@
import { useRect } from '@vant/use'; import { useRect } from '@vant/use';
import { Ref, ref, onMounted, nextTick } from 'vue'; import { Ref, ref, onMounted, nextTick } from 'vue';
import { onPopupReopen } from './on-popup-reopen';
export const useHeight = ( export const useHeight = (
element: Element | Ref<Element | undefined>, element: Element | Ref<Element | undefined>,
@ -25,5 +26,11 @@ export const useHeight = (
} }
}); });
// The result of useHeight might be 0 when the popup is hidden,
// so we need to reset the height when the popup is reopened.
// IntersectionObserver is a better solution, but it is not supported by legacy browsers.
// https://github.com/vant-ui/vant/issues/10628
onPopupReopen(() => nextTick(setHeight));
return height; return height;
}; };

View File

@ -18,6 +18,7 @@ import {
createNamespace, createNamespace,
type Numeric, type Numeric,
} from '../utils'; } from '../utils';
import { setGlobalZIndex } from '../composables/use-global-z-index';
const [name, bem] = createNamespace('config-provider'); const [name, bem] = createNamespace('config-provider');
@ -33,6 +34,7 @@ export const CONFIG_PROVIDER_KEY: InjectionKey<ConfigProviderProvide> =
const configProviderProps = { const configProviderProps = {
tag: makeStringProp<keyof HTMLElementTagNameMap>('div'), tag: makeStringProp<keyof HTMLElementTagNameMap>('div'),
theme: makeStringProp<ConfigProviderTheme>('light'), theme: makeStringProp<ConfigProviderTheme>('light'),
zIndex: Number,
themeVars: Object as PropType<Record<string, Numeric>>, themeVars: Object as PropType<Record<string, Numeric>>,
iconPrefix: String, iconPrefix: String,
}; };
@ -85,6 +87,12 @@ export default defineComponent({
provide(CONFIG_PROVIDER_KEY, props); provide(CONFIG_PROVIDER_KEY, props);
watchEffect(() => {
if (props.zIndex !== undefined) {
setGlobalZIndex(props.zIndex);
}
});
return () => ( return () => (
<props.tag class={bem()} style={style.value}> <props.tag class={bem()} style={style.value}>
{slots.default?.()} {slots.default?.()}

View File

@ -249,6 +249,7 @@ There are all **Basic Variables** below, for component CSS Variables, please ref
| --- | --- | --- | --- | | --- | --- | --- | --- |
| theme | Theme mode, can be set to `dark` | _ConfigProviderTheme_ | `light` | | theme | Theme mode, can be set to `dark` | _ConfigProviderTheme_ | `light` |
| theme-vars | Theme variables | _object_ | - | | theme-vars | Theme variables | _object_ | - |
| z-index `v3.6.0` | Set the z-index of all popup components, this property takes effect globally | _number_ | `2000` |
| tag `v3.1.2` | HTML Tag of root element | _string_ | `div` | | tag `v3.1.2` | HTML Tag of root element | _string_ | `div` |
| icon-prefix `v3.1.3` | Icon className prefix | _string_ | `van-icon` | | icon-prefix `v3.1.3` | Icon className prefix | _string_ | `van-icon` |

View File

@ -254,6 +254,7 @@ Vant 中的 CSS 变量分为 **基础变量** 和 **组件变量**。组件变
| theme | 主题风格,设置为 `dark` 来开启深色模式,全局生效 | _ConfigProviderTheme_ | `light` | | theme | 主题风格,设置为 `dark` 来开启深色模式,全局生效 | _ConfigProviderTheme_ | `light` |
| theme-vars | 自定义主题变量,局部生效 | _object_ | - | | theme-vars | 自定义主题变量,局部生效 | _object_ | - |
| tag `v3.1.2` | 根节点对应的 HTML 标签名 | _string_ | `div` | | tag `v3.1.2` | 根节点对应的 HTML 标签名 | _string_ | `div` |
| z-index `v3.6.0` | 设置所有弹窗类组件的 z-index该属性对全局生效 | _number_ | `2000` |
| icon-prefix `v3.1.3` | 所有图标的类名前缀,等同于 Icon 组件的 [class-prefix 属性](#/zh-CN/icon#props) | _string_ | `van-icon` | | icon-prefix `v3.1.3` | 所有图标的类名前缀,等同于 Icon 组件的 [class-prefix 属性](#/zh-CN/icon#props) | _string_ | `van-icon` |
### 类型定义 ### 类型定义

View File

@ -1,6 +1,8 @@
import { ref } from 'vue';
import { ConfigProvider } from '..'; import { ConfigProvider } from '..';
import { Icon } from '../../icon'; import { Icon } from '../../icon';
import { mount } from '../../../test'; import { later, mount } from '../../../test';
import Popup from '../../popup';
test('should render tag prop correctly', () => { test('should render tag prop correctly', () => {
const wrapper = mount(ConfigProvider, { const wrapper = mount(ConfigProvider, {
@ -23,3 +25,19 @@ test('should change icon class-prefix when using icon-prefix prop', () => {
}); });
expect(wrapper.html()).toMatchSnapshot(); expect(wrapper.html()).toMatchSnapshot();
}); });
test('should change global z-index when using z-index prop', async () => {
const show = ref(true);
const wrapper = mount({
render() {
return (
<ConfigProvider zIndex={0}>
<Popup v-model:show={show.value} />
</ConfigProvider>
);
},
});
await later();
expect(wrapper.find('.van-popup').style.zIndex).toEqual('1');
});

View File

@ -32,6 +32,7 @@ import {
runSyncRule, runSyncRule,
endComposing, endComposing,
mapInputType, mapInputType,
isEmptyValue,
startComposing, startComposing,
getRuleMessage, getRuleMessage,
resizeTextarea, resizeTextarea,
@ -201,6 +202,10 @@ export default defineComponent({
} }
if (rule.validator) { if (rule.validator) {
if (isEmptyValue(value) && rule.validateEmpty === false) {
return;
}
return runRuleValidator(value, rule).then((result) => { return runRuleValidator(value, rule).then((result) => {
if (result && typeof result === 'string') { if (result && typeof result === 'string') {
state.status = 'failed'; state.status = 'failed';

View File

@ -64,6 +64,7 @@ export type FieldRule = {
required?: boolean; required?: boolean;
validator?: FieldRuleValidator; validator?: FieldRuleValidator;
formatter?: FiledRuleFormatter; formatter?: FiledRuleFormatter;
validateEmpty?: boolean;
}; };
export type FieldValidationStatus = 'passed' | 'failed' | 'unvalidated'; export type FieldValidationStatus = 'passed' | 'failed' | 'unvalidated';

View File

@ -8,7 +8,7 @@ import {
} from '../utils'; } from '../utils';
import type { FieldRule, FieldType, FieldAutosizeConfig } from './types'; import type { FieldRule, FieldType, FieldAutosizeConfig } from './types';
function isEmptyValue(value: unknown) { export function isEmptyValue(value: unknown) {
if (Array.isArray(value)) { if (Array.isArray(value)) {
return !value.length; return !value.length;
} }
@ -19,8 +19,13 @@ function isEmptyValue(value: unknown) {
} }
export function runSyncRule(value: unknown, rule: FieldRule) { export function runSyncRule(value: unknown, rule: FieldRule) {
if (rule.required && isEmptyValue(value)) { if (isEmptyValue(value)) {
return false; if (rule.required) {
return false;
}
if (rule.validateEmpty === false) {
return true;
}
} }
if (rule.pattern && !rule.pattern.test(String(value))) { if (rule.pattern && !rule.pattern.test(String(value))) {
return false; return false;

View File

@ -509,12 +509,13 @@ export default {
| Key | Description | Type | | Key | Description | Type |
| --- | --- | --- | | --- | --- | --- |
| required | Whether to be a required field, the value is not allowed to be empty string, empty array, `false`, `undefined`, `null` | _boolean_ | | required | Whether to be a required field, the value is not allowed to be empty (empty string, empty array, `false`, `undefined`, `null`) | _boolean_ |
| message | Error message | _string \| (value, rule) => string_ | | message | Error message, can be a function to dynamically return message content | _string \| (value, rule) => string_ |
| validator | Custom validator | _(value, rule) => boolean \| string \| Promise_ | | validator | Custom validator, can return a Promise to validate dynamically | _(value, rule) => boolean \| string \| Promise_ |
| pattern | Regex pattern | _RegExp_ | | pattern | Regexp pattern, if the regexp cannot match, means that the validation fails | _RegExp_ |
| trigger | When to validate the form, can be set to `onChange``onBlur` | _string_ | | trigger | When to validate the form, priority is higher than the `validate-trigger` of the Form component, can be set to `onChange`, `onBlur`, `onSubmit` | _string \| string[]_ |
| formatter | Format value before validate | _(value, rule) => any_ | | formatter | Format value before validate | _(value, rule) => any_ |
| validateEmpty `v3.6.0` | Controls whether the `validator` and `pattern` options to verify empty values, the default value is `true`, you can set to `false` to disable this behavior | _boolean_ |
### validate-trigger ### validate-trigger

View File

@ -541,16 +541,17 @@ export default {
### Rule 数据结构 ### Rule 数据结构
使用 Field 的`rules`属性可以定义校验规则,可选属性如下: 使用 Field 的 `rules` 属性可以定义校验规则,可选属性如下:
| 键名 | 说明 | 类型 | | 键名 | 说明 | 类型 |
| --- | --- | --- | | --- | --- | --- |
| required | 是否为必选字段,当值为空字符串、空数组、`false``undefined``null` ,校验不通过 | _boolean_ | | required | 是否为必选字段,当值为空值时(空字符串、空数组、`false``undefined``null` ,校验不通过 | _boolean_ |
| message | 错误提示文案 | _string \| (value, rule) => string_ | | message | 错误提示文案,可以设置为一个函数来返回动态的文案内容 | _string \| (value, rule) => string_ |
| validator | 通过函数进行校验 | _(value, rule) => boolean \| string \| Promise_ | | validator | 通过函数进行校验,可以返回一个 Promise 来进行异步校验 | _(value, rule) => boolean \| string \| Promise_ |
| pattern | 通过正则表达式进行校验 | _RegExp_ | | pattern | 通过正则表达式进行校验,正则无法匹配表示校验不通过 | _RegExp_ |
| trigger | 本项规则的触发时机,可选值为 `onChange``onBlur` | _string_ | | trigger | 设置本项规则的触发时机,优先级高于 Form 组件设置的 `validate-trigger` 属性,可选值为 `onChange``onBlur``onSubmit` | _string \| string[]_ |
| formatter | 格式化函数,将表单项的值转换后进行校验 | _(value, rule) => any_ | | formatter | 格式化函数,将表单项的值转换后进行校验 | _(value, rule) => any_ |
| validateEmpty `v3.6.0` | 设置 `validator``pattern` 是否要对空值进行校验,默认值为 `true`,可以设置为 `false` 来禁用该行为 | _boolean_ |
### validate-trigger 可选值 ### validate-trigger 可选值

View File

@ -70,6 +70,40 @@ test('should support message function in rules prop', async () => {
}); });
}); });
test('should skip pattern if validateEmpty is false in rules prop', async () => {
const onFailed = jest.fn();
const rules: FieldRule[] = [{ pattern: /\d{6}/, validateEmpty: false }];
const wrapper = mount({
render() {
return (
<Form onFailed={onFailed}>
<Field name="A" rules={rules} modelValue="" />
</Form>
);
},
});
await submitForm(wrapper);
expect(onFailed).toHaveBeenCalledTimes(0);
});
test('should skip validator if validateEmpty is false in rules prop', async () => {
const onFailed = jest.fn();
const rules: FieldRule[] = [{ validator: () => false, validateEmpty: false }];
const wrapper = mount({
render() {
return (
<Form onFailed={onFailed}>
<Field name="A" rules={rules} modelValue="" />
</Form>
);
},
});
await submitForm(wrapper);
expect(onFailed).toHaveBeenCalledTimes(0);
});
test('should support formatter in rules prop', async () => { test('should support formatter in rules prop', async () => {
const onFailed = jest.fn(); const onFailed = jest.fn();
const rules: FieldRule[] = [ const rules: FieldRule[] = [

View File

@ -31,6 +31,7 @@ import { useExpose } from '../composables/use-expose';
import { useLockScroll } from '../composables/use-lock-scroll'; import { useLockScroll } from '../composables/use-lock-scroll';
import { useLazyRender } from '../composables/use-lazy-render'; import { useLazyRender } from '../composables/use-lazy-render';
import { POPUP_TOGGLE_KEY } from '../composables/on-popup-reopen'; import { POPUP_TOGGLE_KEY } from '../composables/on-popup-reopen';
import { useGlobalZIndex } from '../composables/use-global-z-index';
// Components // Components
import { Icon } from '../icon'; import { Icon } from '../icon';
@ -56,8 +57,6 @@ export type PopupProps = ExtractPropTypes<typeof popupProps>;
const [name, bem] = createNamespace('popup'); const [name, bem] = createNamespace('popup');
let globalZIndex = 2000;
export default defineComponent({ export default defineComponent({
name, name,
@ -103,12 +102,10 @@ export default defineComponent({
const open = () => { const open = () => {
if (!opened) { if (!opened) {
if (props.zIndex !== undefined) {
globalZIndex = +props.zIndex;
}
opened = true; opened = true;
zIndex.value = ++globalZIndex;
zIndex.value =
props.zIndex !== undefined ? +props.zIndex : useGlobalZIndex();
emit('open'); emit('open');
} }

View File

@ -29,8 +29,8 @@ test('should change z-index when using z-index prop', async () => {
}); });
await nextTick(); await nextTick();
expect(wrapper.find('.van-popup').style.zIndex).toEqual('11'); expect(wrapper.find('.van-popup').style.zIndex).toEqual('10');
expect(wrapper.find('.van-overlay').style.zIndex).toEqual('11'); expect(wrapper.find('.van-overlay').style.zIndex).toEqual('10');
}); });
test('should lock scroll when showed', async () => { test('should lock scroll when showed', async () => {

View File

@ -17,7 +17,7 @@ import {
} from '../utils'; } from '../utils';
// Composables // Composables
import { useScrollParent } from '@vant/use'; import { useEventListener, useScrollParent } from '@vant/use';
import { useTouch } from '../composables/use-touch'; import { useTouch } from '../composables/use-touch';
// Components // Components
@ -61,6 +61,7 @@ export default defineComponent({
let reachTop: boolean; let reachTop: boolean;
const root = ref<HTMLElement>(); const root = ref<HTMLElement>();
const track = ref<HTMLElement>();
const scrollParent = useScrollParent(root); const scrollParent = useScrollParent(root);
const state = reactive({ const state = reactive({
@ -220,6 +221,15 @@ export default defineComponent({
} }
); );
// add passive option to avoid Chrome warning
useEventListener('touchstart', onTouchStart as EventListener, {
target: track,
passive: true,
});
useEventListener('touchmove', onTouchMove as EventListener, {
target: track,
});
return () => { return () => {
const trackStyle = { const trackStyle = {
transitionDuration: `${state.duration}ms`, transitionDuration: `${state.duration}ms`,
@ -231,10 +241,9 @@ export default defineComponent({
return ( return (
<div ref={root} class={bem()}> <div ref={root} class={bem()}>
<div <div
ref={track}
class={bem('track')} class={bem('track')}
style={trackStyle} style={trackStyle}
onTouchstart={onTouchStart}
onTouchmove={onTouchMove}
onTouchend={onTouchEnd} onTouchend={onTouchEnd}
onTouchcancel={onTouchEnd} onTouchcancel={onTouchEnd}
> >

View File

@ -149,7 +149,7 @@ export default {
| clear-trigger | When to display the clear icon, `always` means to display the icon when value is not empty, `focus` means to display the icon when input is focused | _string_ | `focus` | | clear-trigger | When to display the clear icon, `always` means to display the icon when value is not empty, `focus` means to display the icon when input is focused | _string_ | `focus` |
| autofocus | Whether to auto focus, unsupported in iOS | _boolean_ | `false` | | autofocus | Whether to auto focus, unsupported in iOS | _boolean_ | `false` |
| show-action | Whether to show right action button | _boolean_ | `false` | | show-action | Whether to show right action button | _boolean_ | `false` |
| action-text | Text of action button | _boolean_ | `Cancel` | | action-text | Text of action button | _string_ | `Cancel` |
| disabled | Whether to disable field | _boolean_ | `false` | | disabled | Whether to disable field | _boolean_ | `false` |
| readonly | Whether to be readonly | _boolean_ | `false` | | readonly | Whether to be readonly | _boolean_ | `false` |
| error | Whether to mark the input content in red | _boolean_ | `false` | | error | Whether to mark the input content in red | _boolean_ | `false` |

View File

@ -161,7 +161,7 @@ export default {
| clear-trigger | 显示清除图标的时机,`always` 表示输入框不为空时展示,<br>`focus` 表示输入框聚焦且不为空时展示 | _string_ | `focus` | | clear-trigger | 显示清除图标的时机,`always` 表示输入框不为空时展示,<br>`focus` 表示输入框聚焦且不为空时展示 | _string_ | `focus` |
| autofocus | 是否自动聚焦iOS 系统不支持该属性 | _boolean_ | `false` | | autofocus | 是否自动聚焦iOS 系统不支持该属性 | _boolean_ | `false` |
| show-action | 是否在搜索框右侧显示取消按钮 | _boolean_ | `false` | | show-action | 是否在搜索框右侧显示取消按钮 | _boolean_ | `false` |
| action-text | 取消按钮文字 | _boolean_ | `取消` | | action-text | 取消按钮文字 | _string_ | `取消` |
| disabled | 是否禁用输入框 | _boolean_ | `false` | | disabled | 是否禁用输入框 | _boolean_ | `false` |
| readonly | 是否将输入框设为只读状态,只读状态下无法输入内容 | _boolean_ | `false` | | readonly | 是否将输入框设为只读状态,只读状态下无法输入内容 | _boolean_ | `false` |
| error | 是否将输入内容标红 | _boolean_ | `false` | | error | 是否将输入内容标红 | _boolean_ | `false` |

View File

@ -40,7 +40,9 @@ body {
&__field { &__field {
flex: 1; flex: 1;
padding: 5px var(--van-padding-xs) 5px 0; align-items: center;
padding: 0 var(--van-padding-xs) 0 0;
height: var(--van-search-input-height);
background-color: transparent; background-color: transparent;
.van-field__left-icon { .van-field__left-icon {