Vue + Spring Boot 项目实战(七)笔记

Vue + Spring Boot 项目实战(七)笔记

前言

这一篇主要是把图书界面给完善一下,加上搜索框以及编辑、添加页面,在原教程基础上,改动了一些地方:

  1. 将添加图书的图标和EditBook组件分离开来,并且进行一个是否是尾页的判断,控制其是否显示。
  2. 将添加功能和编辑功能分开(因为用的不是JPA,没有更新和添加共有的方法)
  3. 解决后端id字段改为int后,select组件显示问题。
  4. 简化一些代码...

Books.vue

<template>
  <div>
    <el-row style="height: 100%;">
      <SearchBar @onSearch="searchResult" ref="searchBar"></SearchBar>
      <el-tooltip effect="dark" placement="right"
                  v-for="item in books.slice((currentPage-1)*pageSize,currentPage*pageSize)"
                  :key="item.id">
        <p slot="content" style="font-size: 14px;margin-bottom: 6px;">{{ item.title }}</p>
        <p slot="content" style="font-size: 13px;margin-bottom: 6px">
          <span>{{ item.author }}</span> /
          <span>{{ item.date }}</span> /
          <span>{{ item.press }}</span>
        </p>
        <p slot="content" style="width: 300px" class="abstract">{{ item.abs }}</p>
        <el-card style="width: 135px;margin-bottom: 20px;float: left;margin-right: 15px" class="book"
                 bodyStyle="padding:10px" shadow="hover">
          <div class="cover" @click="editBook(item)">
            <img :src="item.cover" alt="封面">
          </div>
          <div class="info">
            <div class="title">
              <a href="">{{ item.title }}</a>
            </div>
          </div>
          <div class="author">{{ item.author }}</div>
          <i class="el-icon-delete" @click="deleteBook(item.id)"></i>
        </el-card>
      </el-tooltip>
      <i v-show="currentPage==books.length%2+1" class="el-icon-circle-plus-outline"
         @click="addBook"></i>
      <EditForm @onSubmit="loadBooks()" ref="edit"></EditForm>
    </el-row>
    <el-row>
      <el-pagination
          @current-change="handleCurrentChange"
          :current-page="currentPage"
          :page-size="pageSize"
          :total="books.length">
      </el-pagination>
    </el-row>
  </div>
</template>

<script>
import EditForm from "./EditForm";
import SearchBar from "./SearchBar";

export default {
  name: "Books",
  components: {
    EditForm,
    SearchBar
  },
  data() {
    return {
      books: [],
      currentPage: 1,
      pageSize: 17
    }
  },
  mounted: function () {
    this.loadBooks()
  },
  methods: {
    loadBooks() {
      this.$axios.get('/books').then(resp => {
        if (resp && resp.status === 200) {
          this.books = resp.data
        }
      })
    },
    handleCurrentChange(currentPage) {
      this.currentPage = currentPage
    },
    searchResult() {
      this.$axios.get('/search?keywords=' + this.$refs.searchBar.keywords, {}).then(resp => {
        if (resp && resp.status === 200) {
          this.books = resp.data
        }
      })
    },
    deleteBook(id) {
      this.$confirm('此操作将永久删除该书籍, 是否继续?', '提示', {
        confirmButtonText: '确定',
        cancelButtonText: '取消',
        type: 'warning'
      }).then(() => {
            this.$axios.post('/delete', {id: id}).then(resp => {
              if (resp && resp.status === 200) {
                this.loadBooks()
              }
            })
          }
      ).catch(() => {
        this.$message({
          type: 'info',
          message: '已取消删除'
        })
      })
    },
    editBook(item) {
      this.$refs.edit.dialogFormVisible = true
      this.$refs.edit.form = item
    },
    addBook(){
      this.$refs.edit.dialogFormVisible = true
    }
  }
}
</script>

<style scoped>
.cover {
  width: 115px;
  height: 172px;
  margin-bottom: 7px;
  overflow: hidden;
  cursor: pointer;
}

img {
  width: 115px;
  height: 172px;
}

.title {
  font-size: 14px;
  text-align: left;
}

.author {
  color: #333;
  width: 102px;
  font-size: 13px;
  margin-bottom: 6px;
  text-align: left;
}

.abstract {
  display: block;
  line-height: 17px;
}

a {
  text-decoration: none;
}

a:link, a:visited, a:focus {
  color: #3377aa;
}

.el-icon-circle-plus-outline {
  margin: 50px 0 0 20px;
  font-size: 100px;
  float: left;
  cursor: pointer;
}
</style>

主要改动在将添加的部分从EditForm抽取出来,并且使用v-show控制是否显示。

<i v-show="currentPage==books.length%2+1" class="el-icon-circle-plus-outline"
         @click="addBook"></i>

接下来就是一些数据传递的优化:

editBook(item) {
      this.$refs.edit.dialogFormVisible = true
      this.$refs.edit.form = item
      console.log(item)
    },
    addBook(){
      this.$refs.edit.dialogFormVisible = true
    }

反正也是从item里取数据,直接传个item完事。

LibraryIndex.vue

这个部分没什么好改的,有个地方就是var _this = this可以去掉,我测试之后发现axios的箭头函数中this指向的还是VueComponent。

EditForm.vue

这个组件唯一要改动的地方就是select组件部分:

<el-select v-model="form.category.id" placeholder="请选择分类">
            <el-option label="文学" value="1"></el-option>
            <el-option label="流行" value="2"></el-option>
            <el-option label="文化" value="3"></el-option>
            <el-option label="生活" value="4"></el-option>
            <el-option label="经管" value="5"></el-option>
            <el-option label="科技" value="6"></el-option>
          </el-select>

需要在value前面加个:,但是这个也是看情况:

  1. 数据库数据类型为varchar,pojo中类型是String,那么就不需要加
  2. 数据库类型为int,pojo中是int,这种需要加上冒号,也就是v-bind:value的缩写。

还有修复一个bug,methods中的clear方法需要修改category的类型:

clear() {
      this.form = {
        id: '',
        title: '',
        author: '',
        date: '',
        press: '',
        cover: '',
        abs: '',
        category: {
          id: '',
          name: ''
        }
      }

原代码会造成TypeError: Cannot use 'in' operator to search for 'id' in

上一篇:StringRedisTemplate操作redis数据


下一篇:DevExpress WinForms帮助文档:表单控件 - 动态更新添加到初始表单的自定义控件