Vue三种方式实现全选全不选:@click点击事件方式、computed计算属性方式和watch监听器方式

  1. @click点击事件方式:全选按钮添加点击事件checkAll,点击全选按钮时改变选项的点击状态。下面的选项按钮用一个div包裹,并添加change事件来监听是否全选。
    注:click事件是点击之后,视图里的数据还没有更新到data数据里,所以导致data数据不是最新的状态,需要取反。change事件是视图绑定的数据发生了改变,才会去执行。
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>

<body>
    <div id="app">
        <input type="checkbox" value="OK" v-model="all" @click="checkAll">全选/非全选
        <br>
        <div @change="delegate">
            <div v-for="item,index in hobbyList">
                <input type="checkbox" :value="item.value" :id="item.id|handleID" v-model="checkList">
                <label :for="item.id|handleID">{{item.name}}</label>
            </div>
        </div>
    </div>

    <script src="js/vue.min.js"></script>
    <script type="text/javascript">
        let vm1 = new Vue({
            el: "#app",
            data: {
                hobbyList: [
                    {
                        id: 1,
                        name: "唱歌",
                        value: "sing"
                    },
                    {
                        id: 2,
                        name: "跳舞",
                        value: "dance"
                    },
                    {
                        id: 3,
                        name: "读书",
                        value: "read"
                    },
                    {
                        id: 4,
                        name: "画画",
                        value: "draw"
                    }
                ],
                 // 存储选中的兴趣爱好
                checkList: [],
                // 存储全选按钮的选中状态
                all: []
            },
            filters: {
                handleID(value) {
                    return 'hobby' + value;
                }
            },
            methods: {
                checkAll() {
                    // click事件处理=>下面的判断要取反
                    if (!this.all.includes('OK')) {
                        this.hobbyList.forEach(item => {
                            this.checkList.push(item.value);
                        });
                    }
                    else {
                        this.checkList = [];
                    }
                },
                delegate() {
                    this.all = this.checkList.length === 4 ? ['OK'] : [];
                }
            }
        });
    </script>
</body>

</html>
  1. computed计算属性方式:计算属性必须关联至少一个响应式data数据,响应式数据改变,计算属性就改变。 推荐使用。
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>

<body>
    <div id="app">
        <input type="checkbox" v-model="selected">全选/非全选
        <br>
        <div v-for="item,index in hobbyList">
            <input type="checkbox" :value="item.value" :id="item.id|handleID" v-model="checkList">
            <label :for="item.id|handleID">{{item.name}}</label>
        </div>
    </div>

    <script src="js/vue.min.js"></script>
    <script type="text/javascript">
        let vm1 = new Vue({
            el: "#app",
            data: {
                hobbyList: [
                    {
                        id: 1,
                        name: "唱歌",
                        value: "sing"
                    },
                    {
                        id: 2,
                        name: "跳舞",
                        value: "dance"
                    },
                    {
                        id: 3,
                        name: "读书",
                        value: "read"
                    },
                    {
                        id: 4,
                        name: "画画",
                        value: "draw"
                    }
                ],
                // 存储选中的兴趣爱好
                checkList: []
            },
            filters: {
                handleID(value) {
                    return 'hobby' + value;
                }
            },
            computed: {
                // 存储全选按钮的选中状态
                selected: {
                    get() {
                        return this.checkList.length === this.hobbyList.length ? true : false;
                    },
                    set(value) {
                        // 点击全选按钮时,修改selected的值,value存储的是选中的状态
                        if (value) {
                            this.hobbyList.forEach(item => {
                                this.checkList.push(item.value);
                            });
                            return;
                        }
                        this.checkList = [];
                    }
                }
            },
            methods: {
            }
        });
    </script>
</body>

</html>
  1. watch监听器方式:监听的是data中的数据。
    注:这里不能把全选按钮selected放入watch中监听,因为它的变化是由checkList变化决定的,一起监听会被一直影响,导致结果不对。
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>

<body>
    <div id="app">
        <input type="checkbox" v-model="selected" @change="checkAll">全选/非全选
        <br>
        <div v-for="item,index in hobbyList">
            <input type="checkbox" :value="item.value" :id="item.id|handleID" v-model="checkList">
            <label :for="item.id|handleID">{{item.name}}</label>
        </div>
    </div>

    <script src="js/vue.min.js"></script>
    <script type="text/javascript">
        let vm1 = new Vue({
            el: "#app",
            data: {
                hobbyList: [
                    {
                        id: 1,
                        name: "唱歌",
                        value: "sing"
                    },
                    {
                        id: 2,
                        name: "跳舞",
                        value: "dance"
                    },
                    {
                        id: 3,
                        name: "读书",
                        value: "read"
                    },
                    {
                        id: 4,
                        name: "画画",
                        value: "draw"
                    }
                ],
                // 存储选中的兴趣爱好
                checkList: [],
                // 存储全选按钮的选中状态
                selected: false
            },
            filters: {
                handleID(value) {
                    return 'hobby' + value;
                }
            },
            watch: {
                checkList() {
                    this.selected = this.checkList.length === this.hobbyList.length ? true : false;
                }
            },
            methods: {
                checkAll() {
                    if (this.selected) {
                        this.hobbyList.forEach(item => {
                            this.checkList.push(item.value);
                        });
                        return;
                    }
                    this.checkList = [];
                }
            }
        });
    </script>
