vue 簡單的 diy 搜索框 封裝,全局組件化,對象配置

1. 目錄結構:javascript

2. main.js 全局組件vue

3. KjfSearch.vue 代碼:java

  • <template>
        <div class="kjf_search_box" :class="kjfSearchConfig.className">
            <label class="kjf_search_search_label clearfix">
                <input autocomplete="off" readonly="readonly" type="text" :placeholder="kjfSearchConfig.placeholder"
                       class="search_input" :ref="kjfSearchConfig.refName" v-model="kjfSearchKeyWord"
                       @blur="kjfClearPreSearchData" @focus="kjfHandleSearchEvent" @keyup="kjfHandleSearchEvent($event)" />
                <a class="search_by_inputs" href="javascript:" @click="kjfHandleSearchEvent('notEvent')">搜索</a>
            </label>
    
            <div class="search_list" v-show="kjfPreSearchData && kjfPreSearchData.length>0">
                <ul class="search_result">
                    <li v-for="(item, index) in kjfPreSearchData" :key="item.index"
                        :class="{active: index === kjfPreSearchItem}" @click="kjfChoosePreSearchItem(index, item.kjfShowSearchName)">
                        <span class="ellipsis" :title="item.kjfShowSearchName">
                            {{item.kjfShowSearchName}}
                        </span>
                    </li>
                </ul>
            </div>
        </div>
    </template>
    
    <script>
        import {getElementsByCss} from '../utils';
    
        export default {
            name: 'KjfSearch',
            props: {
                kjfSearchConfig: {
                    type: Object
                }
            },
            data: function() {
                return {
                    kjfSearchKeyWord: '',
                    kjfPreSearchData: [],
                    kjfPreSearchItem: -1
                };
            },
            computed: {
            },
            watch: {
                kjfPreSearchData: {
                    deep: true,
                    handler(newArr) {
                        if (newArr && typeof this.kjfSearchConfig.watchCallBack === 'function') {
                            return this.kjfSearchConfig.watchCallBack(newArr);
                        } else {
                            return [];
                        }
                    }
                }
            },
            mounted() {
            },
            methods: {
                kjfClearPreSearchData() {
                    this.$refs[this.kjfSearchConfig.refName].setAttribute('readonly', 'readonly');
                    window.setTimeout(() => {
                        this.kjfPreSearchData = [];
                        this.kjfPreSearchItem = -1;
                    }, 200);
                },
                kjfChoosePreSearchItem(index, searchName) {
                    this.kjfSearchKeyWord = searchName;
                    this.kjfPreSearchData = [];
                    this.kjfPreSearchItem = -1;
                },
                async kjfHandleSearchEvent(e) {
                    this.$refs[this.kjfSearchConfig.refName].removeAttribute('readonly');
    
                    if (e && (e.key === 'ArrowUp' || e.key === 'ArrowDown')) {
                        this.kjfPreSearchItem = e.key === 'ArrowUp' ? (this.kjfPreSearchItem - 1) : (this.kjfPreSearchItem + 1);
                        if (this.kjfPreSearchItem < 0) {
                            this.kjfPreSearchItem = -1;
                        }
                        if (this.kjfPreSearchItem >= this.kjfPreSearchData.length) {
                            this.kjfPreSearchItem = this.kjfPreSearchData.length - 1;
                        }
                        console.log('.' + this.kjfSearchConfig.className + ' .search_list .search_result');
                        const ele = getElementsByCss('.' + this.kjfSearchConfig.className + ' .search_list .search_result')[0];
                        ele && ele.scrollTo(0, this.kjfPreSearchItem * 31);
                        return;
                    } // 上、下選擇下拉選項
    
                    if (e && (e.key === 'Enter') && (this.kjfPreSearchItem !== -1)) {
                        this.kjfSearchKeyWord = this.kjfPreSearchData[this.kjfPreSearchItem].kjfShowSearchName;
                        this.kjfPreSearchData = [];
                        this.kjfPreSearchItem = -1;
                        this.$refs[this.kjfSearchConfig.refName].focus();
                        return;
                    } // 回車選中下拉選項
    
                    if ((e === 'notEvent') || (e && e.key === 'Enter')) {
                        window.setTimeout(async () => {
                            this.kjfSearchConfig.enterCallBack(this.kjfSearchKeyWord);
    
                            this.kjfPreSearchData = [];
                            this.kjfPreSearchItem = -1;
                        }, 200);
                        this.$refs[this.kjfSearchConfig.refName].focus();
                        return;
                    } // 回車鍵按下 或者 點擊搜索按鈕
    
                    window.clearTimeout(this.searchTimer);
                    this.searchTimer = window.setTimeout(async () => {
                        if (this.kjfSearchKeyWord && this.kjfSearchKeyWord.trim()) {
                            if (!this.kjfSearchKeyWord) {
                                this.kjfPreSearchData = [];
                                return;
                            }
                            if (typeof this.kjfSearchConfig.getPreSearchData === 'function') {
                                this.kjfPreSearchData = await this.kjfSearchConfig.getPreSearchData(this.kjfSearchKeyWord);
                            }
                        } else {
                            this.kjfPreSearchData = [];
                            this.kjfPreSearchItem = -1;
                        }
                    }, 200); // 防抖 搜索
                }
            }
        };
    </script>
    
    <style lang="stylus" rel="stylesheet/stylus" scoped>
        @import "../../common/stylus/mixins.styl"
    
        .kjf_search_box
            width 100%
            height 100%
            input
                height 30px
                text-indent 2px
                border-radius 4px
                outline none
                border 1px solid #DCDFE6
            .kjf_search_search_label
                display block
                width 100%
                height 100%
                >input
                    float left
                    display block
                    width 240px
                >a
                    float right
                    display block
                    width 50px
                    height 100%
                    border-radius 4px
                    box-sizing border-box
                    padding 0 10px
                    background-color $themeColor
                    color #fff
            .search_list
                color $fontColor
                width 238px
                max-height 300px
                overflow hidden
                .search_result
                    position absolute
                    top 32px
                    left 2px
                    z-index 100
                    width 238px
                    max-height 300px
                    overflow-x hidden
                    overflow-y auto
                    border-radius 0 0 10px 10px
                    background-color #eee
                    padding 0 0 10px 10px
                    box-sizing border-box
                    >li
                        width 180px
                        margin-top 10px
                        color $fontColor
                        white-space normal
                        word-break break-all
                        word-wrap break-word
                        &.active,
                        &:hover
                            cursor pointer
                            >span
                                color #fff
                                background-color $themeColor
                        >span
                            color $fontColor
    
    </style>

