作为标签名时无法渲染的问题,比如下面的写法:
+
+```html
+
+
+
+
+
+
+
diff --git a/src-next/image/index.js b/src-next/image/index.js
new file mode 100644
index 000000000..02fa9334a
--- /dev/null
+++ b/src-next/image/index.js
@@ -0,0 +1,184 @@
+import { createNamespace, isDef, addUnit } from '../utils';
+import Icon from '../icon';
+
+const [createComponent, bem] = createNamespace('image');
+
+export default createComponent({
+ props: {
+ src: String,
+ fit: String,
+ alt: String,
+ round: Boolean,
+ width: [Number, String],
+ height: [Number, String],
+ radius: [Number, String],
+ lazyLoad: Boolean,
+ showError: {
+ type: Boolean,
+ default: true,
+ },
+ showLoading: {
+ type: Boolean,
+ default: true,
+ },
+ errorIcon: {
+ type: String,
+ default: 'warning-o',
+ },
+ loadingIcon: {
+ type: String,
+ default: 'photo-o',
+ },
+ },
+
+ data() {
+ return {
+ loading: true,
+ error: false,
+ };
+ },
+
+ watch: {
+ src() {
+ this.loading = true;
+ this.error = false;
+ },
+ },
+
+ computed: {
+ style() {
+ const style = {};
+
+ if (isDef(this.width)) {
+ style.width = addUnit(this.width);
+ }
+
+ if (isDef(this.height)) {
+ style.height = addUnit(this.height);
+ }
+
+ if (isDef(this.radius)) {
+ style.overflow = 'hidden';
+ style.borderRadius = addUnit(this.radius);
+ }
+
+ return style;
+ },
+ },
+
+ created() {
+ const { $Lazyload } = this;
+
+ if ($Lazyload) {
+ $Lazyload.$on('loaded', this.onLazyLoaded);
+ $Lazyload.$on('error', this.onLazyLoadError);
+ }
+ },
+
+ beforeDestroy() {
+ const { $Lazyload } = this;
+
+ if ($Lazyload) {
+ $Lazyload.$off('loaded', this.onLazyLoaded);
+ $Lazyload.$off('error', this.onLazyLoadError);
+ }
+ },
+
+ methods: {
+ onLoad(event) {
+ this.loading = false;
+ this.$emit('load', event);
+ },
+
+ onLazyLoaded({ el }) {
+ if (el === this.$refs.image && this.loading) {
+ this.onLoad();
+ }
+ },
+
+ onLazyLoadError({ el }) {
+ if (el === this.$refs.image && !this.error) {
+ this.onError();
+ }
+ },
+
+ onError(event) {
+ this.error = true;
+ this.loading = false;
+ this.$emit('error', event);
+ },
+
+ onClick(event) {
+ this.$emit('click', event);
+ },
+
+ genPlaceholder() {
+ if (this.loading && this.showLoading) {
+ return (
+
+ {this.$slots.loading ? (
+ this.$slots.loading()
+ ) : (
+
+ )}
+
+ );
+ }
+
+ if (this.error && this.showError) {
+ return (
+
+ {this.$slots.error ? (
+ this.$slots.error()
+ ) : (
+
+ )}
+
+ );
+ }
+ },
+
+ genImage() {
+ const imgData = {
+ class: bem('img'),
+ attrs: {
+ alt: this.alt,
+ },
+ style: {
+ objectFit: this.fit,
+ },
+ };
+
+ if (this.error) {
+ return;
+ }
+
+ if (this.lazyLoad) {
+ return
;
+ }
+
+ return (
+
+ );
+ },
+ },
+
+ render() {
+ return (
+
+ {this.genImage()}
+ {this.genPlaceholder()}
+ {this.$slots.default?.()}
+
+ );
+ },
+});
diff --git a/src-next/image/index.less b/src-next/image/index.less
new file mode 100644
index 000000000..5f5af3f91
--- /dev/null
+++ b/src-next/image/index.less
@@ -0,0 +1,45 @@
+@import '../style/var';
+
+.van-image {
+ position: relative;
+ display: inline-block;
+
+ &--round {
+ overflow: hidden;
+ border-radius: 50%;
+
+ img {
+ border-radius: inherit;
+ }
+ }
+
+ &__img,
+ &__error,
+ &__loading {
+ display: block;
+ width: 100%;
+ height: 100%;
+ }
+
+ &__error,
+ &__loading {
+ position: absolute;
+ top: 0;
+ left: 0;
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ justify-content: center;
+ color: @image-placeholder-text-color;
+ font-size: @image-placeholder-font-size;
+ background-color: @image-placeholder-background-color;
+ }
+
+ &__loading-icon {
+ font-size: @image-loading-icon-size;
+ }
+
+ &__error-icon {
+ font-size: @image-error-icon-size;
+ }
+}
diff --git a/src-next/image/test/__snapshots__/demo.spec.js.snap b/src-next/image/test/__snapshots__/demo.spec.js.snap
new file mode 100644
index 000000000..2c1671de0
--- /dev/null
+++ b/src-next/image/test/__snapshots__/demo.spec.js.snap
@@ -0,0 +1,129 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`renders demo correctly 1`] = `
+
+
+
+

