首页 > 编程 > JavaScript > 正文

使用vue.js制作分页组件

2019-11-20 09:35:59
字体:
来源:转载
供稿:网友

学习了vue.js一段时间,拿它来做2个小组件,练习一下。
我这边是用webpack进行打包,也算熟悉一下它的运用。
源码放在文末的 github 地址上。

首先是index.html

<!DOCTYPE html><html><head> <title>Page</title> <style type="text/css">  * {   margin: 0;   padding: 0;   font-family: 'Open Sans', Arial, sans-serif;  }  .contianer {   width: 50%;   height: auto;   margin: 20px auto;  }  article {   margin-bottom: 50px;  } </style></head><body> <div class='contianer'>  <article>   文章内容...  </article>  <div id='main'>   <app></app>    </div> </div> <script type="text/javascript" src='bundle.js'></script></body></html>

我将 app这个组件放在 <div id='main'></div> 内
通过webpack打包后,入口的js文件是entry.js,用来引入app.vue组件
entry.js

let Vue = require('vue');import App from './components/app';let app_vue = new Vue({ el: '#main', components: {  app: App }});

接下来看下这个app组件

<style type="text/css" scoped> </style><template> <comment :cur-page-index="curPageIndex" :each-page-size="eachPageSize" :comment-url="commentUrl"   :comment-params="commentParams" :comment-is-sync="commentIsSync">   </comment> <page :cur-page-index.sync="curPageIndex" :each-page-size="eachPageSize" :page-url="pageUrl"   :page-params="pageParams" :page-is-sync="pageIsSync"> </page></template> <script type="text/javascript"> import Comment from './comment'; import Page from './page'; export default {  data () {   return {    curPageIndex: 1,    eachPageSize: 7,   }  },  components: {   comment: Comment,   page: Page  }, }</script>

它有2个子组件,分别是comment.vue和page.vue,通过动态绑定数据,进行父子间组件通信,我是这样认为的,对于 当前在第几页 应当由 page.vue传递给app.vue,所以这里我们使用 双向绑定,其余的如 params, url, isSync,即向后台请求数据的东西以及是否同步或异步操作<当然,这里我还没有测试过后台数据,目前是直接js生成静态数据>。

接下来,看下comment.vue评论组件