4. 使用實例 主要代碼: ios

  • <template>
        <div id="company_pay_record">
            <ul class="company_pay_record_nav clearfix">
                <li class="right_company_pay_record_search_li clearfix">
                    <KjfSearch :kjfSearchConfig="companyPayRecordSearchConfig"></KjfSearch>
                </li>
            </ul>
        </div>
    </template>
    
    <script>
        import {mapState} from 'vuex';
        import {requestMyLabs} from '../../../../axios';
    
        export default {
            name: 'CompanyPayRecord',
            props: {
                isPC: Boolean
            },
            data () {
                return {
                    companyPayRecordSearchConfig: {
                        className: 'company_pay_record_search',
                        refName: 'companyPayRecordSearch',
                        placeholder: '請輸入操做人/編號',
                        kjfShowSearchName: 'expName',
                        enterCallBack: async (searchKeyWord) => {
                            searchKeyWord = searchKeyWord && searchKeyWord.trim();
                            await this.$store.dispatch('changeLabsCurPageNo', 1);
                            await this.$store.dispatch('changeLabsKeyWord', searchKeyWord);
                            let response = ' - 223 -';
                            try {
                                response = await this.$store.dispatch('getMyLabs', this.myLabsData); // 獲取 myLabsData
                            } catch (e) {
                                this.myConsole(e);
                                this.myConsole('response 228: ');
                                this.myConsole(response);
                            }
                        },
                        getPreSearchData: async (searchKeyWord) => {
                            await this.$store.dispatch('changeLabsKeyWord', searchKeyWord);
                            const response = await requestMyLabs(this.myLabsData);
                            if (response.status === 200) {
                                return response.data.data.content;
                            }
                        },
                        watchCallBack: (newArr) => {
                            let i = 0;
                            for (; i < newArr.length; i++) {
                                newArr[i].kjfShowSearchName = newArr[i].experiment.experimentName;
                            }
                            return newArr;
                        }
                    }
                };
            },
            computed: {
                ...mapState({
                    myLabsData: state => state.myLabs.myLabsData
                })
            },
            async mounted () {
                this.initCompanyPayRecord();
            },
            methods: {
                async initCompanyPayRecord() {
                }
            }
        };
    </script>
    
    <style lang="stylus" rel="stylesheet/stylus" scoped>
        @import "../../../../common/stylus/mixins.styl"
    
        #company_pay_record
            width 100%
            margin 0 auto
            .company_pay_record_nav
                width 100%
                padding-top 20px
                padding-bottom 20px
                box-sizing border-box
                >li
                    float left
                    width 200px
                    height 30px
                    margin-right 14px
                    line-height 30px
                    &:last-child
                        margin-right 0
                .right_company_pay_record_search_li
                    float right
                    position relative
                    width 300px
                    height 30px
    
            .my_labs_pagination
                margin-top 15px
        #company_pay_record_info.is_mobile
            width 100%
            px('font-size', 14)    // 用於 mobile 設計圖單位
    </style>

5. 效果圖:vuex

相關文章
相關標籤/搜索