当前位置: 首页 > article >正文

h5使用better scroll实现左右列表联动

安装vant^2.12.38和better-scroll^2.5.1

<template>
  <div>
    <div class="dept-filter" @click="showFilter">
      <div class="name">{{ params.name }}</div>
      <img class="img" src="@/assets/images/index/filter.png" />
    </div>
    <van-popup v-model="show" position="bottom" :style="{ height: '70vh' }">
      <div class="title-box">
        <div class="title">店铺列表</div>
        <img class="close" src="@/assets/images/index/close.png" @click="show = false" />
      </div>
      <div class="content-box">
        <div class="left">
          <div
            @click="setDept(item, index)"
            class="item"
            :class="currentDeptIndex === index ? 'active' : ''"
            v-for="(item, index) in list"
            :key="item.deptId"
          >
            <div class="line"></div>
            <div class="dept">{{ item.deptName }}</div>
          </div>
        </div>
        <div class="right" @mousewheel="dataScroll" @touchmove="dataScroll">
          <div
            class="item right-item"
            v-for="(item, index) in list"
            :key="item.deptId"
            :ref="'id' + item.deptId"
          >
            <div class="dept weight">{{ item.deptName }}</div>
            <div class="store" :ref="'dept' + item.deptId" @click="setAllStore(item, index)">
              <div
                class="name"
                :class="params.deptId === item.deptId && params.storeId === null ? 'active' : ''"
              >
                所有店铺
              </div>
              <img
                v-if="params.deptId === item.deptId && params.storeId === null"
                class="checked"
                src="@/assets/images/index/checked.png"
              />
              <!-- 占位 -->
              <div v-else class="checked"></div>
            </div>
            <div class="dept" v-if="item.storeList.length !== 0">
              部门下店铺({{ item.storeList.length }})
            </div>
            <div
              class="store"
              @click="setStore(item, child, index)"
              v-for="child in item.storeList"
              :key="child.storeId"
              :ref="'dept' + item.deptId + child.storeId"
            >
              <div class="name" :class="params.storeId === child.storeId ? 'active' : ''">
                {{ child.storeName }}
              </div>
              <img
                v-if="params.storeId === child.storeId"
                class="checked"
                src="@/assets/images/index/checked.png"
              />
              <div v-else class="checked"></div>
            </div>
          </div>
        </div>
      </div>
    </van-popup>
  </div>
</template>
<script>
export default {
  name: 'DeptFilter1',
  data() {
    return {
      show: false,
      params: {
        deptId: 1,
        storeId: null,
        name: '所有部门'
      },
      currentDeptIndex: 0,
      list: [
        {
          deptId: 1,
          deptName: '所有部门',
          storeList: []
        },
        {
          deptId: 2,
          deptName: '软件研发部',
          storeList: []
        },
        {
          deptId: 3,
          deptName: '事业部',
          storeList: [
            { storeId: 1, storeName: '大药房官方旗舰店' },
            { storeId: 2, storeName: '拼多多事业店-拼多多事业店' },
            { storeId: 3, storeName: '拼多多事业店-拼多多事业店' }
          ]
        },
        {
          deptId: 4,
          deptName: '技术部',
          storeList: [
            { storeId: 13, storeName: '大药房官方旗舰店' },
            { storeId: 4, storeName: '拼多多事业店-拼多多事业店' },
            {
              storeId: 5,
              storeName:
                '拼多多事业部拼多多拼多多事业部拼多多拼多多事业部拼多多拼多多事业部拼多多拼多多事业部拼多多'
            },
            { storeId: 6, storeName: '拼多多事业店-拼多多事业店' },
            { storeId: 7, storeName: '拼多多事业店-拼多多事业店' },
            { storeId: 8, storeName: '拼多多事业店-拼多多事业店' },
            { storeId: 9, storeName: '拼多多事业店-拼多多事业店' },
            { storeId: 10, storeName: '拼多多事业店-拼多多事业店' }
          ]
        },
        {
          deptId: 5,
          deptName: '人事部',
          storeList: [
            { storeId: 11, storeName: '大药房官方旗舰店' },
            { storeId: 12, storeName: '拼多多事业店-拼多多事业店' }
          ]
        }
      ],
      initTop: 0
    }
  },
  methods: {
    setDept(item, index) {
      this.currentDeptIndex = index
      this.$refs['id' + item.deptId][0].scrollIntoView({ behavior: 'smooth' })
    },
    setAllStore(item, index) {
      this.params.deptId = item.deptId
      this.params.storeId = null
      this.params.name = item.deptName
      this.currentDeptIndex = index
      this.show = false
      console.log(this.params)
      this.$emit('getParams', this.params)
    },
    setStore(item, child, index) {
      this.params.storeId = child.storeId
      this.params.deptId = item.deptId
      this.params.name = child.storeName
      this.currentDeptIndex = index
      this.show = false
      console.log(this.params)
      this.$emit('getParams', this.params)
    },
    showFilter() {
      this.show = true
      this.setInitTop()
      this.topTop()
    },
    //设置选中元素置顶
    topTop() {
      this.$nextTick(() => {
        let ref = 'dept' + this.params.deptId
        if (this.params.storeId !== null) {
          ref = 'dept' + this.params.deptId + this.params.storeId
        }
        this.$refs[ref][0].scrollIntoView({ behavior: 'smooth' })
      })
    },
    //获取初始高度
    setInitTop() {
      this.$nextTick(() => {
        setTimeout(() => {
          const element = document.getElementsByClassName('title-box')[0]
          this.initTop = element.getBoundingClientRect().top + element.clientHeight
        }, 400)
      })
    },
    dataScroll() {
      const element = document.getElementsByClassName('right-item')
      this.currentDeptIndex = 0
      for (let i = 0; i < element.length; i++) {
        if (element[i].getBoundingClientRect().top <= this.initTop) {
          this.currentDeptIndex = i
        }
      }
    }
  }
}
</script>

