IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> el-table组件封装,持续更新。两种分页模式(滚动、跳转)。render表格渲染。elementui自带分页组件样式更改; -> 正文阅读

[JavaScript知识库]el-table组件封装,持续更新。两种分页模式(滚动、跳转)。render表格渲染。elementui自带分页组件样式更改;

基础使用

  1. 滚动分页模式;leftRollScroll 是一个函数返回当前分页信息
    在这里插入图片描述
    在这里插入图片描述
    高度计算的话,是获取咱们组件的父级的clientHeight,然后减去这个父级内除了表格的高度。例如:
    initTableHeight() {
      this.$nextTick(() =>
        setTimeout(() => {
          const setFn = (key, thisKey) => {
            this[thisKey] = this.$refs[key].clientHeight;
          };
          [
            "left_table_box,leftTH",
            "rlt_table_box,rltTH",
            "rlb_table_box,rlbTH",
            "rr_table_box,rrTH",
          ].forEach((item) => setFn(...item.split(",")));
        }, 0)
      );
    },

在这里插入图片描述

1. lh-table组件代码:

  1. 下面两点都是分页的代码,可以用自己分页组件替换掉它。 具体替换请看本文的 2.1 lh-table组件这块是去掉分页组件的table组件封装,但是里面功能没有以下的全。
  2. 分页组件可以直接用element-ui的分页组件,但是table组件内涉及分页的得换一换
<template>
  <div class="lh-table" v-loading="loadings">
    <el-table
      ref="LHTable"
      border
      :class="['table-msg color-table ', className]"
      :stripe="stripe"
      :highlight-current-row="highlight"
      :size="size"
      :data="tableDataCheck"
      :max-height="height"
      :row-class-name="rowClassName"
      :style="{ height: pageination ? 'calc(100% - 60px)' : '100%' }"
      :span-method="spanMethod"
      @cell-dblclick="(param) => $emit('cell-dblclick', param)"
      @cell-click="(param) => $emit('cell-click', param)"
      @selection-change="(param) => $emit('selection-change', param)"
    >
      <template
        v-for="colConfig in tableConfig.filter(
          (item) => item.lhDisplay !== 'none'
        )"
      >
        <el-table-column
          v-bind="colConfig"
          align="center"
          :key="colConfig['id']"
          v-if="!colConfig.type"
        >
          <template slot="header" slot-scope="scope">
            <render
              v-if="colConfig.headerRender"
              :render="colConfig.headerRender"
              :row="scope.row"
              :index="scope.$index"
              :column="colConfig"
              :key="colConfig['id']"
            />
            <span v-else>{{ colConfig.label }}</span>
          </template>
          <template slot-scope="scope">
            <render
              v-if="colConfig.render"
              :render="colConfig.render"
              :row="scope.row"
              :index="scope.$index"
              :column="colConfig"
              :key="colConfig['id']"
            />
            <span v-else>{{
              scope.row[colConfig && colConfig.prop] || "-"
            }}</span>
          </template>
        </el-table-column>
        <el-table-column v-else :key="colConfig['id']" v-bind="colConfig" align="center" />
      </template>
    </el-table>
    <LhPageination
      small
      background
      v-if="pageination"
      :layout="pageLayout"
      :page-size="pageSize"
      :pageSizes="pageSizes"
      :current-page="currentPage"
      :total="totalNum"
      @current-change="handleCurrentChange"
      @size-change="handleSizeChange"
    />
  </div>
</template>

<script>
import LhPageination from "@/components/anHui-new/LhPagination/index.js";