+
+
+
+
+
+
+
+
+

+
+
+
+
contain
+
+
+

+
+
+
+
cover
+
+
+

+
+
+
+
fill
+
+
+

+
+
+
+
none
+
+
+

+
+
+
+
scale-down
+
+
+
+
+
+
+

+
+
+
+
contain
+
+
+

+
+
+
+
cover
+
+
+

+
+
+
+
fill
+
+
+

+
+
+
+
none
+
+
+

+
+
+
+
scale-down
+
+
+
+
+
+
+
![]()
+
+
+
+
默认提示
+
+
+
![]()
+
+
+
自定义提示
+
+
+
+
+
+
+

+
+
+
+
默认提示
+
+
+

+
+
+
+
自定义提示
+
+
+
+
+`;
diff --git a/src-next/image/test/__snapshots__/index.spec.js.snap b/src-next/image/test/__snapshots__/index.spec.js.snap
new file mode 100644
index 000000000..95dbee240
--- /dev/null
+++ b/src-next/image/test/__snapshots__/index.spec.js.snap
@@ -0,0 +1,58 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`default slot 1`] = `
+
+
+
Custom Default
+
+`;
+
+exports[`error-icon prop 1`] = `
+
+`;
+
+exports[`lazy load 1`] = `
+![]()
+
+
+
+`;
+
+exports[`lazy-load error event 1`] = `
+
+`;
+
+exports[`lazy-load load event 1`] = ``;
+
+exports[`load event 1`] = ``;
+
+exports[`load event 2`] = `
+![]()
+
+
+
+`;
+
+exports[`loading-icon prop 1`] = `
+![]()
+
+
+
+`;
+
+exports[`radius prop 1`] = `
+
+
+
+
+`;
+
+exports[`show-error prop 1`] = ``;
+
+exports[`show-loading prop 1`] = ``;
diff --git a/src-next/image/test/demo.spec.js b/src-next/image/test/demo.spec.js
new file mode 100644
index 000000000..5c70922b5
--- /dev/null
+++ b/src-next/image/test/demo.spec.js
@@ -0,0 +1,4 @@
+import Demo from '../demo';
+import { snapshotDemo } from '../../../test/demo';
+
+snapshotDemo(Demo);
diff --git a/src-next/image/test/index.spec.js b/src-next/image/test/index.spec.js
new file mode 100644
index 000000000..28529a564
--- /dev/null
+++ b/src-next/image/test/index.spec.js
@@ -0,0 +1,172 @@
+import { mount } from '../../../test';
+import VanImage from '..';
+
+test('click event', () => {
+ const wrapper = mount(VanImage);
+
+ wrapper.trigger('click');
+ expect(wrapper.emitted('click')[0][0]).toBeTruthy();
+ wrapper.destroy();
+});
+
+test('load event', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ });
+
+ wrapper.find('img').trigger('load');
+
+ expect(wrapper.emitted('load')[0][0]).toBeTruthy();
+ expect(wrapper).toMatchSnapshot();
+
+ wrapper.setProps({ src: '' });
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('error event', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ });
+
+ wrapper.find('img').trigger('error');
+
+ expect(wrapper.emitted('error')[0][0]).toBeTruthy();
+});
+
+test('lazy load', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ lazyLoad: true,
+ },
+ });
+
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('lazy-load load event', (done) => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ lazyLoad: true,
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ mocks: {
+ $Lazyload: {
+ $on(eventName, hanlder) {
+ if (eventName === 'loaded') {
+ setTimeout(() => {
+ hanlder({ el: null });
+ hanlder({ el: wrapper.find('img').element });
+ expect(wrapper.emitted('load').length).toEqual(1);
+ expect(wrapper).toMatchSnapshot();
+ wrapper.destroy();
+ });
+ }
+ },
+ $off() {
+ done();
+ },
+ },
+ },
+ });
+});
+
+test('lazy-load error event', (done) => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ lazyLoad: true,
+ },
+ mocks: {
+ $Lazyload: {
+ $on(eventName, hanlder) {
+ if (eventName === 'error') {
+ setTimeout(() => {
+ hanlder({ el: null });
+ hanlder({ el: wrapper.find('img').element });
+ expect(wrapper.emitted('error').length).toEqual(1);
+ expect(wrapper).toMatchSnapshot();
+ wrapper.destroy();
+ });
+ }
+ },
+ $off() {
+ done();
+ },
+ },
+ },
+ });
+});
+
+test('show-loading prop', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ showLoading: false,
+ },
+ });
+
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('show-error prop', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ showError: false,
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ });
+
+ wrapper.find('img').trigger('error');
+
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('error-icon prop', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ errorIcon: 'error',
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ });
+
+ wrapper.find('img').trigger('error');
+
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('loading-icon prop', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ loadingIcon: 'success',
+ },
+ });
+
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('radius prop', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ radius: 3,
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ });
+
+ expect(wrapper).toMatchSnapshot();
+});
+
+test('default slot', () => {
+ const wrapper = mount(VanImage, {
+ propsData: {
+ src: 'https://img.yzcdn.cn/vant/cat.jpeg',
+ },
+ scopedSlots: {
+ default: () => 'Custom Default',
+ },
+ });
+
+ expect(wrapper).toMatchSnapshot();
+});
diff --git a/src-next/utils/validate/date.ts b/src-next/utils/validate/date.ts
new file mode 100644
index 000000000..d06fd6696
--- /dev/null
+++ b/src-next/utils/validate/date.ts
@@ -0,0 +1,8 @@
+import { isNaN } from './number';
+
+export function isDate(val: Date): val is Date {
+ return (
+ Object.prototype.toString.call(val) === '[object Date]' &&
+ !isNaN(val.getTime())
+ );
+}
diff --git a/src-next/utils/validate/email.ts b/src-next/utils/validate/email.ts
new file mode 100644
index 000000000..d5e3c21b8
--- /dev/null
+++ b/src-next/utils/validate/email.ts
@@ -0,0 +1,5 @@
+/* eslint-disable */
+export function isEmail(value: string): boolean {
+ const reg = /^((([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))$/i;
+ return reg.test(value);
+}
diff --git a/src-next/utils/validate/mobile.ts b/src-next/utils/validate/mobile.ts
new file mode 100644
index 000000000..b32401981
--- /dev/null
+++ b/src-next/utils/validate/mobile.ts
@@ -0,0 +1,6 @@
+export function isMobile(value: string): boolean {
+ value = value.replace(/[^-|\d]/g, '');
+ return (
+ /^((\+86)|(86))?(1)\d{10}$/.test(value) || /^0[0-9-]{10,13}$/.test(value)
+ );
+}
diff --git a/src-next/utils/validate/number.ts b/src-next/utils/validate/number.ts
new file mode 100644
index 000000000..8d77e0eee
--- /dev/null
+++ b/src-next/utils/validate/number.ts
@@ -0,0 +1,12 @@
+export function isNumeric(val: string): boolean {
+ return /^\d+(\.\d+)?$/.test(val);
+}
+
+export function isNaN(val: number): val is typeof NaN {
+ if (Number.isNaN) {
+ return Number.isNaN(val);
+ }
+
+ // eslint-disable-next-line no-self-compare
+ return val !== val;
+}
diff --git a/src-next/utils/validate/system.ts b/src-next/utils/validate/system.ts
new file mode 100644
index 000000000..d91f2f131
--- /dev/null
+++ b/src-next/utils/validate/system.ts
@@ -0,0 +1,13 @@
+import { isServer } from '..';
+
+export function isAndroid(): boolean {
+ /* istanbul ignore next */
+ return isServer ? false : /android/.test(navigator.userAgent.toLowerCase());
+}
+
+export function isIOS(): boolean {
+ /* istanbul ignore next */
+ return isServer
+ ? false
+ : /ios|iphone|ipad|ipod/.test(navigator.userAgent.toLowerCase());
+}
diff --git a/vant.config.js b/vant.config.js
index 4dacc6028..9f64456b3 100644
--- a/vant.config.js
+++ b/vant.config.js
@@ -90,10 +90,10 @@ module.exports = {
path: 'icon',
title: 'Icon 图标',
},
- // {
- // path: 'image',
- // title: 'Image 图片',
- // },
+ {
+ path: 'image',
+ title: 'Image 图片',
+ },
// {
// path: 'col',
// title: 'Layout 布局',
@@ -437,10 +437,10 @@ module.exports = {
path: 'icon',
title: 'Icon',
},
- // {
- // path: 'image',
- // title: 'Image',
- // },
+ {
+ path: 'image',
+ title: 'Image',
+ },
// {
// path: 'col',
// title: 'Layout',