<style scoped lang="scss">
.dept-filter {
  display: flex;
  align-items: center;
  background: #191e28;
  color: #fff;
  height: 30px;
  .name {
    font-weight: 500;
    font-size: 13px;
    line-height: 13px;
    white-space: nowrap;
    text-overflow: ellipsis;
    overflow: hidden;
    margin: 0 5px 0 10px;
    max-width: 92px;
  }
  .img {
    width: 12px;
    height: 12px;
  }
}
::v-deep .van-popup {
  border-radius: 10px 10px 0 0;
  background: linear-gradient(
    179.71deg,
    rgba(23, 28, 37, 1) 0.75%,
    rgba(49, 55, 65, 1) 49.55%,
    rgba(23, 28, 37, 1) 98.76%
  );
  overflow-y: hidden;
  color: #ffffff;
}
.title-box {
  font-weight: 600;
  font-size: 15px;
  padding: 16px 20px;
  position: relative;
  .title {
    text-align: center;
  }
  .close {
    width: 30px;
    height: 30px;
    position: absolute;
    right: 20px;
    top: 12px;
  }
}
.content-box {
  display: flex;
  .left {
    width: 108px;
    height: calc(70vh - 52px);
    overflow-y: scroll;
    .item {
      height: 48px;
      display: flex;
      align-items: center;
      .line {
        width: 4px;
        height: 16px;
      }
      .dept {
        padding: 0 12px 0 8px;
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;
        font-size: 14px;
      }
    }
    .active {
      .dept {
        background: linear-gradient(
          136.52deg,
          rgba(4, 189, 248, 0.88) 4.9%,
          rgba(8, 252, 223, 0.88) 132.73%
        );
        -webkit-background-clip: text !important;
        -webkit-text-fill-color: transparent;
      }
      .line {
        background: linear-gradient(
          136.52deg,
          rgba(4, 189, 248, 0.88) 4.9%,
          rgba(8, 252, 223, 0.88) 132.73%
        );
      }
    }
  }

  .right {
    width: calc(100vw - 108px);
    height: calc(70vh - 52px);
    overflow-y: scroll;
    .item {
      border-top: 2px solid rgba(255, 255, 255, 0.5);
      padding: 0 16px 0 16px;
      &:first-child {
        border: none;
      }
      .dept {
        color: #bfbfbf;
        font-size: 12px;
        height: 25px;
        display: flex;
        align-items: flex-end;
      }
      .weight {
        font-weight: 500;
      }
      .store {
        display: flex;
        align-items: center;
        justify-content: space-between;
        padding: 12px 0;
        font-size: 14px;
        .name {
          width: calc(100vw - 170px);
        }
        .active {
          background: linear-gradient(
            136.52deg,
            rgba(4, 189, 248, 0.88) 4.9%,
            rgba(8, 252, 223, 0.88) 132.73%
          );
          -webkit-background-clip: text !important;
          -webkit-text-fill-color: transparent;
        }
        .checked {
          width: 20px;
          height: 20px;
          margin-left: 10px;
        }
      }
    }
  }
}
</style>

展示效果


http://www.kler.cn/a/501684.html

相关文章:

  • SpringCloud系列教程:微服务的未来(十一)服务注册、服务发现、OpenFeign快速入门
  • JAVA安全编码规范
  • virtual box虚拟机误删Python3.6后导致UBUNTU18.04开机无UI界面(进不了desktop)的解决方法
  • 《跟我学Spring Boot开发》系列文章索引❤(2025.01.09更新)
  • C#,图论与图算法,有向图(Direct Graph)广度优先遍历(BFS,Breadth First Search)算法与源程序
  • 利用Java爬虫获取淘宝商品描述item_get_descAPI接口
  • c++ haru生成pdf输出文本实例
  • Java 如何传参xml调用接口获取数据
  • 后端开发 Springboot整合Redis Spring Data Redis 模板
  • 【大数据】数据科学导论---数据科学的概念
  • 状态模式详解与应用
  • 人工智能之基于阿里云快速搭建语音合成
  • Seata的部署与微服务集成
  • pytorch张量的new_zeros方法介绍
  • python-leetcode-有效的数独
  • Java 将RTF文档转换为Word、PDF、HTML、图片
  • uniapp使用scss mixin抽离css常用的公共样式
  • PyTorch reshape函数介绍
  • 使用Cilium/eBPF实现大规模云原生网络和安全
  • MongoDB 删除集合
  • nginx增加新模块
  • Python orjson ujson有什么区别?
  • 【DevOps】Jenkins使用Pipeline构建java代码
  • AIGC是什么?怎么用?简单三步ToDesk云电脑快速用
  • 前端学习-焦点事件以及键盘事件与典型案例(二十五)
  • Node.js——http 模块(二)