parent
3d3e56de12
commit
517c1959d8
7 changed files with 886 additions and 0 deletions
@ -0,0 +1,166 @@ |
|||||||
|
<template> |
||||||
|
<div class="action-columns" ref="root"> |
||||||
|
<a-tooltip title="列设置" :get-popup-container="() => $refs.root"> |
||||||
|
<a-popover v-model="visible" placement="bottomRight" trigger="click" :get-popup-container="() => $refs.root"> |
||||||
|
<div slot="title"> |
||||||
|
<a-checkbox :indeterminate="indeterminate" :checked="checkAll" @change="onCheckAllChange" class="check-all" />列展示 |
||||||
|
<a-button @click="resetColumns" style="float: right" type="link" size="small">重置</a-button> |
||||||
|
</div> |
||||||
|
<a-list style="width: 100%" size="small" :key="i" v-for="(col, i) in columns" slot="content"> |
||||||
|
<a-list-item> |
||||||
|
<a-checkbox v-model="col.visible" @change="e => onCheckChange(e, col)"/> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}}: |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<template slot="actions"> |
||||||
|
<a-tooltip title="固定在列头" :get-popup-container="() => $refs.root"> |
||||||
|
<a-icon :class="['left', {active: col.fixed === 'left'}]" @click="fixColumn('left', col)" type="vertical-align-top" /> |
||||||
|
</a-tooltip> |
||||||
|
<a-tooltip title="固定在列尾" :get-popup-container="() => $refs.root"> |
||||||
|
<a-icon :class="['right', {active: col.fixed === 'right'}]" @click="fixColumn('right', col)" type="vertical-align-bottom" /> |
||||||
|
</a-tooltip> |
||||||
|
<a-tooltip title="添加搜索" :get-popup-container="() => $refs.root"> |
||||||
|
<a-icon :class="{active: col.searchAble}" @click="setSearch(col)" type="search" /> |
||||||
|
</a-tooltip> |
||||||
|
</template> |
||||||
|
</a-list-item> |
||||||
|
</a-list> |
||||||
|
<a-icon class="action" type="setting" /> |
||||||
|
</a-popover> |
||||||
|
</a-tooltip> |
||||||
|
</div> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script> |
||||||
|
import cloneDeep from 'lodash.clonedeep' |
||||||
|
|
||||||
|
export default { |
||||||
|
name: 'ActionColumns', |
||||||
|
props: ['columns', 'visibleColumns'], |
||||||
|
data() { |
||||||
|
return { |
||||||
|
visible: false, |
||||||
|
indeterminate: false, |
||||||
|
checkAll: true, |
||||||
|
checkedCounts: this.columns.length, |
||||||
|
backColumns: cloneDeep(this.columns) |
||||||
|
} |
||||||
|
}, |
||||||
|
watch: { |
||||||
|
checkedCounts(val) { |
||||||
|
this.checkAll = val === this.columns.length |
||||||
|
this.indeterminate = val > 0 && val < this.columns.length |
||||||
|
} |
||||||
|
}, |
||||||
|
created() { |
||||||
|
this.$emit('update:visibleColumns', [...this.columns]) |
||||||
|
for (let col of this.columns) { |
||||||
|
if (col.visible === undefined) { |
||||||
|
this.$set(col, 'visible', true) |
||||||
|
} |
||||||
|
if (!col.visible) { |
||||||
|
this.checkedCounts -= 1 |
||||||
|
this.$set(col, 'colSpan', 0) |
||||||
|
this.$set(col, 'customCell', () => ({style: 'display: none;'})) |
||||||
|
} |
||||||
|
} |
||||||
|
}, |
||||||
|
methods: { |
||||||
|
onCheckChange(e, col) { |
||||||
|
if (!col.visible) { |
||||||
|
this.checkedCounts -= 1 |
||||||
|
this.$set(col, 'colSpan', 0) |
||||||
|
this.$set(col, 'customCell', () => ({style: 'display: none;'})) |
||||||
|
} else { |
||||||
|
this.checkedCounts += 1 |
||||||
|
this.$set(col, 'colSpan', undefined) |
||||||
|
this.$set(col, 'customCell', undefined) |
||||||
|
} |
||||||
|
}, |
||||||
|
fixColumn(fixed, col) { |
||||||
|
if (fixed !== col.fixed) { |
||||||
|
this.$set(col, 'fixed', fixed) |
||||||
|
} else { |
||||||
|
this.$set(col, 'fixed', undefined) |
||||||
|
} |
||||||
|
}, |
||||||
|
setSearch(col) { |
||||||
|
this.$set(col, 'searchAble', !col.searchAble) |
||||||
|
if (!col.searchAble && col.search) { |
||||||
|
this.resetSearch(col) |
||||||
|
} |
||||||
|
}, |
||||||
|
resetSearch(col) { |
||||||
|
col.search.value = col.dataType === 'boolean' ? false : undefined |
||||||
|
col.search.backup = undefined |
||||||
|
}, |
||||||
|
resetColumns() { |
||||||
|
const {columns, backColumns} = this |
||||||
|
let counts = columns.length |
||||||
|
backColumns.forEach((back, index) => { |
||||||
|
const column = columns[index] |
||||||
|
column.visible = back.visible === undefined || back.visible |
||||||
|
if (column.visible) { |
||||||
|
this.$set(column, 'colSpan', undefined) |
||||||
|
this.$set(column, 'customCell', undefined) |
||||||
|
} else { |
||||||
|
counts -= 1 |
||||||
|
this.$set(column, 'colSpan', 0) |
||||||
|
this.$set(column, 'customCell', () => ({style: 'display: none;'})) |
||||||
|
} |
||||||
|
if (back.fixed !== undefined) { |
||||||
|
column.fixed = back.fixed |
||||||
|
} else { |
||||||
|
this.$set(column, 'fixed', undefined) |
||||||
|
} |
||||||
|
column.searchAble = back.searchAble |
||||||
|
this.resetSearch(column) |
||||||
|
}) |
||||||
|
this.checkedCounts = counts |
||||||
|
this.visible = false |
||||||
|
this.$emit('reset', this.getConditions(columns)) |
||||||
|
}, |
||||||
|
onCheckAllChange(e) { |
||||||
|
if (e.target.checked) { |
||||||
|
this.checkedCounts = this.columns.length |
||||||
|
this.columns.forEach(col => { |
||||||
|
col.visible = true |
||||||
|
this.$set(col, 'colSpan', undefined) |
||||||
|
this.$set(col, 'customCell', undefined) |
||||||
|
}) |
||||||
|
} else { |
||||||
|
this.checkedCounts = 0 |
||||||
|
this.columns.forEach(col => { |
||||||
|
col.visible = false |
||||||
|
this.$set(col, 'colSpan', 0) |
||||||
|
this.$set(col, 'customCell', () => ({style: 'display: none;'})) |
||||||
|
}) |
||||||
|
} |
||||||
|
}, |
||||||
|
getConditions(columns) { |
||||||
|
const conditions = {} |
||||||
|
columns.filter(item => item.search.value !== undefined && item.search.value !== '' && item.search.value !== null) |
||||||
|
.forEach(col => { |
||||||
|
conditions[col.dataIndex] = col.search.value |
||||||
|
}) |
||||||
|
return conditions |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</script> |
||||||
|
|
||||||
|
<style scoped lang="less"> |
||||||
|
.action-columns{ |
||||||
|
display: inline-block; |
||||||
|
.check-all{ |
||||||
|
margin-right: 8px; |
||||||
|
} |
||||||
|
.left,.right{ |
||||||
|
transform: rotate(-90deg); |
||||||
|
} |
||||||
|
.active{ |
||||||
|
color: @primary-color; |
||||||
|
} |
||||||
|
} |
||||||
|
</style> |
@ -0,0 +1,44 @@ |
|||||||
|
<template> |
||||||
|
<div class="action-size" ref="root"> |
||||||
|
<a-tooltip title="密度"> |
||||||
|
<a-dropdown placement="bottomCenter" :trigger="['click']" :get-popup-container="() => $refs.root"> |
||||||
|
<a-icon class="action" type="column-height" /> |
||||||
|
<a-menu :selected-keys="[value]" slot="overlay" @click="onClick"> |
||||||
|
<a-menu-item key="default"> |
||||||
|
默认 |
||||||
|
</a-menu-item> |
||||||
|
<a-menu-item key="middle"> |
||||||
|
中等 |
||||||
|
</a-menu-item> |
||||||
|
<a-menu-item key="small"> |
||||||
|
紧密 |
||||||
|
</a-menu-item> |
||||||
|
</a-menu> |
||||||
|
</a-dropdown> |
||||||
|
</a-tooltip> |
||||||
|
</div> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script> |
||||||
|
export default { |
||||||
|
name: 'ActionSize', |
||||||
|
props: ['value'], |
||||||
|
inject: ['table'], |
||||||
|
data() { |
||||||
|
return { |
||||||
|
selectedKeys: ['middle'] |
||||||
|
} |
||||||
|
}, |
||||||
|
methods: { |
||||||
|
onClick({key}) { |
||||||
|
this.$emit('input', key) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</script> |
||||||
|
|
||||||
|
<style scoped lang="less"> |
||||||
|
.action-size{ |
||||||
|
display: inline-block; |
||||||
|
} |
||||||
|
</style> |
@ -0,0 +1,239 @@ |
|||||||
|
<template> |
||||||
|
<div ref="table" :id="id" class="advanced-table"> |
||||||
|
<a-spin :spinning="loading"> |
||||||
|
<div :class="['header-bar', size]"> |
||||||
|
<div class="title"> |
||||||
|
<template v-if="title">{{title}}</template> |
||||||
|
<slot v-else-if="$slots.title" name="title"></slot> |
||||||
|
<template v-else>高级表格</template> |
||||||
|
</div> |
||||||
|
<div class="search"> |
||||||
|
<search-area @change="onSearchChange" :columns="columns" > |
||||||
|
<template :slot="slot" v-for="slot in slots"> |
||||||
|
<slot :name="slot"></slot> |
||||||
|
</template> |
||||||
|
</search-area> |
||||||
|
</div> |
||||||
|
<div class="actions"> |
||||||
|
<a-tooltip title="刷新"> |
||||||
|
<a-icon @click="refresh" class="action" :type="loading ? 'loading' : 'reload'" /> |
||||||
|
</a-tooltip> |
||||||
|
<action-size v-model="sSize" class="action" /> |
||||||
|
<action-columns :columns="columns" @reset="onColumnsReset" class="action"> |
||||||
|
<template :slot="slot" v-for="slot in slots"> |
||||||
|
<slot :name="slot"></slot> |
||||||
|
</template> |
||||||
|
</action-columns> |
||||||
|
<a-tooltip title="全屏"> |
||||||
|
<a-icon @click="toggleScreen" class="action" :type="fullScreen ? 'fullscreen-exit' : 'fullscreen'" /> |
||||||
|
</a-tooltip> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<a-table |
||||||
|
v-bind="{...$options.propsData, title: undefined, loading: false}" |
||||||
|
:size="sSize" |
||||||
|
@expandedRowsChange="onExpandedRowsChange" |
||||||
|
@change="onChange" |
||||||
|
@expand="onExpand" |
||||||
|
> |
||||||
|
<template slot-scope="text, record, index" :slot="slot" v-for="slot in scopedSlots "> |
||||||
|
<slot :name="slot" v-bind="{text, record, index}"></slot> |
||||||
|
</template> |
||||||
|
<template :slot="slot" v-for="slot in slots"> |
||||||
|
<slot :name="slot"></slot> |
||||||
|
</template> |
||||||
|
<template slot-scope="record, index, indent, expanded" :slot="$scopedSlots.expandedRowRender ? 'expandedRowRender' : ''"> |
||||||
|
<slot v-bind="{record, index, indent, expanded}" :name="$scopedSlots.expandedRowRender ? 'expandedRowRender' : ''"></slot> |
||||||
|
</template> |
||||||
|
</a-table> |
||||||
|
</a-spin> |
||||||
|
</div> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script> |
||||||
|
import ActionSize from '@/components/table/advance/ActionSize' |
||||||
|
import ActionColumns from '@/components/table/advance/ActionColumns' |
||||||
|
import SearchArea from '@/components/table/advance/SearchArea' |
||||||
|
export default { |
||||||
|
name: 'AdvanceTable', |
||||||
|
components: {SearchArea, ActionColumns, ActionSize}, |
||||||
|
props: { |
||||||
|
tableLayout: String, |
||||||
|
bordered: Boolean, |
||||||
|
childrenColumnName: Array[String], |
||||||
|
columns: Array, |
||||||
|
components: Object, |
||||||
|
dataSource: Array, |
||||||
|
defaultExpandAllRows: Array[String], |
||||||
|
expandedRowKeys: Array[String], |
||||||
|
expandedRowRender: Function, |
||||||
|
expandIcon: Function, |
||||||
|
expandRowByClick: Boolean, |
||||||
|
expandIconColumnIndex: Number, |
||||||
|
footer: Function, |
||||||
|
indentSize: Number, |
||||||
|
loading: Boolean, |
||||||
|
locale: Object, |
||||||
|
pagination: Object, |
||||||
|
rowClassName: Function, |
||||||
|
rowKey: [String, Function], |
||||||
|
rowSelection: Object, |
||||||
|
scroll: Object, |
||||||
|
showHeader: Boolean, |
||||||
|
size: String, |
||||||
|
title: String, |
||||||
|
customHeaderRow: Function, |
||||||
|
customRow: Function, |
||||||
|
getPopupContainer: Function, |
||||||
|
transformCellText: Function |
||||||
|
}, |
||||||
|
provide() { |
||||||
|
return { |
||||||
|
table: this |
||||||
|
} |
||||||
|
}, |
||||||
|
data() { |
||||||
|
return { |
||||||
|
id: `${new Date().getTime()}-${Math.floor(Math.random() * 10)}`, |
||||||
|
sSize: this.size || 'default', |
||||||
|
fullScreen: false, |
||||||
|
conditions: [] |
||||||
|
} |
||||||
|
}, |
||||||
|
computed: { |
||||||
|
slots() { |
||||||
|
return Object.keys(this.$slots).filter(slot => slot !== 'title') |
||||||
|
}, |
||||||
|
scopedSlots() { |
||||||
|
return Object.keys(this.$scopedSlots).filter(slot => slot !== 'expandedRowRender' && slot !== 'title') |
||||||
|
} |
||||||
|
}, |
||||||
|
created() { |
||||||
|
this.addListener() |
||||||
|
}, |
||||||
|
beforeDestroy() { |
||||||
|
this.removeListener() |
||||||
|
}, |
||||||
|
methods: { |
||||||
|
refresh() { |
||||||
|
this.$emit('refresh', this.conditions) |
||||||
|
}, |
||||||
|
onSearchChange(conditions) { |
||||||
|
this.conditions = conditions |
||||||
|
this.$emit('search', conditions) |
||||||
|
}, |
||||||
|
toggleScreen() { |
||||||
|
if (this.fullScreen) { |
||||||
|
this.outFullScreen() |
||||||
|
} else { |
||||||
|
this.inFullScreen() |
||||||
|
} |
||||||
|
}, |
||||||
|
inFullScreen() { |
||||||
|
const el = this.$refs.table |
||||||
|
if (el.requestFullscreen) { |
||||||
|
el.requestFullscreen() |
||||||
|
return true |
||||||
|
} else if (el.webkitRequestFullScreen) { |
||||||
|
el.webkitRequestFullScreen() |
||||||
|
return true |
||||||
|
} else if (el.mozRequestFullScreen) { |
||||||
|
el.mozRequestFullScreen() |
||||||
|
return true |
||||||
|
} else if (el.msRequestFullscreen) { |
||||||
|
el.msRequestFullscreen() |
||||||
|
return true |
||||||
|
} |
||||||
|
this.$message.warn('对不起,您的浏览器不支持全屏模式') |
||||||
|
return false |
||||||
|
}, |
||||||
|
outFullScreen() { |
||||||
|
if (document.exitFullscreen) { |
||||||
|
document.exitFullscreen() |
||||||
|
} else if (document.webkitCancelFullScreen) { |
||||||
|
document.webkitCancelFullScreen(); |
||||||
|
} else if (document.mozCancelFullScreen) { |
||||||
|
document.mozCancelFullScreen() |
||||||
|
} else if (document.msExitFullscreen) { |
||||||
|
document.msExiFullscreen() |
||||||
|
} |
||||||
|
}, |
||||||
|
onColumnsReset(conditions) { |
||||||
|
this.$emit('reset', conditions) |
||||||
|
}, |
||||||
|
onExpandedRowsChange(expandedRows) { |
||||||
|
this.$emit('expandedRowsChange', expandedRows) |
||||||
|
}, |
||||||
|
onChange(pagination, filters, sorter, options) { |
||||||
|
this.$emit('expandedRowsChange', pagination, filters, sorter, options) |
||||||
|
}, |
||||||
|
onExpand(expanded, record) { |
||||||
|
this.$emit('expandedRowsChange', expanded, record) |
||||||
|
}, |
||||||
|
addListener() { |
||||||
|
document.addEventListener('fullscreenchange', this.fullScreenListener) |
||||||
|
document.addEventListener('webkitfullscreenchange', this.fullScreenListener) |
||||||
|
document.addEventListener('mozfullscreenchange', this.fullScreenListener) |
||||||
|
document.addEventListener('msfullscreenchange', this.fullScreenListener) |
||||||
|
}, |
||||||
|
removeListener() { |
||||||
|
document.removeEventListener('fullscreenchange', this.fullScreenListener) |
||||||
|
document.removeEventListener('webkitfullscreenchange', this.fullScreenListener) |
||||||
|
document.removeEventListener('mozfullscreenchange', this.fullScreenListener) |
||||||
|
document.removeEventListener('msfullscreenchange', this.fullScreenListener) |
||||||
|
}, |
||||||
|
fullScreenListener(e) { |
||||||
|
if (e.target.id === this.id) { |
||||||
|
this.fullScreen = !this.fullScreen |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</script> |
||||||
|
|
||||||
|
<style scoped lang="less"> |
||||||
|
.advanced-table{ |
||||||
|
background-color: @component-background; |
||||||
|
.header-bar{ |
||||||
|
padding: 16px 24px; |
||||||
|
display: flex; |
||||||
|
align-items: center; |
||||||
|
border-radius: 4px; |
||||||
|
transition: all 0.3s; |
||||||
|
&.middle{ |
||||||
|
padding: 12px 16px; |
||||||
|
} |
||||||
|
&.small{ |
||||||
|
padding: 8px 12px; |
||||||
|
border: 1px solid @border-color; |
||||||
|
border-bottom: 0; |
||||||
|
.title{ |
||||||
|
font-size: 16px; |
||||||
|
} |
||||||
|
} |
||||||
|
.title{ |
||||||
|
transition: all 0.3s; |
||||||
|
font-size: 18px; |
||||||
|
color: @title-color; |
||||||
|
font-weight: 700; |
||||||
|
} |
||||||
|
.search{ |
||||||
|
flex: 1; |
||||||
|
text-align: right; |
||||||
|
margin: 0 24px; |
||||||
|
} |
||||||
|
.actions{ |
||||||
|
text-align: right; |
||||||
|
font-size: 17px; |
||||||
|
color: @text-color; |
||||||
|
.action{ |
||||||
|
margin: 0 8px; |
||||||
|
cursor: pointer; |
||||||
|
&:hover{ |
||||||
|
color: @primary-color; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</style> |
@ -0,0 +1,262 @@ |
|||||||
|
<template> |
||||||
|
<div class="search-area" ref="root"> |
||||||
|
<div class="search-item" :key="index" v-for="(col, index) in searchCols"> |
||||||
|
<div v-if="col.dataType === 'boolean'" class="title active"> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}}: |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<a-switch @change="onSwitchChange" class="switch" v-model="col.search.value" size="small" checked-children="是" un-checked-children="否" /> |
||||||
|
</div> |
||||||
|
<div v-else-if="col.dataType === 'time'" class="title active"> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}}: |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<a-time-picker v-model="col.search.value" placeholder="选择时间" @change="(time, timeStr) => onCalendarChange(time, timeStr, col)" @openChange="open => onCalendarOpenChange(open, col)" class="time-picker" size="small" /> |
||||||
|
</div> |
||||||
|
<div v-else-if="col.dataType === 'date'" class="title active"> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}}: |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<a-date-picker v-model="col.search.value" @change="onDateChange(col)" class="date-picker" size="small" /> |
||||||
|
</div> |
||||||
|
<div v-else-if="col.dataType === 'datetime'" class="title datetime active"> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}}: |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<a-date-picker v-model="col.search.value" @change="(date, dateStr) => onCalendarChange(date, dateStr, col)" @openChange="open => onCalendarOpenChange(open, col)" show-time class="datetime-picker" size="small" /> |
||||||
|
</div> |
||||||
|
<div v-else-if="col.dataType === 'select'" class="title active"> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}}: |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<a-select :allowClear="true" :options="col.search.selectOptions" v-model="col.search.value" placeholder="请选择..." @change="onSelectChange(col)" class="select" slot="content" size="small"> |
||||||
|
</a-select> |
||||||
|
</div> |
||||||
|
<a-popover v-else @visibleChange="onVisibleChange(col, index)" v-model="col.search.visible" placement="bottom" :trigger="['click']" :get-popup-container="() => $refs.root"> |
||||||
|
<div :class="['title', {active: col.search.value}]"> |
||||||
|
<template v-if="col.title"> |
||||||
|
{{col.title}} |
||||||
|
</template> |
||||||
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot> |
||||||
|
<div class="value " v-if="col.search.value">: {{col | searchValue}}</div> |
||||||
|
<a-icon class="icon-down" type="down"/> |
||||||
|
</div> |
||||||
|
<div class="operations" slot="content"> |
||||||
|
<a-button @click="onCancel(col)" class="btn" size="small" type="link">取消</a-button> |
||||||
|
<a-button @click="onConfirm(col)" class="btn" size="small" type="primary">确认</a-button> |
||||||
|
</div> |
||||||
|
<div class="search-overlay" slot="title"> |
||||||
|
<a-input :id="`${searchIdPrefix}${index}`" :allow-clear="true" @keyup.esc="onCancel(col)" @keyup.enter="onConfirm(col)" v-model="col.search.value" size="default" /> |
||||||
|
</div> |
||||||
|
</a-popover> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script> |
||||||
|
import fastEqual from 'fast-deep-equal' |
||||||
|
import moment from 'moment' |
||||||
|
|
||||||
|
export default { |
||||||
|
name: 'SearchArea', |
||||||
|
props: ['columns'], |
||||||
|
inject: ['table'], |
||||||
|
created() { |
||||||
|
this.columns.forEach(item => { |
||||||
|
this.$set(item, 'search', {...item.search, visible: false, value: item.dataType === 'boolean' ? false : undefined, format: this.getCalendarFormat(item)}) |
||||||
|
}) |
||||||
|
}, |
||||||
|
filters: { |
||||||
|
searchValue(col) { |
||||||
|
if (col.dataType === 'time' && col.search.value) { |
||||||
|
return col.search.value.format('HH:mm:ss') |
||||||
|
} |
||||||
|
return col.search.value |
||||||
|
} |
||||||
|
}, |
||||||
|
watch: { |
||||||
|
searchCols(newVal, oldVal) { |
||||||
|
if (newVal.length != oldVal.length) { |
||||||
|
const newConditions = this.getConditions(newVal) |
||||||
|
if (!fastEqual(newConditions, this.conditions)) { |
||||||
|
this.conditions = newConditions |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
}, |
||||||
|
data() { |
||||||
|
return { |
||||||
|
conditions: [] |
||||||
|
} |
||||||
|
}, |
||||||
|
computed: { |
||||||
|
searchCols() { |
||||||
|
return this.columns.filter(item => item.searchAble) |
||||||
|
}, |
||||||
|
searchIdPrefix() { |
||||||
|
return this.table.id + '-ipt-' |
||||||
|
} |
||||||
|
}, |
||||||
|
methods: { |
||||||
|
onCancel(col) { |
||||||
|
col.search.value = col.search.backup |
||||||
|
col.search.visible = false |
||||||
|
}, |
||||||
|
onConfirm(col) { |
||||||
|
col.search.backup = col.search.value |
||||||
|
col.search.visible = false |
||||||
|
const conditions = this.getConditions(this.searchCols) |
||||||
|
if (!fastEqual(conditions, this.conditions)) { |
||||||
|
this.conditions = conditions |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
} |
||||||
|
}, |
||||||
|
onSwitchChange() { |
||||||
|
this.conditions = this.getConditions(this.searchCols) |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
}, |
||||||
|
onSelectChange() { |
||||||
|
this.conditions = this.getConditions(this.searchCols) |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
}, |
||||||
|
onCalendarOpenChange(open, col) { |
||||||
|
col.search.visible = open |
||||||
|
const {momentEqual, getConditions} = this |
||||||
|
const {value, backup, format} = col.search |
||||||
|
if (!open && !momentEqual(value, backup, format)) { |
||||||
|
col.search.backup = moment(value) |
||||||
|
this.conditions = getConditions(this.searchCols) |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
} |
||||||
|
}, |
||||||
|
onCalendarChange(date, dateStr, col) { |
||||||
|
const {momentEqual, getConditions} = this |
||||||
|
const {value, backup, format} = col.search |
||||||
|
if (!col.search.visible && !momentEqual(value, backup, format)) { |
||||||
|
col.search.backup = moment(value) |
||||||
|
this.conditions = getConditions(this.searchCols) |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
} |
||||||
|
}, |
||||||
|
onDateChange(col) { |
||||||
|
const {momentEqual, getConditions} = this |
||||||
|
const {value, backup} = col.search |
||||||
|
if (!momentEqual(value, backup, 'YYYY-MM-DD')) { |
||||||
|
col.search.backup = moment(value) |
||||||
|
this.conditions = getConditions(this.searchCols) |
||||||
|
this.$emit('change', this.conditions) |
||||||
|
} |
||||||
|
}, |
||||||
|
getCalendarFormat(col) { |
||||||
|
const dataType = col.dataType |
||||||
|
switch(dataType) { |
||||||
|
case 'time': return 'HH:mm:ss' |
||||||
|
case 'date': return 'YYYY-MM-DD' |
||||||
|
case 'datetime': return 'YYYY-MM-DD HH:mm:ss' |
||||||
|
default: return col.search && col.search.format |
||||||
|
} |
||||||
|
}, |
||||||
|
getConditions(columns) { |
||||||
|
const conditions = {} |
||||||
|
columns.filter(item => item.search.value !== undefined && item.search.value !== '' && item.search.value !== null) |
||||||
|
.forEach(col => { |
||||||
|
conditions[col.dataIndex] = col.search.value |
||||||
|
}) |
||||||
|
return conditions |
||||||
|
}, |
||||||
|
onVisibleChange(col, index) { |
||||||
|
if (!col.search.visible) { |
||||||
|
col.search.value = col.search.backup |
||||||
|
} else { |
||||||
|
let input = document.getElementById(`${this.searchIdPrefix}${index}`) |
||||||
|
if (input) { |
||||||
|
setTimeout(() => {input.focus()}, 0) |
||||||
|
} else { |
||||||
|
this.$nextTick(() => { |
||||||
|
input = document.getElementById(`${this.searchIdPrefix}${index}`) |
||||||
|
input.focus() |
||||||
|
}) |
||||||
|
} |
||||||
|
} |
||||||
|
}, |
||||||
|
momentEqual(target, source, format) { |
||||||
|
if (target === source) { |
||||||
|
return true |
||||||
|
} else if (target && source && target.format(format) === source.format(format)) { |
||||||
|
return true |
||||||
|
} |
||||||
|
return false |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</script> |
||||||
|
|
||||||
|
<style scoped lang="less"> |
||||||
|
.search-area{ |
||||||
|
margin: -4px 0; |
||||||
|
.search-item{ |
||||||
|
margin: 4px 4px; |
||||||
|
display: inline-block; |
||||||
|
.title{ |
||||||
|
padding: 4px 8px; |
||||||
|
cursor: pointer; |
||||||
|
border-radius: 4px; |
||||||
|
user-select: none; |
||||||
|
display: inline-flex; |
||||||
|
align-items: center; |
||||||
|
.switch{ |
||||||
|
margin-left: 4px; |
||||||
|
} |
||||||
|
.time-picker{ |
||||||
|
margin-left: 4px; |
||||||
|
width: 96px; |
||||||
|
} |
||||||
|
.date-picker{ |
||||||
|
margin-left: 4px; |
||||||
|
width: 120px; |
||||||
|
} |
||||||
|
.datetime-picker{ |
||||||
|
margin-left: 4px; |
||||||
|
width: 195px; |
||||||
|
} |
||||||
|
.value{ |
||||||
|
display: inline-block; |
||||||
|
overflow: hidden; |
||||||
|
flex:1; |
||||||
|
max-width: 144px; |
||||||
|
text-overflow: ellipsis; |
||||||
|
word-break: break-all; |
||||||
|
white-space: nowrap; |
||||||
|
} |
||||||
|
&.active{ |
||||||
|
background-color: @layout-bg-color; |
||||||
|
} |
||||||
|
} |
||||||
|
.icon-down{ |
||||||
|
margin-left: 4px; |
||||||
|
font-size: 12px; |
||||||
|
} |
||||||
|
} |
||||||
|
.search-overlay{ |
||||||
|
padding: 8px 0px; |
||||||
|
text-align: center; |
||||||
|
} |
||||||
|
.select{ |
||||||
|
margin-left: 4px; |
||||||
|
max-width: 144px; |
||||||
|
min-width: 96px; |
||||||
|
} |
||||||
|
.operations{ |
||||||
|
display: flex; |
||||||
|
justify-content: space-between; |
||||||
|
.btn{ |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</style> |
@ -0,0 +1,2 @@ |
|||||||
|
import AdvanceTable from './AdvanceTable' |
||||||
|
export default AdvanceTable |
@ -0,0 +1,168 @@ |
|||||||
|
<template> |
||||||
|
<div> |
||||||
|
<advance-table |
||||||
|
:columns="columns" |
||||||
|
:data-source="dataSource" |
||||||
|
title="高级表格-Beta" |
||||||
|
:loading="loading" |
||||||
|
rowKey="id" |
||||||
|
@search="onSearch" |
||||||
|
@refresh="onRefresh" |
||||||
|
@reset="onReset" |
||||||
|
> |
||||||
|
<template slot="statusTitle"> |
||||||
|
状态<a-icon style="margin: 0 4px" type="info-circle" /> |
||||||
|
</template> |
||||||
|
<template slot="send" slot-scope="{text}"> |
||||||
|
{{text ? '是' : '否'}} |
||||||
|
</template> |
||||||
|
<template slot="status" slot-scope="{text}"> |
||||||
|
{{text | statusStr}} |
||||||
|
</template> |
||||||
|
</advance-table> |
||||||
|
</div> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script> |
||||||
|
import AdvanceTable from '@/components/table/advance/AdvanceTable' |
||||||
|
import moment from 'moment' |
||||||
|
|
||||||
|
const goods = ['运动鞋', 'T恤', '长裤', '短裤'] |
||||||
|
const dataSource = [] |
||||||
|
const current = new Date().getTime() |
||||||
|
for (let i = 0; i < 100; i++) { |
||||||
|
dataSource.push({ |
||||||
|
id: i, |
||||||
|
name: goods[Math.floor((Math.random() * 4))], |
||||||
|
orderId: `${new Date().getTime()}-${Math.floor(Math.random() * 10)}`, |
||||||
|
status: Math.floor((Math.random() * 4) + 1), |
||||||
|
send: (i % 2) === 1, |
||||||
|
sendTime: moment(current - Math.floor((Math.random() * 8000000))).format('YYYY-MM-DD HH:mm:ss'), |
||||||
|
orderDate: moment(current - Math.floor((Math.random() * 800000000))).format('YYYY-MM-DD'), |
||||||
|
auditTime: moment(current - Math.floor((Math.random() * 8000000))).format('HH:mm:ss'), |
||||||
|
}) |
||||||
|
} |
||||||
|
export default { |
||||||
|
name: 'Table', |
||||||
|
components: {AdvanceTable}, |
||||||
|
filters: { |
||||||
|
statusStr(val) { |
||||||
|
switch (val) { |
||||||
|
case 1: return '已下单' |
||||||
|
case 2: return '已付款' |
||||||
|
case 3: return '已审核' |
||||||
|
case 4: return '已发货' |
||||||
|
} |
||||||
|
} |
||||||
|
}, |
||||||
|
data() { |
||||||
|
return { |
||||||
|
loading: false, |
||||||
|
columns: [ |
||||||
|
{ |
||||||
|
title: '商品名称', |
||||||
|
dataIndex: 'name', |
||||||
|
searchAble: true |
||||||
|
}, |
||||||
|
{ |
||||||
|
title: '订单号', |
||||||
|
dataIndex: 'orderId' |
||||||
|
}, |
||||||
|
{ |
||||||
|
searchAble: true, |
||||||
|
dataIndex: 'status', |
||||||
|
dataType: 'select', |
||||||
|
slots: {title: 'statusTitle'}, |
||||||
|
scopedSlots: {customRender: 'status'}, |
||||||
|
search: { |
||||||
|
selectOptions: [ |
||||||
|
{title: '已下单', value: 1}, |
||||||
|
{title: '已付款', value: 2}, |
||||||
|
{title: '已审核', value: 3}, |
||||||
|
{title: '已发货', value: 4} |
||||||
|
] |
||||||
|
} |
||||||
|
}, |
||||||
|
{ |
||||||
|
title: '发货', |
||||||
|
searchAble: true, |
||||||
|
dataIndex: 'send', |
||||||
|
dataType: 'boolean', |
||||||
|
scopedSlots: {customRender: 'send'} |
||||||
|
}, |
||||||
|
{ |
||||||
|
title: '发货时间', |
||||||
|
dataIndex: 'sendTime', |
||||||
|
dataType: 'datetime' |
||||||
|
}, |
||||||
|
{ |
||||||
|
title: '下单日期', |
||||||
|
searchAble: true, |
||||||
|
dataIndex: 'orderDate', |
||||||
|
dataType: 'date' |
||||||
|
}, |
||||||
|
{ |
||||||
|
title: '审核时间', |
||||||
|
searchAble: true, |
||||||
|
dataIndex: 'auditTime', |
||||||
|
dataType: 'time', |
||||||
|
}, |
||||||
|
], |
||||||
|
dataSource: dataSource |
||||||
|
} |
||||||
|
}, |
||||||
|
methods: { |
||||||
|
onSearch(conditions) { |
||||||
|
this.loading = true |
||||||
|
this.searchGoods(conditions).then(result => { |
||||||
|
this.dataSource = result |
||||||
|
this.loading = false |
||||||
|
}) |
||||||
|
}, |
||||||
|
onRefresh(conditions) { |
||||||
|
this.loading = true |
||||||
|
this.searchGoods(conditions).then(result => { |
||||||
|
this.dataSource = result |
||||||
|
this.loading = false |
||||||
|
}) |
||||||
|
}, |
||||||
|
onReset(conditions) { |
||||||
|
this.loading = true |
||||||
|
this.searchGoods(conditions).then(result => { |
||||||
|
this.dataSource = result |
||||||
|
this.loading = false |
||||||
|
}) |
||||||
|
}, |
||||||
|
async searchGoods(conditions) { |
||||||
|
const promise = new Promise((resolve, reject) => { |
||||||
|
try { |
||||||
|
const result = dataSource.filter(item => { |
||||||
|
for (let key of Object.keys(conditions)) { |
||||||
|
if (key === 'sendTime') { |
||||||
|
if (conditions[key].format('YYYY-MM-DD HH:mm:ss') !== item[key]) return false |
||||||
|
} else if (key === 'orderDate') { |
||||||
|
if (conditions[key].format('YYYY-MM-DD') !== item[key]) return false |
||||||
|
} else if (key === 'auditTime') { |
||||||
|
if (conditions[key].format('HH:mm:ss') !== item[key]) return false |
||||||
|
} else if (item[key] !== conditions[key]) { |
||||||
|
return false |
||||||
|
} |
||||||
|
} |
||||||
|
return true |
||||||
|
}) |
||||||
|
setTimeout(() => { |
||||||
|
resolve(result) |
||||||
|
}, 300) |
||||||
|
} catch (e) { |
||||||
|
reject(e) |
||||||
|
} |
||||||
|
}) |
||||||
|
return promise |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
</script> |
||||||
|
|
||||||
|
<style scoped> |
||||||
|
|
||||||
|
</style> |
Loading…
Reference in new issue