Vue大事件练习(保姆级教程)11完结篇

144 阅读1分钟

1.11-分页功能

1656528231596.png

  • 1.布局

    • size-change : 监听page-size变化
    • current-change : 监听页码变化
  • <!-- 分页区域 -->
    <el-pagination
      @size-change="handleSizeChange"
      @current-change="handleCurrentChange"
      :current-page="q.pagenum"
      :page-sizes="[2, 3, 5, 10]"
      :page-size="q.pagesize"
      layout="total, sizes, prev, pager, next, jumper"
      :total="total"
    >
    </el-pagination>
    
  • 2.功能实现

  • //5.pageSize 发生了变化
    handleSizeChange(newSize) {
      // 为 pagesize 赋值
      this.q.pagesize = newSize;
      // 默认展示第一页数据
      this.pagenum = 1;
      // 重新发起请求
      this.initArtList();
    },
    //6.页码变化
    handleCurrentChange(newPage) {
      // 为页码值赋值
      this.q.pagenum = newPage;
      // 重新发起请求
      this.initArtList();
    }
    

1.12-筛选与重置功能

  • 1.渲染分类
  • <el-form-item label="文章分类">
          <el-select
            v-model="q.cate_id"
            placeholder="请选择分类"
            size="small"
          >
            <!-- 循环渲染可选项 -->
            <el-option
              v-for="item in cateList"
              :key="item.id"
              :label="item.cate_name"
              :value="item.id"
            >
            </el-option>
          </el-select>
      </el-form-item>
    
  • 2.给筛选按钮和重置按钮绑定事件
  • <el-button type="primary" size="small" @click="initArtList">筛选</el-button>
    <el-button type="info" size="small" @click="resetList">重置</el-button>
    
  • 3.筛选功能不用单独写代码,直接重新加载列表即可(因为我们的请求参数已经与筛选框双向绑定了)
  • 4.重置功能需要重新将请求参数恢复默认值
  • //7.重置筛选
    resetList() {
      // 1. 重置查询参数对象
      this.q = {
        pagenum: 1,
        pagesize: 2,
        cate_id: "",
        state: ""
      };
      // 2. 重新发起请求
      this.initArtList();
    }
    

1.13-文章详情

1656530081098.png

  • 1.给文章标题绑定一个点击事件

    • 需要使用作用域插槽获取当前文章id(用于事件传参)

      • @click="showDetail(row.id)"
<!-- 文章表格区域 -->
  <el-table :data="artList" style="width: 100%;" border stripe>
    <!-- (1)文章标题 -->
    <el-table-column label="文章标题">
      <template v-slot="{ row }">
        <el-link type="primary" @click="showDetail(row.id)">{{
          row.title
        }}</el-link>
      </template>
    </el-table-column>
    <!-- (2)文章分类 -->
    <el-table-column label="分类" prop="cate_name"></el-table-column>
    <!-- (3)发表时间 -->
    <el-table-column label="发表时间">
      <template v-slot="{ row }">
        {{ formatTime(row.pub_date) }}
      </template>
    </el-table-column>
    <!-- (4)文章状态 -->
    <el-table-column label="状态" prop="state"></el-table-column>
    <!-- (5)操作按钮 -->
    <el-table-column label="操作"></el-table-column>
  </el-table>
  • 2.添加data绑定数据
data() {
  return {
    // 控制文章详情对话框的显示与隐藏
    detailVisible: false,
    // 文章的详情信息对象
    artDetail: {}
  }
}
  • 3.添加展示文章详情的dialog弹框
<!-- 文章详情对话框 -->
<el-dialog title="文章预览" :visible.sync="detailVisible" width="80%">
  <h1 class="title">{{ artDetail.title }}</h1>
  <div class="info">
    <span>作者:{{ artDetail.nickname || artDetail.username }}</span>
    <span>发布时间:{{ formatTime(artDetail.pub_date) }}</span>
    <span>所属分类:{{ artDetail.cate_name }}</span>
    <span>状态:{{ artDetail.state }}</span>
  </div>
  <!-- 分割线 -->
  <el-divider></el-divider>
  <img
    :src="'http://big-event-vue-api-t.itheima.net' + artDetail.cover_img"
    alt=""
  />
  <div v-html="artDetail.content"></div>
