1.11-分页功能
-
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-文章详情
-
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-删除文章
- 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>