[improvement] Area: 使用picker重构组件 (#1175)

This commit is contained in:
rex 2019-01-02 21:40:57 +08:00 committed by neverland
parent c5a3a45742
commit ec80527423
8 changed files with 103 additions and 176 deletions

View File

@ -1,4 +1,5 @@
import Page from '../../common/page'; import Page from '../../common/page';
import Toast from '../../dist/toast/toast';
Page({ Page({
data: { data: {
@ -20,6 +21,16 @@ Page({
}, },
onChange(event) { onChange(event) {
const { values } = event.detail;
Toast(values.map(item => item.name).join('-'));
},
onConfirm(event) {
console.log(event);
},
onCancel(event) {
console.log(event); console.log(event);
} }
}); });

View File

@ -2,6 +2,9 @@
<van-area <van-area
loading="{{ loading }}" loading="{{ loading }}"
area-list="{{ areaList }}" area-list="{{ areaList }}"
bind:change="onChange"
bind:confirm="onConfirm"
bind:cancel="onCancel"
/> />
</demo-block> </demo-block>
@ -10,6 +13,8 @@
value="{{ value }}" value="{{ value }}"
loading="{{ loading }}" loading="{{ loading }}"
area-list="{{ areaList }}" area-list="{{ areaList }}"
bind:change="onChange"
bind:confirm="onConfirm"
/> />
</demo-block> </demo-block>
@ -20,6 +25,9 @@
loading="{{ loading }}" loading="{{ loading }}"
area-list="{{ areaList }}" area-list="{{ areaList }}"
bind:change="onChange" bind:change="onChange"
bind:confirm="onChange" bind:confirm="onConfirm"
/> />
</demo-block> </demo-block>
<van-toast id="van-toast" />

View File

@ -1,6 +1,6 @@
{ {
"component": true, "component": true,
"usingComponents": { "usingComponents": {
"van-loading": "../loading/index" "van-picker": "../picker/index"
} }
} }

View File

@ -1,71 +0,0 @@
@import '../common/style/var.less';
.van-picker {
-webkit-text-size-adjust: 100%; /* avoid iOS text size adjust */
position: relative;
overflow: hidden;
background-color: @white;
user-select: none;
&__toolbar {
display: flex;
justify-content: space-between;
height: 44px;
line-height: 44px;
}
&__cancel,
&__confirm {
color: @blue;
padding: 0 15px;
font-size: 14px;
&:active {
background-color: @active-color;
}
}
&__title {
max-width: 50%;
font-size: 16px;
font-weight: 500;
text-align: center;
}
&__columns {
position: relative;
}
&__loading {
display: flex;
z-index: 4;
position: absolute;
top: 0;
right: 0;
bottom: 0;
left: 0;
align-items: center;
justify-content: center;
background-color: rgba(255, 255, 255, .9);
}
&-column {
flex: 1;
overflow: hidden;
font-size: 16px;
text-align: center;
&__item {
padding: 0 5px;
color: @text-color;
&--selected {
font-weight: 500;
}
&--disabled {
opacity: .3;
}
}
}
}

View File

@ -1,11 +1,13 @@
import { VantComponent } from '../common/component'; import { VantComponent } from '../common/component';
type AreaItem = { type AreaItem = {
name: string; name: string
code: string; code: string
}; };
VantComponent({ VantComponent({
classes: ['active-class', 'toolbar-class', 'column-class'],
props: { props: {
title: String, title: String,
value: String, value: String,
@ -29,8 +31,8 @@ VantComponent({
}, },
data: { data: {
pickerValue: [0, 0, 0], columns: [{ values: [] }, { values: [] }, { values: [] }],
columns: [] displayColumns: [{ values: [] }, { values: [] }, { values: [] }]
}, },
watch: { watch: {
@ -39,55 +41,60 @@ VantComponent({
this.setValues(); this.setValues();
}, },
areaList: 'setValues' areaList: 'setValues',
columnsNum(value) {
this.set({
displayColumns: this.data.columns.slice(0, +value)
});
}
}, },
methods: { methods: {
onCancel() { getPicker() {
this.$emit('cancel', { if (this.picker == null) {
values: this.getValues(), this.picker = this.selectComponent('.van-area__picker');
indexs: this.getIndexs(), }
detail: this.getDetail() return this.picker;
});
}, },
onConfirm() { onCancel(event) {
this.$emit('confirm', { this.emit('cancel', event.detail);
values: this.getValues(), },
indexs: this.getIndexs(),
detail: this.getDetail() onConfirm(event) {
}); this.emit('confirm', event.detail);
},
emit(type, detail) {
detail.values = detail.value;
delete detail.value;
this.$emit(type, detail);
}, },
onChange(event: Weapp.Event) { onChange(event: Weapp.Event) {
const { value } = event.detail; const { index, picker, value } = event.detail;
const { pickerValue } = this.data; this.code = value[index].code;
const displayColumns = this.getDisplayColumns();
const index = pickerValue.findIndex(
(item, index) => item !== value[index]
);
const values = displayColumns[index];
if (index < 0 || value[index] < 0 || !values[value[index]]) {
return;
}
this.code = values[value[index]].code;
this.setValues(); this.setValues();
this.$emit('change', { this.$emit('change', {
picker: this, picker,
values: this.getValues(), values: picker.getValues(),
index index
}); });
}, },
getConfig(type: string) {
const { areaList } = this.data;
return (areaList && areaList[`${type}_list`]) || {};
},
getList(type: string, code?: string): AreaItem[] { getList(type: string, code?: string): AreaItem[] {
let result = []; let result = [];
if (type !== 'province' && !code) { if (type !== 'province' && !code) {
return result; return result;
} }
const list = this.data.areaList && this.data.areaList[`${type}_list`] || {}; const list = this.getConfig(type);
result = Object.keys(list).map(code => ({ result = Object.keys(list).map(code => ({
code, code,
name: list[code] name: list[code]
@ -125,44 +132,35 @@ VantComponent({
}, },
setValues() { setValues() {
let code = const county = this.getConfig('county');
this.code || let code = this.code || Object.keys(county)[0] || '';
(this.data.areaList &&
Object.keys(this.data.areaList.county_list || {})[0]) ||
'';
const province = this.getList('province'); const province = this.getList('province');
const city = this.getList('city', code.slice(0, 2)); const city = this.getList('city', code.slice(0, 2));
this.set({ const picker = this.getPicker();
'columns[0]': province,
'columns[1]': city
});
if (city.length && code.slice(2, 4) === '00') { if (!picker) {
code = city[0].code; return;
} }
this.set({ picker.setColumnValues(0, province);
'columns[2]': this.getList('county', code.slice(0, 4)), picker.setColumnValues(1, city);
pickerValue: [
this.getIndex('province', code), if (city.length && code.slice(2, 4) === '00') {
this.getIndex('city', code), ;[{ code }] = city;
this.getIndex('county', code) }
]
}); picker.setColumnValues(2, this.getList('county', code.slice(0, 4)));
picker.setIndexes([
this.getIndex('province', code),
this.getIndex('city', code),
this.getIndex('county', code)
]);
}, },
getValues() { getValues() {
const { pickerValue = [] } = this.data; const picker = this.getPicker();
const displayColumns = this.getDisplayColumns(); return picker ? picker.getValues().filter(value => !!value) : [];
return displayColumns
.map((option, index) => option[pickerValue[index]])
.filter(value => !!value);
},
getIndexs() {
const { pickerValue, columnsNum } = this.data;
return pickerValue.slice(0, columnsNum);
}, },
getDetail() { getDetail() {
@ -196,11 +194,6 @@ VantComponent({
reset() { reset() {
this.code = ''; this.code = '';
this.setValues(); this.setValues();
},
getDisplayColumns() {
const { columns = [], columnsNum } = this.data;
return columns.slice(0, +columnsNum);
} }
} }
}); });

View File

@ -1,35 +1,16 @@
<view class="van-picker"> <van-picker
<view class="van-picker__toolbar van-hairline--bottom"> class="van-area__picker"
<view class="van-picker__cancel" bindtap="onCancel">取消</view> active-class="active-class"
<view class="van-picker__title">{{ title }}</view> toolbar-class="toolbar-class"
<view class="van-picker__confirm" bindtap="onConfirm">确定</view> column-class="column-class"
</view> show-toolbar
value-key="name"
<view wx:if="{{ loading }}" class="van-picker__loading"> title="{{ title }}"
<van-loading color="#1989fa"/> loading="{{ loading }}"
</view> columns="{{ displayColumns }}"
item-height="{{ itemHeight }}"
<picker-view visible-item-count="{{ visibleItemCount }}"
indicator-style="height: {{ itemHeight }}px;" bind:change="onChange"
style="width: 100%; height: {{ itemHeight * visibleItemCount + 'px' }}" bind:confirm="onConfirm"
bindchange="onChange" bind:cancel="onCancel"
value="{{ pickerValue }}" />
class="van-picker__columns"
>
<picker-view-column
wx:if="{{ rowIndex < columnsNum }}"
wx:for="{{ columns }}"
wx:for-item="row"
wx:for-index="rowIndex"
wx:key="rowIndex"
class="van-picker-column"
>
<view
wx:for="{{ row }}"
wx:key="{{ item.code }}"
style="line-height: {{ itemHeight }}px;"
class="van-picker-column__item {{ index === pickerValue[rowIndex] ? 'van-picker-column__item--selected' : '' }}"
>{{ item.name }}</view>
</picker-view-column>
</picker-view>
</view>

View File

@ -38,7 +38,8 @@
} }
&__column { &__column {
flex: 1; flex: 1 1;
width: 0;
} }
&__loading { &__loading {

View File

@ -38,6 +38,10 @@ VantComponent({
} }
}, },
beforeCreate() {
this.children = [];
},
methods: { methods: {
noop() {}, noop() {},