const pageination = {
  currentPage: 1,
  totalNum: 0,
  pageSize: 30,
  pageSizes: [10, 15, 20, 30, 50, 100],
};
const render = {
  functional: true,
  props: {
    row: Object,
    render: Function,
    index: Number,
    column: {
      type: Object,
      default: null,
    },
  },
  render: (h, data) => {
    const params = {
      row: data.props.row,
      index: data.props.index,
    };
    if (data.props.column) params.column = data.props.column;
    return data.props.render(h, params);
  },
};
export default {
  name: "LhTable",
  props: {
    // table配置信息
    tableConfig: {
      type: Array,
      default: () => [
        {
          label: "序号",
          type: "index",
          width: "80px",
        },
        {
          label: "名字",
          prop: "name",
        },
        {
          label: "公司名称",
          prop: "cmyName",
        },
      ],
    },
    // table的一些信息
    p_tableInfo: {
      type: Object,
      default: () => ({
        // table的数据源
        tableData: [],
        // 是否开启加载等待
        loading: false,
      }),
    },
    // 分页配置
    p_pageinationInfo: {
      type: Object,
      default: () => pageination,
    },
    // 高度设置 具体参考官网的
    height: {
      type: Number,
      default: 250,
    },
    // 是否显示分页器
    pageination: {
      type: Boolean,
      default: true,
    },
    // 官网的
    size: {
      type: String,
      default: "small",
    },
    // 官网的:highlight-current-row;高亮行
    highlight: {
      type: Boolean,
      default: true,
    },
    // 给el-table添加class
    className: {
      type: String,
      default: "",
    },
    // 斑马条
    stripe: {
      type: Boolean,
      default: true,
    },
    // 行样式回调函数  (官网的)
    rowClassName: {
      type: Function,
    },
    // 单元格合并回调函数(官网的)
    spanMethod: {
      type: Function,
    },
    // 是否开启滚动加载功能
    rollScroll: {
      type: Function | Boolean,
      default: false,
    },
  },
  // 如果只有一个表格直接 provide/inject 多个表格则组件传参方式
  inject: {
    // 分页信息;格式等同于组件传参 p_pageinationInfo
    pageinationInfo: {
      default: () => ({
        // 唯一标识
        lh_spe_flag: true,
      }),
    },
    // 表格信息;格式等同于组件传参 p_tableInfo
    tableInfo: {
      default: () => ({
        // 唯一标识
        lh_spe_flag: true,
      }),
    },
  },
  watch: {
    // 监听表格 总数 变化
    "pageinationInfo.totalNum"(n) {
      this.totalNum = n;
    },
    // 监听表格 当前页 变化
    "pageinationInfo.currentPage"(n) {
      this.currentPage = n;
    },
  },
  computed: {
    // 表格开启 加载等待
    loadings() {
      return this.tableInfo.lh_spe_flag
        ? this.p_tableInfo.loading || false
        : (this.tableInfo && this.tableInfo.loading) || false;
    },
    // 判断表格的渲染内容( tableData )选择哪一个【判断是provide/inject还是组件传参模式】
    tableDataCheck() {
      return this.tableInfo.lh_spe_flag
        ? this.p_tableInfo.tableData
        : this.tableInfo.tableData;
    },
    // 判断分页信息选择哪一个
    pageinationInfoCheck() {
      return this.pageinationInfo.lh_spe_flag
        ? this.p_pageinationInfo
        : this.pageinationInfo;
    },
  },
  components: {
    // 其实就是element的分页组件,只不过我把它样式重构了一下
    LhPageination,
    // table配置里面使用h函数的渲染原理
    render,
  },
  data() {
    const { currentPage, totalNum, pageSize, pageSizes, pageLayout } = this
      .pageinationInfo.lh_spe_flag
      ? this.p_pageinationInfo
      : this.pageinationInfo || pageination;
    return {
      pageLayout: pageLayout || [
        ["total", "sizes"],
        ["prev", "pager", "next", "jumper"],
      ],
      currentPage, //当前页数
      totalNum, //数据总条数
      pageSize, //每页数据的条数
      pageSizes, //选择每页的条数
    };
  },
  created() {
    // 是否开启滚动加载
    this.rollScroll && this.$nextTick(() => this.addRollListener());
  },
  methods: {
    handleCurrentChange(v) {
      this.$emit("current-change", v);
    },
    handleSizeChange(v) {
      this.$emit("size-change", v);
    },
    // 添加滚动加载
    addRollListener() {
      const dom = this.$refs.LHTable.bodyWrapper;
      dom.addEventListener("scroll", () => {
        const { pageSize, currentPage, totalNum } = this.pageinationInfoCheck;
        const all = pageSize * currentPage;
        const scrollTop = dom.scrollTop;
        // 变量windowHeight是可视区的高度
        const windowHeight = dom.clientHeight || dom.clientHeight;
        // 变量scrollHeight是滚动条的总高度
        const scrollHeight = dom.scrollHeight || dom.scrollHeight;
        if (scrollTop + windowHeight + 1 > scrollHeight) {
          if (all < totalNum) {
            // 以下两个都是 通知父组件
            this.rollScroll(this.pageinationInfoCheck);
            // 这里使用到了 lodash 的 _.isBoolean 方法,判断是否是Boolean;可以自己写个或者去掉;
            _.isBoolean(this.rollScroll) &&
              this.$emit("roll-scroll", this.pageinationInfoCheck);
          }
        }
      });
    },
  },
};
</script>