</body>

</html>
  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,针对您的需求,我为您提供一个基于Vue3和Element UI的示例代码。您可以参考下面的代码来实现您的功能: ```vue <template> <div> <!-- 全选框 --> <el-checkbox v-model="isAllChecked" @change="handleAllCheckedChange" :indeterminate="isIndeterminate"> {{ isAllChecked ? '已全选' : '全选' }} </el-checkbox> <!-- 卡片列表 --> <el-row> <el-col v-for="card in currentCards" :key="card.id" :span="6"> <el-card> <el-checkbox v-model="checkedIds" :label="card.id" @change="handleCardCheckedChange(card.id)"> {{ card.name }} </el-checkbox> </el-card> </el-col> </el-row> <!-- 分页器 --> <el-pagination :current-page="currentPage" :page-size="pageSize" :total="totalCards" @current-change="handlePageChange" /> </div> </template> <script> import { ref, computed, watch } from 'vue' export default { name: 'MyPage', data() { return { cards: [], // 所有卡片 checkedIds: [], // 已选中的卡片id数组 isAllChecked: false, // 是否已全选 isIndeterminate: false, // 是否半选 currentPage: 1, // 当前页码 pageSize: 12, // 每页显示数量 totalCards: 0, // 总共的卡片数量 } }, computed: { // 当前页的卡片列表 currentCards() { const start = (this.currentPage - 1) * this.pageSize const end = this.currentPage * this.pageSize return this.cards.slice(start, end) }, }, methods: { // 全选框变化时的处理函数 handleAllCheckedChange(value) { this.isAllChecked = value this.checkedIds = value ? this.cards.map(card => card.id) : [] }, // 卡片复选框变化时的处理函数 handleCardCheckedChange(id) { if (this.checkedIds.includes(id)) { // 取消选中 this.checkedIds.splice(this.checkedIds.indexOf(id), 1) this.isAllChecked = false } else { // 选中 this.checkedIds.push(id) if (this.checkedIds.length === this.currentCards.length) { this.isAllChecked = true } } }, // 页面变化时的处理函数 handlePageChange(page) { this.currentPage = page }, }, watch: { // 监听已选中的卡片id数组变化,计算全选框状态和是否半选 checkedIds: { handler(value) { if (value.length === 0) { this.isAllChecked = false this.isIndeterminate = false } else if (value.length === this.currentCards.length) { this.isAllChecked = true this.isIndeterminate = false } else { this.isAllChecked = false this.isIndeterminate = true } }, immediate: true, }, // 监听currentPage变化,重新计算当前页的卡片列表 currentPage: { handler(value) { const start = (value - 1) * this.pageSize const end = value * this.pageSize this.currentCards = this.cards.slice(start, end) }, immediate: true, }, }, created() { // 模拟异步获取卡片数据 setTimeout(() => { this.cards = [ { id: 1, name: '卡片1' }, { id: 2, name: '卡片2' }, { id: 3, name: '卡片3' }, { id: 4, name: '卡片4' }, { id: 5, name: '卡片5' }, { id: 6, name: '卡片6' }, { id: 7, name: '卡片7' }, { id: 8, name: '卡片8' }, { id: 9, name: '卡片9' }, { id: 10, name: '卡片10' }, { id: 11, name: '卡片11' }, { id: 12, name: '卡片12' }, { id: 13, name: '卡片13' }, { id: 14, name: '卡片14' }, { id: 15, name: '卡片15' }, { id: 16, name: '卡片16' }, { id: 17, name: '卡片17' }, { id: 18, name: '卡片18' }, { id: 19, name: '卡片19' }, { id: 20, name: '卡片20' }, { id: 21, name: '卡片21' }, { id: 22, name: '卡片22' }, { id: 23, name: '卡片23' }, { id: 24, name: '卡片24' }, { id: 25, name: '卡片25' }, ] this.totalCards = this.cards.length }, 1000) }, } </script> ``` 以上代码中,我们使用了Element UI的组件来实现全选框、卡片列表和分页器。通过计算属性监听器,我们实现了卡片列表的分页、全选和半选的状态切换,以及保留已选中卡片状态的功能。同时,我们使用了Vue3的Composition API来组织代码,使其更加清晰易懂。 希望以上代码能够对您有所帮助。如果您还有其他问题或需要进一步的帮助,请随时问我。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值