index.vue 4.37 KB
Newer Older
dragondean@qq.com's avatar
dragondean@qq.com committed
1
<template>
2
    <div class="dict-selector">
3 4
        <el-select v-if="formType == 'select'" v-model="valueSync" :placeholder="placeholder" clearable :multiple="multiple" :disabled="disabled">
            <el-option v-for="dict in formattedList"
5 6
                        :key="dict.value" :label="dict.label" :value="dict.value"/>
        </el-select>
7 8
        <el-radio-group  v-if="formType == 'radio'" v-model="valueSync" :disabled="disabled">
            <el-radio v-for="dict in formattedList" :label="dict.value" :checked="valueSync === dict.value" :key="dict.value">{{dict.label}}</el-radio>
9
        </el-radio-group>
10 11
        <el-checkbox-group  v-if="formType == 'checkbox'" v-model="valueSync" :disabled="disabled">
            <el-checkbox v-for="dict in formattedList" :label="dict.value" :key="dict.value">{{dict.label}}</el-checkbox>
12 13
        </el-checkbox-group>
    </div>
dragondean@qq.com's avatar
dragondean@qq.com committed
14 15
</template>
<script>
16 17 18 19 20 21 22 23 24 25
const FORMATTERS = {
    "string": String,
    "bool": function(val){
        return [false, 'false', 0, "0"].indexOf(val) < 0
    },
    'number': Number,
    'array': function(val){
        return typeof val == 'string' ? val.split(',').filter(item => item && item !== '') : val
    }
}
dragondean@qq.com's avatar
dragondean@qq.com committed
26 27 28 29 30 31 32
export default {
    props:{
        placeholder: {
            type: String,
            default: '请选择'
        },
        type: String,
33
        value: [String, Number, Array, Boolean],
dragondean@qq.com's avatar
dragondean@qq.com committed
34
        multiple: Boolean,
35 36 37 38
        formType:{
            type: String,
            default: 'select'
        },
39 40 41 42 43
        formatter: {
            type: [Function, String],
            default: String
        },
        /* forceString: {
dragondean@qq.com's avatar
dragondean@qq.com committed
44 45
            type: Boolean,
            default: true
46
        }, */
47
        defaultable: Boolean, // 是否默认选择第一个
48 49 50 51 52 53 54 55
        disabled: Boolean,
        /**
         * 过滤字典项,用于只使用部分字典项的场景
         */
        filter: {
            type: Function,
            default: () => true
        }
dragondean@qq.com's avatar
dragondean@qq.com committed
56 57 58
    },
    data(){
        return {
59 60 61 62 63
            valueSync: this.multiple ? [] : null
        }
    },
    computed:{
        dictList(){
64
            return this.getList(this.type).filter(this.filter)
65 66 67 68 69 70 71 72 73 74 75 76
        },
        formattedList(){
            let arr = []
            this.dictList.forEach(item => {
                arr.push({
                    label: item.label,
                    value: this.format(item.value),
                    cssClass: item.cssClass,
                    colorType: item.colorType
                })
            })
            return arr
dragondean@qq.com's avatar
dragondean@qq.com committed
77 78 79
        }
    },
    watch:{
80 81 82
        valueSync(val){
            //this.$emit('input', this.forceString ? String(this.valueSync) : this.valueSync)
            this.$emit('input', this.format(val))
dragondean@qq.com's avatar
dragondean@qq.com committed
83 84
        },
        value(val){
85
            this.setValueSync()
86 87 88
        },
        dictList(){
            this.setDefault()
dragondean@qq.com's avatar
dragondean@qq.com committed
89 90
        }
    },
91 92
    created(){
        this.setValueSync()
93
        this.setDefault()
94
    },  
dragondean@qq.com's avatar
dragondean@qq.com committed
95
    methods:{
96 97 98 99 100 101 102 103 104
        format(val){
            if(val === null || val == undefined) return val
            let formatter = typeof this.formatter == 'function' ? this.formatter : FORMATTERS[this.formatter]
            if(!formatter){
                console.warn('格式器无效', this.formatter)
                return val
            }
            return formatter(val)
        },
105
        setValueSync(){
106 107 108 109 110 111 112 113 114 115 116 117
            if(this.value === null || this.value === undefined || this.value === '') return 
            if(this.multiple){
                let value = []
                if(typeof this.value == 'string'){
                    value = this.value.split(',').filter(item => item && item != '')
                }
                this.valueSync = value.map(item => this.format(item))
            }else{
                this.valueSync = this.format(this.value)  
            }
            
            /* if(this.forceString && this.multiple){
118
                this.valueSync = this.value.split(',')
119
            }else this.valueSync = this.forceString ? String(this.value) : this.value */
120
        },
dragondean@qq.com's avatar
dragondean@qq.com committed
121
        getList(){
122 123 124 125 126
            return this.getDictDatas(this.type)
        },
        setDefault(){
            if(!this.defaultable) return 
            if(this.dictList.length && (this.valueSync === null || this.valueSync == '')){
127
                this.valueSync = this.multiple ? [] : this.formattedList[0].value
128
            }
dragondean@qq.com's avatar
dragondean@qq.com committed
129 130 131
        }
    }
}
132 133 134 135 136 137
</script>
<style scoped>
.dict-selector{
    display: inline-block;
}
</style>