<style lang="less" scoped>
.lh-table {
  position: relative;
  height: 100%;
  .table-msg {
    position: absolute;
  }
  .el-pagination {
    position: absolute;
    bottom: 0;
  }
  /deep/.el-table__body-wrapper {
    height: 100% !important;
    .el-table__empty-block {
      min-height: 10px !important;
    }
  }
}
</style>

2. LhPagination.js

  1. 其实就是element-ui的el-Pagination,但是我根据需求更改了一下样式配置;
import Pager from './pager.vue';
import Locale from 'element-ui/src/mixins/locale';
const valueEquals = (a, b) => {
  // see: https://stackoverflow.com/questions/3115982/how-to-check-if-two-arrays-are-equal-with-javascript
  if (a === b) return true;
  if (!(a instanceof Array)) return false;
  if (!(b instanceof Array)) return false;
  if (a.length !== b.length) return false;
  for (let i = 0; i !== a.length; ++i) {
    if (a[i] !== b[i]) return false;
  }
  return true;
};
export default {
  name: 'ElPagination',

  props: {
    pageSize: {
      type: Number,
      default: 10
    },

    small: Boolean,

    total: Number,

    pageCount: Number,

    pagerCount: {
      type: Number,
      validator(value) {
        return (value | 0) === value && value > 4 && value < 22 && (value % 2) === 1;
      },
      default: 7
    },

    currentPage: {
      type: Number,
      default: 1
    },

    layout: {
      default: 'prev, pager, next, jumper, ->, total'
    },

    pageSizes: {
      type: Array,
      default() {
        return [10, 20, 30, 40, 50, 100];
      }
    },

    popperClass: String,

    prevText: String,

    nextText: String,

    background: Boolean,

    disabled: Boolean,

    hideOnSinglePage: Boolean,
    lhType: Boolean // true: 两端对齐, false:居中
  },

  data() {
    return {
      internalCurrentPage: 1,
      internalPageSize: 0,
      lastEmittedPage: -1,
      userChangePageSize: false
    };
  },

  render(h) {
    const layout = this.layout;
    if (!layout) return null;
    if (this.hideOnSinglePage && (!this.internalPageCount || this.internalPageCount === 1)) return null;
    const TEMPLATE_MAP = {
      prev: <prev></prev>,
      jumper: <jumper></jumper>,
      pager: <pager currentPage={ this.internalCurrentPage } pageCount={ this.internalPageCount } pagerCount={ this.pagerCount } on-change={ this.handleCurrentChange } disabled={ this.disabled }></pager>,
      next: <next></next>,
      sizes: <sizes pageSizes={ this.pageSizes }></sizes>,
      slot: <slot>{ this.$slots.default ? this.$slots.default : '' }</slot>,
      total: <total></total>
    };
    const content = layout || [['total', 'sizes'],['prev', 'pager', 'next', 'jumper']];
    let template = <div class={['el-pagination lh-pagination', this.lhType?'flex-jcenter':'flex-between', {
      'is-background': this.background,
      'el-pagination--small': this.small,
    }] }>
        {
          content.map(item=>(
            <div class='lh-flex'>
              {
                item.map(value=>(
                  TEMPLATE_MAP[value]
                ))
              }
            </div>
          ))
        }
    </div>;
    const rightWrapper = <div class="el-pagination__rightwrapper"></div>;
    let haveRightWrapper = false;

    template.children = template.children || [];
    rightWrapper.children = rightWrapper.children || [];

    if (haveRightWrapper) {
      template.children.unshift(rightWrapper);
    }

    return template;
  },

  components: {
    Prev: {
      render(h) {
        return (
          <button
            type="button"
            class="btn-prev"
            disabled={ this.$parent.disabled || this.$parent.internalCurrentPage <= 1 }
            on-click={ this.$parent.prev }>
            {
              this.$parent.prevText
                ? <span>{ this.$parent.prevText }</span>
                : <i class="el-icon el-icon-arrow-left"></i>
            }
          </button>
        );
      }
    },

    Next: {
      render(h) {
        return (
          <button
            type="button"
            class="btn-next"
            disabled={ this.$parent.disabled || this.$parent.internalCurrentPage === this.$parent.internalPageCount || this.$parent.internalPageCount === 0 }
            on-click={ this.$parent.next }>
            {
              this.$parent.nextText
                ? <span>{ this.$parent.nextText }</span>
                : <i class="el-icon el-icon-arrow-right"></i>
            }
          </button>
        );
      }
    },

    Sizes: {
      mixins: [Locale],

      props: {
        pageSizes: Array
      },

      watch: {
        pageSizes: {
          immediate: true,
          handler(newVal, oldVal) {
            if (valueEquals(newVal, oldVal)) return;
            if (Array.isArray(newVal)) {
              this.$parent.internalPageSize = newVal.indexOf(this.$parent.pageSize) > -1
                ? this.$parent.pageSize
                : this.pageSizes[0];
            }
          }
        }
      },

      render(h) {
        return (
          <div class="el-pagination__sizes">
            每页显示&ensp;
            <el-select
              value={ this.$parent.internalPageSize }
              popperClass={ this.$parent.popperClass || '' }
              size="mini"
              on-input={ this.handleChange }
              disabled={ this.$parent.disabled }>
              {
                this.pageSizes.map(item =>
                  <el-option
                    value={ item }
                    label={ item }>
                  </el-option>
                )
              }
            </el-select>
            &ensp;条记录
          </div>
        );
      },
      methods: {
        handleChange(val) {
          if (val !== this.$parent.internalPageSize) {
            this.$parent.internalPageSize = val = parseInt(val, 10);
            this.$parent.userChangePageSize = true;
            this.$parent.$emit('update:pageSize', val);
            this.$parent.$emit('size-change', val);
          }
        }
      }
    },

    Jumper: {
      mixins: [Locale],
      data() {
        return {
          userInput: null
        };
      },

      watch: {
        '$parent.internalCurrentPage'() {
          this.userInput = null;
        }
      },

      methods: {
        handleKeyup({ keyCode, target }) {
          // Chrome, Safari, Firefox triggers change event on Enter
          // Hack for IE: https://github.com/ElemeFE/element/issues/11710
          // Drop this method when we no longer supports IE
          if (keyCode === 13) {
            this.handleChange(target.value);
          }
        },
        handleInput(value) {
          this.userInput = value;
        },
        handleChange(value) {
          this.$parent.internalCurrentPage = this.$parent.getValidCurrentPage(value);
          this.$parent.emitChange();
          this.userInput = null;
        }
      },

      render(h) {
        return (
          <span class="el-pagination__jump">
            { this.t('el.pagination.goto') }
            <el-input
              class="el-pagination__editor is-in-pagination"
              min={ 1 }
              max={ this.$parent.internalPageCount }
              value={ this.userInput !== null ? this.userInput : this.$parent.internalCurrentPage }
              type="number"
              disabled={ this.$parent.disabled }
              nativeOnKeyup={ this.handleKeyup }
              onInput={ this.handleInput }
              onChange={ this.handleChange }/>
            { this.t('el.pagination.pageClassifier') }
          </span>
        );
      }
    },

    Total: {
      mixins: [Locale],

      render(h) {
        return (
          typeof this.$parent.total === 'number'
            ? <span class="el-pagination__total">总共 { this.$parent.total } 条记录</span>
            : ''
        );
      }
    },

    Pager
  },

  methods: {
    handleCurrentChange(val) {
      this.internalCurrentPage = this.getValidCurrentPage(val);
      this.userChangePageSize = true;
      this.emitChange();
    },

    prev() {
      if (this.disabled) return;
      const newVal = this.internalCurrentPage - 1;
      this.internalCurrentPage = this.getValidCurrentPage(newVal);
      this.$emit('prev-click', this.internalCurrentPage);
      this.emitChange();
    },

    next() {
      if (this.disabled) return;
      const newVal = this.internalCurrentPage + 1;
      this.internalCurrentPage = this.getValidCurrentPage(newVal);
      this.$emit('next-click', this.internalCurrentPage);
      this.emitChange();
    },

    getValidCurrentPage(value) {
      value = parseInt(value, 10);

      const havePageCount = typeof this.internalPageCount === 'number';

      let resetValue;
      if (!havePageCount) {
        if (isNaN(value) || value < 1) resetValue = 1;
      } else {
        if (value < 1) {
          resetValue = 1;
        } else if (value > this.internalPageCount) {
          resetValue = this.internalPageCount;
        }
      }

      if (resetValue === undefined && isNaN(value)) {
        resetValue = 1;
      } else if (resetValue === 0) {
        resetValue = 1;
      }

      return resetValue === undefined ? value : resetValue;
    },

    emitChange() {
      this.$nextTick(() => {
        if (this.internalCurrentPage !== this.lastEmittedPage || this.userChangePageSize) {
          this.$emit('current-change', this.internalCurrentPage);
          this.lastEmittedPage = this.internalCurrentPage;
          this.userChangePageSize = false;
        }
      });
    }
  },

  computed: {
    internalPageCount() {
      if (typeof this.total === 'number') {
        return Math.max(1, Math.ceil(this.total / this.internalPageSize));
      } else if (typeof this.pageCount === 'number') {
        return Math.max(1, this.pageCount);
      }
      return null;
    }
  },

  watch: {
    currentPage: {
      immediate: true,
      handler(val) {
        this.internalCurrentPage = this.getValidCurrentPage(val);
      }
    },

    pageSize: {
      immediate: true,
      handler(val) {
        this.internalPageSize = isNaN(val) ? 10 : val;
      }
    },

    internalCurrentPage: {
      immediate: true,
      handler(newVal) {
        this.$emit('update:currentPage', newVal);
        this.lastEmittedPage = -1;
      }
    },

    internalPageCount(newVal) {
      /* istanbul ignore if */
      const oldPage = this.internalCurrentPage;
      if (newVal > 0 && oldPage === 0) {
        this.internalCurrentPage = 1;
      } else if (oldPage > newVal) {
        this.internalCurrentPage = newVal === 0 ? 1 : newVal;
        this.userChangePageSize && this.emitChange();
      }
      this.userChangePageSize = false;
    }
  }
};

