<button type="button" class="layui-btn layui-btn-xs btn-edit" data-id=" {
{$value.Id}}">编辑</button>
<!-- 修改分类的弹出层 -->
<script type="text/html" id="dialog-edit">
<form class="layui-form" id="form-edit" lay-filter="form-edit">
<!-- 隐藏域,保存 Id 的值 -->
<input type="hidden" name="Id">
<div class="layui-form-item">
<label class="layui-form-label">分类名称</label>
<div class="layui-input-block">
<input type="text" name="name" required lay-verify="required" placeholder="请输入分类名称" autocomplete="off" class="layui-input">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">分类别名</label>
<div class="layui-input-block">
<input type="text" name="alias" required lay-verify="required" placeholder="请输入分类别名" autocomplete="off" class="layui-input">
</div>
</div>
<div class="layui-form-item">
<div class="layui-input-block">
<button class="layui-btn" lay-submit lay-filter="formDemo">确认修改</button>
</div>
</div>
</form>
</script>
// 通过代理的形式,为 btn-edit 按钮绑定点击事件
var indexEdit = null
$('tbody').on('click', '.btn-edit', function() {
1. 弹出一个修改文章分类信息的层
indexEdit = layer.open({
type: 1,
area: ['500px', '250px'],
title: '修改文章分类',
content: $('#dialog-edit').html()
})
var id = $(this).attr('data-id')
2. 发起请求获取对应分类的数据
$.ajax({
method: 'GET',
url: '/my/article/cates/' + id,
success: function(res) {
form.val('form-edit', res.data)
}
})
})
<button type="button" class="layui-btn layui-btn-xs btn-edit" data-id=" {
{$value.Id}}">编辑</button>
在上方
// 通过代理的形式,为修改分类的表单绑定 submit 事件
$('body').on('submit', '#form-edit', function(e) {
e.preventDefault()
$.ajax({
method: 'POST',
url: '/my/article/updatecate',
data: $(this).serialize(),
success: function(res) {
if (res.status !== 0) {
return layer.msg('更新分类数据失败!')
}
layer.msg('更新分类数据成功!')
layer.close(indexEdit) 这个是 关闭弹窗
initArtCateList()
}
})
})
<button type="button" class="layui-btn layui-btn-danger layui-btn-xs btn- delete" data-id="{
{$value.Id}}">删除</button>
// 通过代理的形式,为删除按钮绑定点击事件
$('tbody').on('click', '.btn-delete', function() {
var id = $(this).attr('data-id')
// 提示用户是否要删除
layer.confirm('确认删除?', {
icon: 3, title: '提示' }, function(index) {
$.ajax({
method: 'GET',
url: '/my/article/deletecate/' + id,
success: function(res) {
if (res.status !== 0) {
return layer.msg('删除分类失败!')
}
layer.msg('删除分类成功!')
layer.close(index)
initArtCateList()
}
})
})
})
// 定义一个查询的参数对象,将来请求数据的时候,
// 需要将请求参数对象提交到服务器
var q = {
pagenum: 1, // 页码值,默认请求第一页的数据
pagesize: 2, // 每页显示几条数据,默认每页显示2条
cate_id: '', // 文章分类的 Id
state: '' // 文章的发布状态
}
initTable()
// 获取文章列表数据的方法
function initTable() {
$.ajax({
method: 'GET',
url: '/my/article/list',
data: q,
success: function(res) {
if (res.status !== 0) {
return layer.msg('获取文章列表失败!')
}
// 使用模板引擎渲染页面的数据
var htmlStr = template('tpl-table', res)
$('tbody').html(htmlStr)
// 调用渲染分页的方法
renderPage(res.total)
}
})
}
<!-- 列表区域 -->
<table class="layui-table">
<colgroup>
<col />
<col width="150" /> 这个代表每个区域的 宽度
<col width="180" />
<col width="150" />
<col width="150" />
</colgroup>
<thead>
<tr>
<th>文章标题</th>
<th>分类</th>
<th>发表时间</th>
<th>状态</th>
<th>操作</th>
</tr>
</thead>
<tbody></tbody> tbody 作为引擎模板使用
</table>
<!-- 列表数据的模板引擎 -->
<script type="text/html" id="tpl-table">
{
{
each data}}
<tr>
<td>{
{
$value.title}}</td>
<td>{
{
$value.cate_name}}</td>
<td>{
{
$value.pub_date|dataFormat}}</td>
<td>{
{
$value.state}}</td>
<td>
<button type="button" class="layui-btn layui-btn-xs">编辑</button>
<button type="button" class="layui-btn layui-btn-danger layui-btn-xs btn-delete" data-id="{
{$value.Id}}">删除</button>
</td>
</tr>
{
{
/each}}
</script>
// 定义美化时间的过滤器
template.defaults.imports.dataFormat = function(date) {
const dt = new Date(date)
var y = dt.getFullYear()
var m = padZero(dt.getMonth() + 1)
var d = padZero(dt.getDate())
var hh = padZero(dt.getHours())
var mm = padZero(dt.getMinutes())
var ss = padZero(dt.getSeconds())
return y + '-' + m + '-' + d + ' ' + hh + ':' + mm + ':' + ss
}
// 定义补零的函数
function padZero(n) {
return n > 9 ? n : '0' + n
}
<td>{
{
$value.pub_date|dataFormat}}</td>
<!-- 筛选区域 -->
<form class="layui-form" id="form-search">
<div class="layui-form-item layui-inline">
<select name="cate_id"></select>
</div>
<div class="layui-form-item layui-inline">
<select name="state">
<option value="">所有状态</option>
<option value="已发布">已发布</option>
<option value="草稿">草稿</option>
</select>
</div>
<div class="layui-form-item layui-inline">
<button class="layui-btn" lay-submit lay-filter="formDemo">筛选</button>
</div>
</form>
initCate()
// 初始化文章分类的方法
function initCate() {
$.ajax({
method: 'GET',
url: '/my/article/cates',
success: function(res) {
if (res.status !== 0) {
return layer.msg('获取分类数据失败!')
}
// 调用模板引擎渲染分类的可选项
var htmlStr = template('tpl-cate', res)
$('[name=cate_id]').html(htmlStr)
// 通过 layui 重新渲染表单区域的UI结构
form.render()
}
})
}
<!-- 分类可选项的模板结构 -->
<script type="text/html" id="tpl-cate">
<option value="">所有分类</option>
{
{
each data}}
<option value="{
{$value.Id}}">{
{
$value.name}}</option>
{
{
/each}}
</script>
// 为筛选表单绑定 submit 事件
$('#form-search').on('submit', function(e) {
e.preventDefault()
// 获取表单中选中项的值
var cate_id = $('[name=cate_id]').val()
var state = $('[name=state]').val()
// 为查询参数对象 q 中对应的属性赋值
q.cate_id = cate_id
q.state = state
// 根据最新的筛选条件,重新渲染表格的数据
initTable()
})
// 定义渲染分页的方法
function renderPage(total) {
// 调用 laypage.render() 方法来渲染分页的结构
laypage.render({
elem: 'pageBox', // 分页容器的 Id
count: total, // 总数据条数
limit: q.pagesize, // 每页显示几条数据
curr: q.pagenum, // 设置默认被选中的分页
layout: ['count', 'limit', 'prev', 'page', 'next', 'skip'],
limits: [2, 3, 5, 10],
// 分页发生切换的时候,触发 jump 回调
// 触发 jump 回调的方式有两种:
// 1. 点击页码的时候,会触发 jump 回调
// 2. 只要调用了 laypage.render() 方法,就会触发 jump 回调
jump: function(obj, first) {
// 可以通过 first 的值,来判断是通过哪种方式,触发的 jump 回调
// 如果 first 的值为 true,证明是方式2触发的
// 否则就是方式1触发的
console.log(first)
console.log(obj.curr)
// 把最新的页码值,赋值到 q 这个查询参数对象中
q.pagenum = obj.curr
// 把最新的条目数,赋值到 q 这个查询参数对象的 pagesize 属性中
q.pagesize = obj.limit
// 根据最新的 q 获取对应的数据列表,并渲染表格
// initTable()
if (!first) {
重要!!!!!!!(不然会死循环)
initTable()
}
}
})
}
<!-- 分页区域 -->
<div id="pageBox"></div>
// 定义渲染分页的方法
function renderPage(total) {
// 调用 laypage.render() 方法来渲染分页的结构
laypage.render({
elem: 'pageBox', // 分页容器的 Id
count: total, // 总数据条数
limit: q.pagesize, // 每页显示几条数据
curr: q.pagenum, // 设置默认被选中的分页
jump: function(obj, first) {
// 可以通过 first 的值,来判断是通过哪种方式,触发的 jump 回调
// 如果 first 的值为 true,证明是方式2触发的
// 否则就是方式1触发的
console.log(first)
console.log(obj.curr)
// 把最新的页码值,赋值到 q 这个查询参数对象中
q.pagenum = obj.curr
// 把最新的条目数,赋值到 q 这个查询参数对象的 pagesize 属性中
q.pagesize = obj.limit
// 根据最新的 q 获取对应的数据列表,并渲染表格
// initTable()
if (!first) {
重要!!!!!!!(不然会死循环)
initTable()
}
}
触发 jump 回调的方式有两种:
jump: function(obj, first) {
// 可以通过 first 的值,来判断是通过哪种方式,触发的 jump 回调
// 如果 first 的值为 true,证明是方式2触发的
// 否则就是方式1触发的
console.log(first)
console.log(obj.curr)
// 把最新的页码值,赋值到 q 这个查询参数对象中
q.pagenum = obj.curr
// 把最新的条目数,赋值到 q 这个查询参数对象的 pagesize 属性中
q.pagesize = obj.limit
// 根据最新的 q 获取对应的数据列表,并渲染表格
// initTable()
if (!first) {
initTable()
}
}
<button type="button" class="layui-btn layui-btn-danger layui-btn-xs btn- delete" data-id="{
{$value.Id}}">删除</button>
// 通过代理的形式,为删除按钮绑定点击事件处理函数
$('tbody').on('click', '.btn-delete', function() {
// 获取删除按钮的个数
var len = $('.btn-delete').length
console.log(len)
// 获取到文章的 id
var id = $(this).attr('data-id')
// 询问用户是否要删除数据
layer.confirm('确认删除?', {
icon: 3, title: '提示' }, function(index) {
$.ajax({
method: 'GET',
url: '/my/article/delete/' + id,
success: function(res) {
if (res.status !== 0) {
return layer.msg('删除文章失败!')
}
layer.msg('删除文章成功!') 小bug!!!!!!!!!!!
// 当数据删除完成后,需要判断当前这一页中,是否还有剩余的数据
// 如果没有剩余的数据了,则让页码值 -1 之后,
// 再重新调用 initTable 方法
// 4
if (len === 1) {
// 如果 len 的值等于1,证明删除完毕之后,页面上就没有任何数据了
// 页码值最小必须是 1
q.pagenum = q.pagenum === 1 ? 1 : q.pagenum - 1
}
initTable()
}
})
layer.close(index)
})
})
<!-- 分类的模板结构 -->
<script type="text/html" id="tpl-cate">
<option value="">请选择文章类别</option>
{
{
each data}}
<option value="{
{$value.Id}}">{
{
$value.name}}</option>
{
{
/each}}
</script>
var layer = layui.layer
var form = layui.form
initCate()
// 初始化富文本编辑器
initEditor()
// 定义加载文章分类的方法
function initCate() {
$.ajax({
method: 'GET',
url: '/my/article/cates',
success: function(res) {
if (res.status !== 0) {
return layer.msg('初始化文章分类失败!')
}
// 调用模板引擎,渲染分类的下拉菜单
var htmlStr = template('tpl-cate', res)
$('[name=cate_id]').html(htmlStr)
// 一定要记得调用 form.render() 方法
form.render()
}
})
}
添加如下的 layui
表单行:
<div class="layui-form-item">
<label class="layui-form-label">文章内容label>
<div class="layui-input-block" style="height: 400px;">
<textarea name="content">textarea>
div>
div>
导入富文本必须的 script
脚本:
<script src="/assets/lib/tinymce/tinymce.min.js">script>
<script src="/assets/lib/tinymce/tinymce_setup.js">script>
调用 initEditor()
方法,初始化富文本编辑器:
// 初始化富文本编辑器
initEditor()
1. 图片封面裁剪的实现步骤
添加如下的 layui
表单行:
在 中导入
cropper.css
样式表:
<link rel="stylesheet" href="/assets/lib/cropper/cropper.css" />
在 的结束标签之前,按顺序导入如下的 js 脚本:
<script src="/assets/lib/jquery.js">script>
<script src="/assets/lib/cropper/Cropper.js">script>
<script src="/assets/lib/cropper/jquery-cropper.js">script>
在表单中,添加如下的表单行结构:
<div class="layui-form-item">
<label class="layui-form-label">文章封面label>
<div class="layui-input-block cover-box">
<div class="cover-left">
<img id="image" src="/assets/images/sample2.jpg" alt="" />
div>
<div class="cover-right">
<div class="img-preview">div>
<button type="button" class="layui-btn layui-btn-danger">选择封面button>
div>
div>
div>
美化的样式:
/* 封面容器的样式 */
.cover-box {
display: flex;
}
/* 左侧裁剪区域的样式 */
.cover-left {
width: 400px;
height: 280px;
overflow: hidden;
margin-right: 20px;
}
/* 右侧盒子的样式 */
.cover-right {
display: flex;
flex-direction: column;
align-items: center;
}
/* 预览区域的样式 */
.img-preview {
width: 200px;
height: 140px;
background-color: #ccc;
margin-bottom: 20px;
overflow: hidden;
}
实现基本裁剪效果:
// 1. 初始化图片裁剪器
var $image = $('#image')
// 2. 裁剪选项
var options = {
aspectRatio: 400 / 280,
preview: '.img-preview'
}
// 3. 初始化裁剪区域
$image.cropper(options)
2.更换裁剪的图片
拿到用户选择的文件
var file = e.target.files[0]
根据选择的文件,创建一个对应的 URL 地址:
var newImgURL = URL.createObjectURL(file)
先销毁
旧的裁剪区域,再重新设置图片路径
,之后再创建新的裁剪区域
:
$image
.cropper('destroy') // 销毁旧的裁剪区域
.attr('src', newImgURL) // 重新设置图片路径
.cropper(options) // 重新初始化裁剪区域
3. 将裁剪后的图片,输出为文件
$image
.cropper('getCroppedCanvas', {
// 创建一个 Canvas 画布
width: 400,
height: 280
})
.toBlob(function(blob) {
// 将 Canvas 画布上的内容,转化为文件对象
// 得到文件对象后,进行后续的操作
})
<!-- 右侧预览区域和选择封面区域 -->
<div class="cover-right">
<!-- 预览的区域 -->
<div class="img-preview"></div>
<!-- 选择封面按钮 -->
<button type="button" class="layui-btn layui-btn-danger" id="btnChooseImage">选择封面</button>
<!-- 隐藏的文件选择框 -->
<input type="file" id="coverFile" style="display: none;" accept="image/png,image/jpeg,image/gif" />
</div>
// 为选择封面的按钮,绑定点击事件处理函数
$('#btnChooseImage').on('click', function() {
$('#coverFile').click()
})
// 监听 coverFile 的 change 事件,获取用户选择的文件列表
$('#coverFile').on('change', function(e) {
// 获取到文件的列表数组
var files = e.target.files
// 判断用户是否选择了文件
if (files.length === 0) {
return
}
// 根据文件,创建对应的 URL 地址
var newImgURL = URL.createObjectURL(files[0])
// 为裁剪区域重新设置图片
$image
.cropper('destroy') // 销毁旧的裁剪区域
.attr('src', newImgURL) // 重新设置图片路径
.cropper(options) // 重新初始化裁剪区域
})
var art_state = '已发布'
// 定义文章的发布状态
var art_state = '已发布'
// 为存为草稿按钮,绑定点击事件处理函数
$('#btnSave2').on('click', function() {
art_state = '草稿'
})
<form class="layui-form" id="form-pub"></form> 1
// 为表单绑定 submit 提交事件
$('#form-pub').on('submit', function(e) {
// 1. 阻止表单的默认提交行为
e.preventDefault()
// 2. 基于 form 表单,快速创建一个 FormData 对象
var fd = new FormData($(this)[0])
// 3. 将文章的发布状态,存到 fd 中
fd.append('state', art_state)
// 4. 将封面裁剪过后的图片,输出为一个文件对象
$image
.cropper('getCroppedCanvas', {
// 创建一个 Canvas 画布
width: 400,
height: 280
})
.toBlob(function(blob) {
// 将 Canvas 画布上的内容,转化为文件对象
// 得到文件对象后,进行后续的操作
// 5. 将文件对象,存储到 fd 中
fd.append('cover_img', blob)
// 6. 发起 ajax 数据请求
publishArticle(fd)
})
})
// 定义一个发布文章的方法
function publishArticle(fd) {
$.ajax({
method: 'POST',
url: '/my/article/add',
data: fd,
// 注意:如果向服务器提交的是 FormData 格式的数据,
// 必须添加以下两个配置项
contentType: false,
processData: false,
success: function(res) {
if (res.status !== 0) {
return layer.msg('发布文章失败!')
}
layer.msg('发布文章成功!')
// 发布文章成功后,跳转到文章列表页面
location.href = '/article/art_list.html'
}
})
}