<style type="text/css" scoped> .comt-mask {  opacity: 0.5; } .comt-title {   } .comt-line {  width: 100%;  height: 2px;  background-color: #CCC;  margin: 10px 0; } .comt-wrap {   } .comt-user {  float: left; } .comt-img {  width: 34px;  height: 34px;  border-radius: 17px; } .comt-context {  margin: 0 0 0 60px; } .comt-name {  color: #2B879E;  margin-bottom: 10px;  font-size: 18px; }</style><template> <div v-if="hasComment" :class="{'comt-mask': loading}">  <h3 class='comt-title'>{{ totalCommentCount }} 条评论</h3>  <div class="comt-line"></div>  <div class="comt-wrap" v-for="comment of commentArr">   <div class="comt-user">    <img src='{{ comment.avatar }}' class="comt-img"/>   </div>   <div class="comt-context">    <p class="comt-name">{{ comment.name }}</p>       <p>     {{ comment.context }}    </p>   </div>   <div class="comt-line"></div>  </div> </div></template><script type="text/javascript"> import {getCommentData, getTotalCommentCount} from './getData'; export default {  props: {   curPageIndex: {    type: Number,    default: 1,   },   eachPageSize: {    type: Number,    default: 7,   },   commentUrl: {    type: String,    default: '',   },   commentParams: {    type: Object,    default: null,   },   commentIsSync: {    type: Boolean,    default: true,   },  },  data () {   return {    totalCommentCount: 0,    hasComment: false,    loading: true,      }  },  computed: {   commentArr () {    this.loading = true;    let res = getCommentData(this.commentUrl, this.commentParams, this.commentIsSync, this.curPageIndex, this.eachPageSize);    this.loading = false;    return res;   },  },  created () {   let cnt = getTotalCommentCount(this.commentUrl, this.commentParams);   this.totalCommentCount = cnt;   this.hasComment = cnt > 0;  } }</script>

这里的 getData.js 将在下面提到,是我们获取数据的位置。
loading: 本意是在跳转页码加载评论时,对于当前评论加载0.5的透明度的遮罩,然后ajax通过它的回调函数来取消遮罩,现在这样就不能实现了,只能强行写下,然而是没有用的..
hasComment: comment组件第一次加载的时候,我们就去请求获得总共的数据长度,如果没有数据,则不显示comment组件布局内容
・curPageIndex・: 通过父组件app传递下来,使用的是props
这些数据,我们都设置一个默认值与类型比较好。
page.vue

<style type="text/css" scoped> .page {  text-align: center;  margin: 30px; } .page-btn {  color: gray;  background-color: white;  border: white;  width: 30px;  height: 30px;  margin: 5px;  font-size: 18px;  outline: none; } .page-btn-link {  cursor: Crosshair; } .page-btn-active {  border: 1px solid gray;  border-radius: 15px; }</style><template> <div class="page">  <button v-for="pageIndex of pageArr" track-by='$index' :class="{'page-btn': true, 'page-btn-active':    this.curPageIndex === pageIndex, 'page-btn-link': checkNum(pageIndex)}"    @click="clickPage(pageIndex)" >    {{ pageIndex }}  </button>   </div></template><script type="text/javascript"> import {getTotalPageCount} from './getData'; export default {  props: {   totalPageCount: {    type: Number,    default: 0,   },   curPageIndex: {    type: Number,    default: 1,   },   eachPageSize: {    type: Number,    default: 7,   },   pageAjcn: {    type: Number,    default: 4,   },   pageUrl: {    type: String,    default: '',   },   pageParams: {    type: Object,    default: null,   },   pageIsSync: {    type: Boolean,    default: true,   }        },  data () {   return {   }  },  computed: {   pageArr () {    let st = 1,     end = this.totalPageCount,     cur = this.curPageIndex,     ajcn = this.pageAjcn,     arr = [],     left = Math.floor(ajcn / 2),     right = ajcn - left;         if (end == 0 || cur == 0) {     return arr;    } else {     console.log(st, end, cur, left, right);     arr.push(st);     console.log(st+1, cur-left);     if (st + 1 < cur - left) {      arr.push('...');     }     for (let i = Math.max(cur - left, st + 1); i <= cur - 1; ++i) {      arr.push(i);     }     if (cur != st) {      arr.push(cur);     }     for (let i = cur + 1; i <= cur + right && i <= end - 1 ; ++i) {      arr.push(i);     }     if (cur + right < end - 1) {      arr.push('...');     }     if (end != cur) {      arr.push(end);     }     return arr;    }    }  },  methods: {   clickPage (curIndex) {    if (Number.isInteger(curIndex)) {     this.curPageIndex = curIndex;    }   },   checkNum (curIndex) {    return Number.isInteger(curIndex);   }     },  created () {   this.totalPageCount = getTotalPageCount(this.pageUrl,  this.pageParams, this.pageIsSync,     this.eachPageSiz);  } }</script>

主要是个对于 组件事件的运用,=最常见的click事件,以及class与style的绑定,根据 curPageIndex与this.pageIndex来比较,判断是否拥有这个class,通过computed计算属性,来获得 页码数组 因为会根据当前页 有所变化,created的时候 计算出总页码。
最后一个是 目前生成获取静态数据的js文件.

// let data = {//  avatar: '', 头像//  name: '', 用户名//  context: '', 评论内容// }let dataArr = [];function randomStr (len) { return Math.random().toString(36).substr(len);}function initData () { for (var i = 0; i<45 ; ++i) {  let _avator = "./resources/" + i%7 + ".jpg";  let _name = randomStr(20);  let _context = randomStr(2);  dataArr.push({   avatar: _avator,   name: _name,   context: _context  }); }}if (!dataArr.length) { initData();}export function getCommentData (url = '', params = null, isSync = true, curPageIndex = 1, eachPageSize = 7) { /* ajax */ let st = (curPageIndex - 1) * eachPageSize; let end = st + eachPageSize; return dataArr.slice(st, end);}export function getTotalCommentCount(url = '', params = null, isSync = true) { /* ajax */ return dataArr.length;}export function getTotalPageCount(url = '', params = null, isSync = true, eachPageSize = 7) { /* ajax */ return Math.floor((dataArr.length + eachPageSize -1 ) / eachPageSize);}

就这样了吧。

github地址

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表