</el-dialog>
  • 3.请求文章详情数据
  • //8.文章详情
    async showDetail(id) {
      // 请求详情数据
      const { data: res } = await this.$axios.get("/my/article/info", {
        params: { id }
      })
      if( res.code === 0 ){
        //绑定data中数据
        this.artDetail = res.data
        //显示对话框
         this.detailVisible = true
      }
    }
    

1.14-删除文章

1656530566164.png

  • 1.给el-table添加作用域插槽
  • <!-- (5)操作按钮 -->
    <el-table-column label="操作">
      <template v-slot="{ row }">
        <el-button type="danger" size="mini" @click="doDelete(row.id)">删除</el-button>
      </template>
    </el-table-column>
    
  • 2.完成删除功能
  • //9.删除文章
    async doDelete(id) {
      // 1. 询问用户是否要删除
      const confirm = await this.$confirm(
        "此操作将永久删除该文件, 是否继续?",
        "提示",
        {
          confirmButtonText: "确定",
          cancelButtonText: "取消",
          type: "warning"
        }
      ).catch(err => err);
      // 2. 点击确认
      if (confirm) {
        const { data: res } = await this.$axios.delete("/my/article/info", {
          params: { id }
        })
        //提示
        this.$message.success(res.message)
        //刷新列表数据
        this.initArtList()
      }
    }
    

1.15-完整代码

<template>
  <div>
    <el-card class="box-card">
      <div slot="header" class="clearfix">
        <span>文章列表</span>
      </div>
      <!-- 搜索区域 -->
      <div class="search-box">
        <el-form :inline="true">
          <el-form-item label="文章分类">
            <el-select
              v-model="q.cate_id"
              placeholder="请选择分类"
              size="small"
            >
              <!-- 循环渲染可选项 -->
              <el-option
                v-for="item in cateList"
                :key="item.id"
                :label="item.cate_name"
                :value="item.id"
              >
              </el-option>
            </el-select>
          </el-form-item>
          <el-form-item label="发布状态" style="margin-left: 15px;">
            <el-select placeholder="请选择状态" size="small" value="">
              <el-option label="已发布" value="已发布"></el-option>
              <el-option label="草稿" value="草稿"></el-option>
            </el-select>
          </el-form-item>
          <el-form-item>
            <el-button type="primary" size="small" @click="initArtList"
              >筛选</el-button
            >
            <el-button type="info" size="small">重置</el-button>
          </el-form-item>
        </el-form>
        <!-- 发表文章的按钮 -->
        <el-button
          type="primary"
          size="small"
          class="btn-pub"
          @click="dialogVisible = true"
          >发表文章</el-button
        >
      </div>
      <!-- 文章表格区域 -->
      <el-table :data="artList" style="width: 100%;" border stripe>
        <!-- (1)文章标题 -->
        <el-table-column label="文章标题">
          <template v-slot="{ row }">
            <el-link type="primary" @click="showDetail(row.id)">{{
              row.title
            }}</el-link>
          </template>
        </el-table-column>
        <!-- (2)文章分类 -->
        <el-table-column label="分类" prop="cate_name"></el-table-column>
        <!-- (3)发表时间 -->
        <el-table-column label="发表时间">
          <template v-slot="{ row }">
            {{ formatTime(row.pub_date) }}
          </template>
        </el-table-column>
        <!-- (4)文章状态 -->
        <el-table-column label="状态" prop="state"></el-table-column>
        <!-- (5)操作按钮 -->
        <el-table-column label="操作">
          <template v-slot="{ row }">
            <el-button type="danger" size="mini" @click="doDelete(row.id)"
              >删除</el-button
            >
          </template>
        </el-table-column>
      </el-table>
      <!-- 分页区域 -->
