[new feature] Toast: add icon prop (#3485)

This commit is contained in:
neverland 2019-06-12 17:21:59 +08:00 committed by GitHub
parent 2d18f67a7d
commit 153fb9d69f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 185 additions and 48 deletions

View File

@ -1,5 +1,5 @@
<p align="center"> <p align="center">
<img alt="logo" src="https://img.yzcdn.cn/public_files/2017/12/18/fd78cf6bb5d12e2a119d0576bedfd230.png" width="120" height="120" style="margin-bottom: 10px;"> <img alt="logo" src="https://img.yzcdn.cn/vant/logo.png" width="120" height="120" style="margin-bottom: 10px;">
</p> </p>
<h3 align="center" style="margin: 30px 0 35px;">Mobile UI Components built on Vue</h3> <h3 align="center" style="margin: 30px 0 35px;">Mobile UI Components built on Vue</h3>

View File

@ -1,5 +1,5 @@
<p align="center"> <p align="center">
<img alt="logo" src="https://img.yzcdn.cn/public_files/2017/12/18/fd78cf6bb5d12e2a119d0576bedfd230.png" width="120" style="margin-bottom: 10px;"> <img alt="logo" src="https://img.yzcdn.cn/vant/logo.png" width="120" style="margin-bottom: 10px;">
</p> </p>
<h3 align="center" style="margin: 30px 0 35px;">轻量、可靠的移动端 Vue 组件库</h3> <h3 align="center" style="margin: 30px 0 35px;">轻量、可靠的移动端 Vue 组件库</h3>

View File

@ -16,7 +16,7 @@
</div> </div>
<h1 class="vant-title"> <h1 class="vant-title">
<img src="https://img.yzcdn.cn/public_files/2017/12/18/fd78cf6bb5d12e2a119d0576bedfd230.png" > <img src="https://img.yzcdn.cn/vant/logo.png" >
<span>Vant</span> <span>Vant</span>
</h1> </h1>
<h2 class="vant-desc">{{ description }}</h2> <h2 class="vant-desc">{{ description }}</h2>

View File

@ -15,7 +15,7 @@ export default {
'zh-CN': { 'zh-CN': {
header: { header: {
logo: { logo: {
image: 'https://img.yzcdn.cn/public_files/2017/12/18/fd78cf6bb5d12e2a119d0576bedfd230.png', image: 'https://img.yzcdn.cn/vant/logo.png',
title: 'Vant', title: 'Vant',
href: '#/' href: '#/'
}, },
@ -349,7 +349,7 @@ export default {
header: { header: {
logo: { logo: {
image: image:
'https://img.yzcdn.cn/public_files/2017/12/18/fd78cf6bb5d12e2a119d0576bedfd230.png', 'https://img.yzcdn.cn/vant/logo.png',
title: 'Vant', title: 'Vant',
href: '#/' href: '#/'
}, },

View File

@ -552,7 +552,7 @@
@toast-line-height: 20px; @toast-line-height: 20px;
@toast-border-radius: 4px; @toast-border-radius: 4px;
@toast-background-color: rgba(@text-color, .88); @toast-background-color: rgba(@text-color, .88);
@toast-icon-size: 48px; @toast-icon-size: 40px;
@toast-text-min-width: 96px; @toast-text-min-width: 96px;
@toast-text-padding: 8px 12px; @toast-text-padding: 8px 12px;
@toast-default-padding: 15px; @toast-default-padding: 15px;

View File

@ -4,12 +4,12 @@ import Icon from '../icon';
import Loading from '../loading'; import Loading from '../loading';
const [sfc, bem] = use('toast'); const [sfc, bem] = use('toast');
const STYLE = ['success', 'fail', 'loading'];
export default sfc({ export default sfc({
mixins: [PopupMixin], mixins: [PopupMixin],
props: { props: {
icon: String,
className: null, className: null,
loadingType: String, loadingType: String,
forbidClick: Boolean, forbidClick: Boolean,
@ -62,6 +62,7 @@ export default sfc({
} }
}, },
/* istanbul ignore next */
onAfterEnter() { onAfterEnter() {
this.$emit('opened'); this.$emit('opened');
@ -72,31 +73,43 @@ export default sfc({
}, },
render(h) { render(h) {
const { type, message, loadingType } = this; const { type, icon, message, loadingType } = this;
const style = STYLE.indexOf(type) !== -1 ? 'default' : type;
function Content() { const hasIcon = icon || (type === 'success' || type === 'fail');
switch (style) {
case 'text': function ToastIcon() {
return <div>{message}</div>; if (hasIcon) {
case 'html': return <Icon class={bem('icon')} name={icon || type} />;
return <div domPropsInnerHTML={message} />;
} }
return [ if (type === 'loading') {
type === 'loading' ? ( return <Loading class={bem('loading')} color="white" type={loadingType} />;
<Loading color="white" type={loadingType} /> }
) : ( }
<Icon class={bem('icon')} name={type} />
), function Message() {
isDef(message) && <div class={bem('text')}>{message}</div> if (!isDef(message) || message === '') {
]; return;
}
if (type === 'html') {
return <div class={bem('text')} domPropsInnerHTML={message} />;
}
return <div class={bem('text')}>{message}</div>;
} }
return ( return (
<transition name="van-fade" onAfterEnter={this.onAfterEnter}> <transition name="van-fade" onAfterEnter={this.onAfterEnter}>
<div vShow={this.value} class={[bem([style, this.position]), this.className]}> <div
{Content()} vShow={this.value}
class={[
bem([this.position, { text: !hasIcon && type !== 'loading' }]),
this.className
]}
>
{ToastIcon()}
{Message()}
</div> </div>
</transition> </transition>
); );

View File

@ -39,6 +39,22 @@
</van-button> </van-button>
</demo-block> </demo-block>
<demo-block :title="$t('customIcon')">
<van-button
type="primary"
@click="showIconToast"
>
{{ $t('customIcon') }}
</van-button>
<van-button
type="primary"
@click="showImageToast"
>
{{ $t('customImage') }}
</van-button>
</demo-block>
<demo-block :title="$t('advancedUsage')"> <demo-block :title="$t('advancedUsage')">
<van-button <van-button
type="primary" type="primary"
@ -63,6 +79,8 @@ export default {
longText: '这是一条长文字提示,超过一定字数就会换行', longText: '这是一条长文字提示,超过一定字数就会换行',
text2: '成功文案', text2: '成功文案',
text3: '失败文案', text3: '失败文案',
customIcon: '自定义图标',
customImage: '展示图片',
text4: second => `倒计时 ${second}`, text4: second => `倒计时 ${second}`,
longTextButton: '长文字提示' longTextButton: '长文字提示'
}, },
@ -76,6 +94,8 @@ export default {
longText: 'This is a long message, text will wrap when over a certain length', longText: 'This is a long message, text will wrap when over a certain length',
text2: 'Success', text2: 'Success',
text3: 'Fail', text3: 'Fail',
customIcon: 'Custom Icon',
customImage: 'Custom Image',
text4: second => `${second} seconds`, text4: second => `${second} seconds`,
longTextButton: 'Long Text' longTextButton: 'Long Text'
} }
@ -94,6 +114,20 @@ export default {
this.$toast.fail(this.$t('text3')); this.$toast.fail(this.$t('text3'));
}, },
showIconToast() {
this.$toast({
message: this.$t('customIcon'),
icon: 'like-o'
});
},
showImageToast() {
this.$toast({
message: this.$t('customImage'),
icon: 'https://img.yzcdn.cn/vant/logo.png'
});
},
showCustomizedToast() { showCustomizedToast() {
const toast = this.$toast.loading({ const toast = this.$toast.loading({
duration: 0, duration: 0,

View File

@ -32,6 +32,20 @@ Toast.success('Success');
Toast.fail('Fail'); Toast.fail('Fail');
``` ```
### Custom Icon
```js
Toast({
text: 'Custom Icon',
icon: 'like-o'
});
Toast({
text: 'Custom Image',
icon: 'https://img.yzcdn.cn/vant/logo.png'
});
```
### Advanced Usage ### Advanced Usage
```javascript ```javascript
@ -102,6 +116,7 @@ toast2.clear();
| type | Can be set to `loading` `success` `fail` `html` | `String` | `text` | | type | Can be set to `loading` `success` `fail` `html` | `String` | `text` |
| position | Can be set to `top` `middle` `bottom` | `String` | `middle` | | position | Can be set to `top` `middle` `bottom` | `String` | `middle` |
| message | Message | `String` | `''` | | message | Message | `String` | `''` |
| icon | Custom icon | `String` | - |
| mask | Whether to show mask | `Boolean` | `false` | | mask | Whether to show mask | `Boolean` | `false` |
| forbidClick | Whether to forbid click background | `Boolean` | `false` | | forbidClick | Whether to forbid click background | `Boolean` | `false` |
| loadingType | Loading icon type, can be set to `spinner` | `String` | `circular` | | loadingType | Loading icon type, can be set to `spinner` | `String` | `circular` |

View File

@ -3,6 +3,7 @@ import VueToast from './Toast';
import { isObj, isServer } from '../utils'; import { isObj, isServer } from '../utils';
const defaultOptions = { const defaultOptions = {
icon: '',
type: 'text', type: 'text',
mask: false, mask: false,
value: true, value: true,

View File

@ -11,8 +11,10 @@
box-sizing: content-box; box-sizing: content-box;
// hack for avoid max-width when use left & fixed // hack for avoid max-width when use left & fixed
width: fit-content; width: @toast-default-width;
max-width: @toast-max-width; max-width: @toast-max-width;
min-height: @toast-default-min-height;
padding: @toast-default-padding;
color: @toast-text-color; color: @toast-text-color;
font-size: @toast-font-size; font-size: @toast-font-size;
line-height: @toast-line-height; line-height: @toast-line-height;
@ -34,25 +36,13 @@
} }
&--text { &--text {
width: fit-content;
min-width: @toast-text-min-width; min-width: @toast-text-min-width;
min-height: unset;
padding: @toast-text-padding; padding: @toast-text-padding;
}
&--default {
width: @toast-default-width;
min-height: @toast-default-min-height;
padding: @toast-default-padding;
.van-toast__icon {
font-size: @toast-icon-size;
}
.van-loading {
margin: 10px 0;
}
.van-toast__text { .van-toast__text {
padding-top: 5px; margin-top: 0;
} }
} }
@ -64,4 +54,16 @@
top: auto; top: auto;
bottom: @toast-position-bottom-distance; bottom: @toast-position-bottom-distance;
} }
&__icon {
font-size: @toast-icon-size;
}
&__loading {
padding: 5px;
}
&__text {
margin-top: 10px;
}
} }

View File

@ -15,6 +15,11 @@ exports[`renders demo correctly 1`] = `
</span></button> <button class="van-button van-button--danger van-button--normal"><span class="van-button__text"> </span></button> <button class="van-button van-button--danger van-button--normal"><span class="van-button__text">
失败提示 失败提示
</span></button></div> </span></button></div>
<div><button class="van-button van-button--primary van-button--normal"><span class="van-button__text">
自定义图标
</span></button> <button class="van-button van-button--primary van-button--normal"><span class="van-button__text">
展示图片
</span></button></div>
<div><button class="van-button van-button--primary van-button--normal"><span class="van-button__text"> <div><button class="van-button van-button--primary van-button--normal"><span class="van-button__text">
高级用法 高级用法
</span></button></div> </span></button></div>

View File

@ -1,7 +1,28 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP // Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`create a forbidClick toast 1`] = ` exports[`create a forbidClick toast 1`] = `
<div class="van-toast van-toast--text van-toast--middle" style="display: none;" name="van-fade"> <div class="van-toast van-toast--middle" style="z-index: 2000;" name="van-fade"><i class="van-icon van-icon-success van-toast__icon">
<div></div> <!----></i></div>
`;
exports[`icon prop 1`] = `
<div class="van-toast van-toast--middle" style="z-index: 2004;" name="van-fade"><i class="van-icon van-icon-star-o van-toast__icon">
<!----></i>
<div class="van-toast__text">Message</div>
</div>
`;
exports[`show html toast 1`] = `
<div class="van-toast van-toast--middle van-toast--text" style="z-index: 2003;" name="van-fade">
<div class="van-toast__text">
<div>Message</div>
</div>
</div>
`;
exports[`show loading toast 1`] = `
<div class="van-toast van-toast--middle" style="z-index: 2002;" name="van-fade">
<div class="van-loading van-loading--circular van-toast__loading"><span class="van-loading__spinner van-loading__spinner--circular" style="color: white;"><svg viewBox="25 25 50 50" class="van-loading__circular"><circle cx="50" cy="50" r="20" fill="none"></circle></svg></span></div>
<div class="van-toast__text">Message</div>
</div> </div>
`; `;

View File

@ -11,6 +11,7 @@ test('create a forbidClick toast', async () => {
type: 'success' type: 'success'
}); });
await later();
expect(toast.$el.outerHTML).toMatchSnapshot(); expect(toast.$el.outerHTML).toMatchSnapshot();
await later(); await later();
@ -30,6 +31,35 @@ it('toast disappeared after duration', async () => {
expect(toast.$el.style.display).toEqual('none'); expect(toast.$el.style.display).toEqual('none');
}); });
test('show loading toast', async () => {
const toast = Toast.loading({
message: 'Message'
});
await later();
expect(toast.$el.outerHTML).toMatchSnapshot();
});
test('show html toast', async () => {
const toast = Toast({
type: 'html',
message: '<div>Message</div>'
});
await later();
expect(toast.$el.outerHTML).toMatchSnapshot();
});
test('icon prop', async () => {
const toast = Toast({
message: 'Message',
icon: 'star-o'
});
await later();
expect(toast.$el.outerHTML).toMatchSnapshot();
});
test('clear toast', () => { test('clear toast', () => {
const toast1 = Toast(); const toast1 = Toast();
expect(toast1.value).toBeTruthy(); expect(toast1.value).toBeTruthy();

View File

@ -2,7 +2,7 @@
### 引入 ### 引入
```javascript ```js
import { Toast } from 'vant'; import { Toast } from 'vant';
Vue.use(Toast); Vue.use(Toast);
@ -12,13 +12,13 @@ Vue.use(Toast);
### 文字提示 ### 文字提示
```javascript ```js
Toast('提示内容'); Toast('提示内容');
``` ```
### 加载提示 ### 加载提示
```javascript ```js
Toast.loading({ Toast.loading({
mask: true, mask: true,
message: '加载中...' message: '加载中...'
@ -27,14 +27,28 @@ Toast.loading({
### 成功/失败提示 ### 成功/失败提示
```javascript ```js
Toast.success('成功文案'); Toast.success('成功文案');
Toast.fail('失败文案'); Toast.fail('失败文案');
``` ```
### 自定义图标
```js
Toast({
text: '自定义图标',
icon: 'like-o'
});
Toast({
text: '展示图片',
icon: 'https://img.yzcdn.cn/vant/logo.png'
});
```
### 高级用法 ### 高级用法
```javascript ```js
const toast = Toast.loading({ const toast = Toast.loading({
duration: 0, // 持续展示 toast duration: 0, // 持续展示 toast
forbidClick: true, // 禁用背景点击 forbidClick: true, // 禁用背景点击
@ -102,6 +116,7 @@ toast2.clear();
| type | 提示类型,可选值为 `loading` `success`<br>`fail` `html` | `String` | `text` | - | | type | 提示类型,可选值为 `loading` `success`<br>`fail` `html` | `String` | `text` | - |
| position | 位置,可选值为 `top` `bottom` | `String` | `middle` | - | | position | 位置,可选值为 `top` `bottom` | `String` | `middle` | - |
| message | 文本内容,支持通过`\n`换行 | `String` | `''` | - | - | | message | 文本内容,支持通过`\n`换行 | `String` | `''` | - | - |
| icon | 自定义图标,支持传入图标名称或图片链接,可选值见 Icon 组件 | `String` | - | 2.0.1 |
| mask | 是否显示背景遮罩层 | `Boolean` | `false` | - | | mask | 是否显示背景遮罩层 | `Boolean` | `false` | - |
| forbidClick | 是否禁止背景点击 | `Boolean` | `false` | - | | forbidClick | 是否禁止背景点击 | `Boolean` | `false` | - |
| loadingType | 加载图标类型, 可选值为 `spinner` | `String` | `circular` | 1.1.3 | | loadingType | 加载图标类型, 可选值为 `spinner` | `String` | `circular` | 1.1.3 |

1
types/toast.d.ts vendored
View File

@ -4,6 +4,7 @@ import { VanPopupMixin } from './mixins/popup';
type ToastMessage = string | number; type ToastMessage = string | number;
export type ToastOptions = { export type ToastOptions = {
icon?: string;
type?: string; type?: string;
mask?: boolean; mask?: boolean;
position?: string; position?: string;