3.pager.vue

  1. element-ui的组件
<template>
  <ul @click="onPagerClick" class="el-pager">
    <li
      :class="{ active: currentPage === 1, disabled }"
      v-if="pageCount > 0"
      class="number">1</li>
    <li
      class="el-icon more btn-quickprev"
      :class="[quickprevIconClass, { disabled }]"
      v-if="showPrevMore"
      @mouseenter="onMouseenter('left')"
      @mouseleave="quickprevIconClass = 'el-icon-more'">
    </li>
    <li
      v-for="pager in pagers"
      :key="pager"
      :class="{ active: currentPage === pager, disabled }"
      class="number">{{ pager }}</li>
    <li
      class="el-icon more btn-quicknext"
      :class="[quicknextIconClass, { disabled }]"
      v-if="showNextMore"
      @mouseenter="onMouseenter('right')"
      @mouseleave="quicknextIconClass = 'el-icon-more'">
    </li>
    <li
      :class="{ active: currentPage === pageCount, disabled }"
      class="number"
      v-if="pageCount > 1">{{ pageCount }}</li>
  </ul>
</template>

<script type="text/babel">
  export default {
    name: 'ElPager',

    props: {
      currentPage: Number,

      pageCount: Number,

      pagerCount: Number,

      disabled: Boolean
    },

    watch: {
      showPrevMore(val) {
        if (!val) this.quickprevIconClass = 'el-icon-more';
      },

      showNextMore(val) {
        if (!val) this.quicknextIconClass = 'el-icon-more';
      }
    },

    methods: {
      onPagerClick(event) {
        const target = event.target;
        if (target.tagName === 'UL' || this.disabled) {
          return;
        }

        let newPage = Number(event.target.textContent);
        const pageCount = this.pageCount;
        const currentPage = this.currentPage;
        const pagerCountOffset = this.pagerCount - 2;

        if (target.className.indexOf('more') !== -1) {
          if (target.className.indexOf('quickprev') !== -1) {
            newPage = currentPage - pagerCountOffset;
          } else if (target.className.indexOf('quicknext') !== -1) {
            newPage = currentPage + pagerCountOffset;
          }
        }

        /* istanbul ignore if */
        if (!isNaN(newPage)) {
          if (newPage < 1) {
            newPage = 1;
          }

          if (newPage > pageCount) {
            newPage = pageCount;
          }
        }

        if (newPage !== currentPage) {
          this.$emit('change', newPage);
        }
      },

      onMouseenter(direction) {
        if (this.disabled) return;
        if (direction === 'left') {
          this.quickprevIconClass = 'el-icon-d-arrow-left';
        } else {
          this.quicknextIconClass = 'el-icon-d-arrow-right';
        }
      }
    },

    computed: {
      pagers() {
        const pagerCount = this.pagerCount;
        const halfPagerCount = (pagerCount - 1) / 2;

        const currentPage = Number(this.currentPage);
        const pageCount = Number(this.pageCount);

        let showPrevMore = false;
        let showNextMore = false;

        if (pageCount > pagerCount) {
          if (currentPage > pagerCount - halfPagerCount) {
            showPrevMore = true;
          }

          if (currentPage < pageCount - halfPagerCount) {
            showNextMore = true;
          }
        }

        const array = [];

        if (showPrevMore && !showNextMore) {
          const startPage = pageCount - (pagerCount - 2);
          for (let i = startPage; i < pageCount; i++) {
            array.push(i);
          }
        } else if (!showPrevMore && showNextMore) {
          for (let i = 2; i < pagerCount; i++) {
            array.push(i);
          }
        } else if (showPrevMore && showNextMore) {
          const offset = Math.floor(pagerCount / 2) - 1;
          for (let i = currentPage - offset ; i <= currentPage + offset; i++) {
            array.push(i);
          }
        } else {
          for (let i = 2; i < pageCount; i++) {
            array.push(i);
          }
        }

        this.showPrevMore = showPrevMore;
        this.showNextMore = showNextMore;

        return array;
      }
    },

    data() {
      return {
        current: null,
        showPrevMore: false,
        showNextMore: false,
        quicknextIconClass: 'el-icon-more',
        quickprevIconClass: 'el-icon-more'
      };
    }
  };
</script>

<style lang="less" scoped>
</style>
  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-06-29 18:55:53  更:2022-06-29 18:56:58 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 9:58:22-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码