​
      <el-pagination
        @size-change="handleSizeChange"
        @current-change="handleCurrentChange"
        :current-page="q.pagenum"
        :page-sizes="[2, 3, 5, 10]"
        :page-size="q.pagesize"
        layout="total, sizes, prev, pager, next, jumper"
        :total="total"
      >
      </el-pagination>
    </el-card>
    <!-- 发表文章对话框 -->
    <el-dialog title="发表文章" :visible.sync="dialogVisible" fullscreen>
      <!-- 对话框内容 -->
      <el-form label-width="100px">
        <!-- 1.文章标题 -->
        <el-form-item label="文章标题">
          <el-input placeholder="请输入标题" v-model="pubForm.title"></el-input>
        </el-form-item>
        <!-- 2.文章分类 -->
        <el-form-item label="文章分类">
          <el-select
            placeholder="请选择分类"
            style="width: 100%;"
            v-model="pubForm.cate_id"
          >
            <el-option
              v-for="item in cateList"
              :key="item.id"
              :label="item.cate_name"
              :value="item.id"
            ></el-option>
          </el-select>
        </el-form-item>
        <!-- 3.文章内容 -->
        <el-form-item label="文章内容">
          <quill-editor v-model="pubForm.content"></quill-editor>
        </el-form-item>
        <!-- 4.文章封面 -->
        <el-form-item label="文章封面">
          <!-- 用来显示封面的图片 -->
          <img
            src="@/assets/images/cover.jpg"
            alt=""
            class="cover-img"
            ref="imgRef"
          />
          <br />
          <!-- 文件选择框,默认被隐藏 -->
          <input
            type="file"
            style="display: none;"
            accept="image/*"
            ref="iptFile"
            @change="doChange"
          />
          <!-- 选择封面的按钮 -->
          <el-button type="text" @click="$refs.iptFile.click()"
            >+ 选择封面</el-button
          >
        </el-form-item>
        <!-- 5.底部按钮 -->
        <el-form-item>
          <el-button type="primary" @click="pubArticle('已发布')"
            >发布</el-button
          >
          <el-button type="info" @click="pubArticle('草稿')"
            >存为草稿</el-button
          >
        </el-form-item>
      </el-form>
    </el-dialog>
    <!-- 文章详情对话框 -->
    <el-dialog title="文章预览" :visible.sync="detailVisible" width="80%">
      <h1 class="title">{{ artDetail.title }}</h1>
      <div class="info">
        <span>作者:{{ artDetail.nickname || artDetail.username }}</span>
        <span>发布时间:{{ formatTime(artDetail.pub_date) }}</span>
        <span>所属分类:{{ artDetail.cate_name }}</span>
        <span>状态:{{ artDetail.state }}</span>
      </div>
      <!-- 分割线 -->
      <el-divider></el-divider>
      <img
        :src="'http://big-event-vue-api-t.itheima.net' + artDetail.cover_img"
        alt=""
      />
      <div v-html="artDetail.content"></div>
    </el-dialog>
  </div>
