mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-06 03:58:05 +08:00
[new feature] Actionsheet: 新增行动按钮组件 (#97)
* actionsheet init * actionsheet process * action sheet 功能补齐 * 增加 actionsheet readme * 增加示例入口 * 增加标题配置
This commit is contained in:
parent
cf62caf927
commit
31e9bb83e8
@ -1,6 +1,7 @@
|
||||
{
|
||||
"pages": [
|
||||
"pages/dashboard/index",
|
||||
"pages/actionsheet/index",
|
||||
"pages/btn/index",
|
||||
"pages/badge/index",
|
||||
"pages/capsule/index",
|
||||
|
51
example/pages/actionsheet/index.js
Normal file
51
example/pages/actionsheet/index.js
Normal file
@ -0,0 +1,51 @@
|
||||
const { Actionsheet, extend } = require('../../dist/index');
|
||||
|
||||
Page(extend({}, Actionsheet, {
|
||||
data: {
|
||||
baseActionsheet: {
|
||||
show: false,
|
||||
cancelText: '关闭 Action',
|
||||
closeOnClickOverlay: true,
|
||||
componentId: 'baseActionsheet',
|
||||
actions: [{
|
||||
name: '选项1',
|
||||
subname: '选项描述语1',
|
||||
className: 'action-class',
|
||||
loading: false
|
||||
}, {
|
||||
name: '选项2',
|
||||
subname: '选项描述语2',
|
||||
className: 'action-class',
|
||||
loading: false
|
||||
}]
|
||||
}
|
||||
},
|
||||
|
||||
toggleActionsheet() {
|
||||
this.setData({
|
||||
'baseActionsheet.show': true
|
||||
});
|
||||
},
|
||||
|
||||
handleZanActionsheetCancel({ componentId }) {
|
||||
this.setData({
|
||||
[`${componentId}.show`]: false
|
||||
});
|
||||
},
|
||||
|
||||
handleZanActionsheetClick({ componentId, index }) {
|
||||
console.log(`item index ${index} clicked`);
|
||||
|
||||
this.setData({
|
||||
[`${componentId}.actions[${index}].loading`]: true
|
||||
});
|
||||
|
||||
setTimeout(() => {
|
||||
this.setData({
|
||||
[`${componentId}.show`]: false,
|
||||
[`${componentId}.actions[${index}].loading`]: false
|
||||
});
|
||||
}, 1500);
|
||||
}
|
||||
|
||||
}));
|
3
example/pages/actionsheet/index.json
Normal file
3
example/pages/actionsheet/index.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"navigationBarTitleText": "Actionsheet 行动按钮"
|
||||
}
|
15
example/pages/actionsheet/index.wxml
Normal file
15
example/pages/actionsheet/index.wxml
Normal file
@ -0,0 +1,15 @@
|
||||
<import src="/dist/actionsheet/index.wxml" />
|
||||
|
||||
<view class="container">
|
||||
|
||||
<view class="doc-title zan-hairline--bottom zan-hairline--bottom">ACTIONSHEET</view>
|
||||
|
||||
<view class="zan-btns" style="margin-top: 30vh;">
|
||||
<button class="zan-btn" bindtap="toggleActionsheet">
|
||||
Actionsheet
|
||||
</button>
|
||||
</view>
|
||||
|
||||
<template is="zan-actionsheet" data="{{ ...baseActionsheet }}"></template>
|
||||
|
||||
</view>
|
@ -72,6 +72,9 @@ export default {
|
||||
title: '操作反馈',
|
||||
content: [
|
||||
{
|
||||
name: 'Actionsheet 行动按钮',
|
||||
path: '/pages/actionsheet/index'
|
||||
}, {
|
||||
name: 'Dialog 弹出框',
|
||||
path: '/pages/dialog/index'
|
||||
}, {
|
||||
|
78
packages/actionsheet/README.md
Normal file
78
packages/actionsheet/README.md
Normal file
@ -0,0 +1,78 @@
|
||||
## Actionsheet 行动按钮
|
||||
|
||||
### 使用指南
|
||||
在 app.wxss 中引入组件库所有样式
|
||||
```css
|
||||
@import "path/to/zanui-weapp/dist/index.wxss";
|
||||
```
|
||||
|
||||
在需要使用的页面里引入组件库模板和脚本
|
||||
```html
|
||||
<import src="/dist/actionsheet/index.wxml" />
|
||||
<!-- 直接使用 zan-actionsheet 模板,并且直接传入参数配置 -->
|
||||
<template is="zan-actionsheet" data="{{ ...actionsheet }}"></template>
|
||||
```
|
||||
```js
|
||||
const { Actionsheet, extend } = require('path/to/zanui-weapp/dist/index');
|
||||
|
||||
// 在 Page 中混入 Actionsheet 里面声明的方法
|
||||
Page(extend({}, Actionsheet, {
|
||||
data: {
|
||||
actionsheet: {
|
||||
show: false,
|
||||
actions: []
|
||||
}
|
||||
}
|
||||
// ...
|
||||
}));
|
||||
```
|
||||
|
||||
### 代码演示
|
||||
#### 基础功能
|
||||
在 js 中设置传入的 show 为 true,即可展示出 actionsheet。actionsheet 会根据传入的 actions 展示按钮。
|
||||
```js
|
||||
this.setData({
|
||||
'actionsheet.show': true
|
||||
})
|
||||
```
|
||||
|
||||
当行动按钮被点击或者弹层被关掉时,都可以在可以在页面中注册 `handleZanActionsheetClick` 和 `handleZanActionsheetCancel` 方法来监听。
|
||||
```js
|
||||
Page({
|
||||
// 当行动按钮被关闭时,触发该函数
|
||||
// componentId 即为在模板中传入的 componentId
|
||||
// 用于在一个页面上使用多个 actionsheet 时,进行区分
|
||||
handleZanActionsheetCancel({ componentId }) {
|
||||
},
|
||||
|
||||
// 当行动按钮中有一个被点击时触发
|
||||
// index 代表被点击按钮在传入参数 actions 中的位置
|
||||
handleZanActionsheetClick({ componentId, index }) {
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
#### `Actionsheet` 支持的具体参数如下
|
||||
| 参数 | 说明 | 类型 | 默认值 | 必须 |
|
||||
|-----------|-----------|-----------|-------------|-------------|
|
||||
| show | 用来表示是否展示行动按钮 | Boolean | false | |
|
||||
| cancelText | 行动按钮底部取消按钮的文案,不传则不显示取消按钮 | String | | |
|
||||
| closeOnClickOverlay | 是否在点击背景时,关闭行动按钮 | Boolean | false | |
|
||||
| componentId | 用于区分行动按钮之间的唯一名称 | String | | |
|
||||
| actions | 行动按钮的按钮显示配置 | Array | [] | |
|
||||
|
||||
actions 的具体数据结构
|
||||
```js
|
||||
// actions 为数组结构传入
|
||||
[{
|
||||
// 按钮文案
|
||||
name: '选项1',
|
||||
// 按钮描述文案,不传就不显示
|
||||
subname: '选项描述语1',
|
||||
// 按钮特殊类,可以通过传入这个,为按钮增加特殊样式
|
||||
className: 'action-class',
|
||||
// 按钮是否显示为 loading
|
||||
loading: false
|
||||
}]
|
||||
```
|
||||
|
41
packages/actionsheet/index.js
Normal file
41
packages/actionsheet/index.js
Normal file
@ -0,0 +1,41 @@
|
||||
const { extractComponentId } = require('../common/helper');
|
||||
|
||||
module.exports = {
|
||||
_handleZanActionsheetMaskClick({ currentTarget = {} }) {
|
||||
const dataset = currentTarget.dataset || {};
|
||||
const { componentId, closeOnClickOverlay } = dataset;
|
||||
|
||||
// 判断是否在点击背景时需要关闭弹层
|
||||
if (!closeOnClickOverlay) {
|
||||
return;
|
||||
}
|
||||
|
||||
resolveCancelClick.call(this, { componentId });
|
||||
},
|
||||
|
||||
_handleZanActionsheetCancelBtnClick(e) {
|
||||
const componentId = extractComponentId(e);
|
||||
|
||||
resolveCancelClick.call(this, { componentId });
|
||||
},
|
||||
|
||||
_handleZanActionsheetBtnClick({ currentTarget = {} }) {
|
||||
const dataset = currentTarget.dataset || {};
|
||||
const { componentId, index } = dataset;
|
||||
|
||||
if (this.handleZanActionsheetClick) {
|
||||
this.handleZanActionsheetClick({ componentId, index });
|
||||
} else {
|
||||
console.warn('页面缺少 handleZanActionsheetClick 回调函数');
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
function resolveCancelClick({ componentId }) {
|
||||
console.info('[zan:actionsheet:cancel]');
|
||||
if (this.handleZanActionsheetCancel) {
|
||||
this.handleZanActionsheetCancel({ componentId });
|
||||
} else {
|
||||
console.warn('页面缺少 handleZanActionsheetCancel 回调函数');
|
||||
}
|
||||
}
|
71
packages/actionsheet/index.pcss
Normal file
71
packages/actionsheet/index.pcss
Normal file
@ -0,0 +1,71 @@
|
||||
@import "../common/_var";
|
||||
@import "../common/_mixins";
|
||||
|
||||
.zan-actionsheet {
|
||||
background-color: #f8f8f8;
|
||||
}
|
||||
|
||||
.zan-actionsheet__mask {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
z-index: 10;
|
||||
background: rgba(0, 0, 0, 0.7);
|
||||
display: none;
|
||||
}
|
||||
|
||||
.zan-actionsheet__container {
|
||||
position: fixed;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
background: $background-color;
|
||||
transform: translate3d(0, 50%, 0);
|
||||
transform-origin: center;
|
||||
transition: all 0.2s ease;
|
||||
z-index: 11;
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.zan-actionsheet__btn.zan-btn {
|
||||
height: 50px;
|
||||
line-height: 50px;
|
||||
margin-bottom: 0;
|
||||
|
||||
&::after {
|
||||
border-width: 0;
|
||||
border-bottom-width: 1px;
|
||||
}
|
||||
}
|
||||
|
||||
.zan-actionsheet__btn.zan-btn:last-child {
|
||||
&::after {
|
||||
border-bottom-width: 0;
|
||||
}
|
||||
}
|
||||
|
||||
.zan-actionsheet__subname {
|
||||
margin-left: 2px;
|
||||
font-size: 12px;
|
||||
color: $gray-darker;
|
||||
}
|
||||
|
||||
.zan-actionsheet__footer {
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
/* btn-loading 状态 */
|
||||
.zan-actionsheet__btn.zan-btn--loading .zan-actionsheet__subname {
|
||||
color: transparent;
|
||||
}
|
||||
|
||||
/* zan-actionsheet 展示出来的样式 */
|
||||
.zan-actionsheet--show .zan-actionsheet__container {
|
||||
opacity: 1;
|
||||
transform: translate3d(0, 0, 0);
|
||||
}
|
||||
.zan-actionsheet--show .zan-actionsheet__mask {
|
||||
display: block;
|
||||
}
|
39
packages/actionsheet/index.wxml
Normal file
39
packages/actionsheet/index.wxml
Normal file
@ -0,0 +1,39 @@
|
||||
<template name="zan-actionsheet">
|
||||
<view class="zan-actionsheet {{ show ? 'zan-actionsheet--show' : '' }}">
|
||||
<view
|
||||
class="zan-actionsheet__mask"
|
||||
catchtap="_handleZanActionsheetMaskClick"
|
||||
data-close-on-click-overlay="{{ closeOnClickOverlay }}"
|
||||
data-component-id="{{ componentId }}"></view>
|
||||
<view class="zan-actionsheet__container">
|
||||
<!-- 实际按钮显示 -->
|
||||
<view
|
||||
wx:for="{{ actions }}"
|
||||
wx:for-index="index"
|
||||
wx:for-item="item"
|
||||
wx:key="{{ index }}-{{ item.name }}"
|
||||
catchtap="_handleZanActionsheetBtnClick"
|
||||
data-component-id="{{ componentId }}"
|
||||
data-index="{{ index }}"
|
||||
class="zan-btn zan-actionsheet__btn {{ item.loading ? 'zan-btn--loading' : '' }} {{ item.className }}"
|
||||
>
|
||||
<text>{{ item.name }}</text>
|
||||
<text
|
||||
wx:if="{{ item.subname }}"
|
||||
class="zan-actionsheet__subname">{{ item.subname }}</text>
|
||||
</view>
|
||||
|
||||
<!-- 关闭按钮 -->
|
||||
<view
|
||||
wx:if="{{ cancelText }}"
|
||||
class="zan-actionsheet__footer"
|
||||
>
|
||||
<button
|
||||
class="zan-btn zan-actionsheet__btn"
|
||||
catchtap="_handleZanActionsheetCancelBtnClick"
|
||||
data-component-id="{{ componentId }}"
|
||||
>{{ cancelText }}</button>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
@ -1,3 +1,4 @@
|
||||
exports.Actionsheet = require('./actionsheet/index');
|
||||
exports.CheckLabel = require('./select/index');
|
||||
exports.Dialog = require('./dialog/index');
|
||||
exports.Field = require('./field/index');
|
||||
|
@ -1,3 +1,4 @@
|
||||
@import "actionsheet/index.pcss";
|
||||
@import "badge/index.pcss";
|
||||
@import "btn/index.pcss";
|
||||
@import "capsule/index.pcss";
|
||||
|
Loading…
x
Reference in New Issue
Block a user