</template><script>
// 导入富文本编辑器
import VueQuillEditor from "vue-quill-editor";
// 导入富文本编辑器的样式
import "quill/dist/quill.core.css";
import "quill/dist/quill.snow.css";
import "quill/dist/quill.bubble.css";
//导入vue(在main.js中这行可以省略。 )
import Vue from "vue";
// 全局注册富文本编辑器
Vue.use(VueQuillEditor);
//导入moment.js
import moment from "moment";
​
export default {
  name: "ArtList",
  //1.数据
  data() {
    return {
      //1.1 控制对话框显示/隐藏
      dialogVisible: false,
      //1.2 文章分类
      cateList: [],
      //1.3 文章的列表数据
      artList: [],
      //1.4 总数据条数
      total: 0,
      //1.5 查询参数对象
      q: {
        // 页码值
        pagenum: 1,
        pagesize: 2,
        cate_id: "",
        state: ""
      },
      //1.6 发布文章数据
      pubForm: {
        // 文章的标题
        title: "",
        // 所属的分类 Id
        cate_id: "",
        // 文章的内容
        content: "",
        // 选中的封面的文件(null 表示没有选中任何封面文件)
        cover_img: null,
        // 文章的发布状态,可选值只有(草稿、已发布)
        state: ""
      },
      //1.7 控制文章详情对话框的显示与隐藏
      detailVisible: false,
      //1.8 文章的详情信息对象
      artDetail: {}
    };
  },
  methods: {
    //1.图片预览
    doChange(e) {
      // 获取到用户选择的封面
      const file = e.target.files[0];
      if (file) {
        // 绑定到data中
        this.pubForm.cover_img = file;
        //生成url
        const url = URL.createObjectURL(file);
        //显示到img
        this.$refs.imgRef.setAttribute("src", url);
      }
    },
    //2.发表文章
    async pubArticle(state) {
      //(1)把文章的发表状态保存到 data 中
      this.pubForm.state = state;
      //(2)使用formdata发送文件数据
      const fd = new FormData();
      //将data对象参数添加到fd中(因为fd会自动我们设置请求头+处理文件二进制)
      for (const k in this.pubForm) {
        fd.append(k, this.pubForm[k]);
      }
      //(3)调用接口
      const { data: res } = await this.$axios.post("/my/article/add", fd);
      //提示消息
      this.$message.success(res.message);
      //成功
      if (res.code === 0) {
        // 关闭对话框
        this.dialogVisible = false;
        // 刷新文章的列表数据
        this.initArtList();
      }
    },
    //3.获取文章列表
    async initArtList() {
      const { data: res } = await this.$axios.get("/my/article/list", {
        params: this.q
      });
      if (res.code === 0) {
        this.artList = res.data;
        this.total = res.total;
      }
    },
    //4.时间格式化
    formatTime(time) {
      return moment(time).format("YYYY-MM-DD HH:mm:ss");
    },
    //5.pageSize 发生了变化
    handleSizeChange(newSize) {
      // 为 pagesize 赋值
      this.q.pagesize = newSize;
      // 默认展示第一页数据
      this.pagenum = 1;
      // 重新发起请求
      this.initArtList();
    },
    //6.页码变化
    handleCurrentChange(newPage) {
      // 为页码值赋值
      this.q.pagenum = newPage;
      // 重新发起请求
      this.initArtList();
    },
    //7.重置筛选
    resetList() {
      // 1. 重置查询参数对象
      this.q = {
        pagenum: 1,
        pagesize: 2,
        cate_id: "",
        state: ""
      };
      // 2. 重新发起请求
      this.initArtList();
    },
    //8.文章详情
    async showDetail(id) {
      // 请求详情数据
      const { data: res } = await this.$axios.get("/my/article/info", {
        params: { id }
      });
      if (res.code === 0) {
        //绑定data中数据
        this.artDetail = res.data;
        //显示对话框
        this.detailVisible = true;
      }
    },
    //9.删除文章
    async doDelete(id) {
      // 1. 询问用户是否要删除
      const confirm = await this.$confirm(
        "此操作将永久删除该文件, 是否继续?",
        "提示",
        {
          confirmButtonText: "确定",
          cancelButtonText: "取消",
          type: "warning"
        }
      ).catch(err => err);
      // 2. 点击确认
      if (confirm) {
        const { data: res } = await this.$axios.delete("/my/article/info", {
          params: { id }
        })
        //提示
        this.$message.success(res.message)
        //刷新列表数据
        this.initArtList()
      }
    }
  },
  async created() {
    //1.获取文章分类(不需要反复调用这个接口,可以直接在created钩子里面发送ajax)
    const { data: res } = await this.$axios.get("/my/cate/list");
    if (res.code === 0) {
      // 把数据转存到 data 中
      this.cateList = res.data;
    }
    //2.获取文章列表(需要多次调用,封装成函数)
    this.initArtList();
  }
};
</script><style lang="less" scoped>
.search-box {
  display: flex;
  justify-content: space-between;
  align-items: flex-start;
  .btn-pub {
    margin-top: 5px;
  }
}
​
/deep/ .ql-editor {
  min-height: 300px;
}
​
// 设置图片封面的宽高
.cover-img {
  width: 400px;
  height: 280px;
  object-fit: cover;
}
​
.el-pagination {
  margin-top: 15px;
}
​
.title {
  font-size: 24px;
  text-align: center;
  font-weight: normal;
  color: #000;
  margin: 0 0 10px 0;
}
​
.info {
  font-size: 12px;
  span {
    margin-right: 20px;
  }
}
​
// 修改 dialog 内部元素的样式,需要添加样式穿透
/deep/ .detail-box {
  img {
    width: 500px;
  }
}
</style>