Vue内容管理系统的搭建(三)

一定要相信越努力越幸运

文/ziven先生
标题图片/来自 IM free网站

很抱歉好久没有把系列文章及时的更下去,隔了好久才更第三篇文章,这中间拖拖拉拉发生了一些事,没有时间把要做的做下去,说来也挺沮丧的,简单的事情都没有坚持。还是那句老话,没有坚持哪有梦想,又不是富二代,官二代,你说是吧?

关于该系列文章,将按照从简到难地记录开发过程与遇到的问题,在过程中按照一个个小模块进行细分,其中如果某个模块遇到一个知识点,将在系列文章外的文章记录博主关于该知识点的学习与想法,并且在系列文章内附上链接。

前两篇做到前端登入页面和后端express的搭建。登录之后就进入主界面了。在开发主界面前,必须要知道主要界面的需求是啥,有了需求后,怎么布局主界面。当然也可以先想好主界面怎么布局,再统计下需求是什么。

关于视觉设计方面,一来这是个管理系统,视觉上的体验要求其实并不是那么高,二来由于时间关系,将在系列文章后面记录该内容,当然正规上线的项目是需要把设计做好才进行开发的。

按照博主的需求,要管理系统管理的网站主要有两个模块需要进行管理,一个是反馈页面的反馈信息进行管理,一个是全景图页面的管理,所谓管理就是要增删改查,图片要可以上传等等。
除此之外管理系统,要有使用该管理系统的用户管理,所以,管理系统还需要用户管理模块,分为权限管理,用户查询,个性设置等等

总结起来就是反馈管理,全景图管理,和用户管理三个模块。从反馈管理最简单开始,它主要涉及到table的查询与增删改查,反馈信息表,主要有反馈的时间,反馈人的姓名,反馈的信息等,管理主要是删除管理信息,时间和关键字来查询反馈信息。这样思路就有了。

主页面布局
element-ui 导航栏的小问题

在开发过程中遇到一个问题,因为之前开发经验里,element ui 导航栏通过index 来进行路由跳转的,不需要另外进行手动编写代码来启动这个跳转,然而这次开发过程,设置好index后,发现跳转无效,路由没有发生变化,前前后后黏贴官方代码也同样如此,于是根据官方文档,我在el-ment 标签添加router属性并且设置为false(虽然文档说默认是false),奇迹出现了,路由实现了跳转,不过浏览器出现如下的警告性错误,字面意思就是router这个属性获取的不是boolean类型,这个问题我现在也不知道如何修正,如果有谁知道可以在评论里告诉博主。


el-ment 标签添加router='false'或者router=false的警告性错误

我暂时把它当做一个现象吧,当然可能时间版本更换有所不同,elementUI处理机制不同,而我不知道。解决上述不跳转的现象,需要另辟方法,那就是通过绑定事件,手动写方法,获取index 来跳转。

使用flex布局

之前说了,侧边栏和显示栏是两栏并列显示,可以使用el-row ,el-col来布局,然而当浏览器宽度缩小时,因为是按百分比布局,侧边栏和显示栏都会随之压缩,会导致导致布局混乱,比如显示栏换行显示,甚至是ui的样式也发生重叠的现象,后来发现element-ui在宽度不满足的情况下,都有类似情况,所以我们必须设置好min-width,侧边栏和显示栏都要设置,只设置一方,无法解决问题。
如果觉得用element的布局标签觉得很麻烦,或者在灵活度上不太舒服,那就使用flex布局,要实现同样效果可以轻松多了。

element-ui导航栏,刷新页面后,导航栏对应的导航文字高亮消失了

我不知道这是bug,还是开发团队的设计本来就是这样做,没有找到确切的api说明,但是不管怎么样,还是可以解决,解决方法是,写个方法,判断路由路径,把路径设置到default-active属性里,这样刷新页面,对应导航的文字高亮就不消失了。


反馈页面初略显示

好了,前端视图层的效果和编码差不多了(这里代码就不一一贴出来,因为有点多,也不是重点,所以有兴趣了解代码是什么样的,去github吧),可是数据是死的,接下来,我们就必须把前端逻辑层和后台用express获得数据,把这些数据查出来,并且能通过关键字和时间区间来进行查询。

前端要做的事情就是逻辑处理,路由处理等。逻辑处理有反馈页面的删除记录操作、条件查询、分页查询。路由处理就是,怎么处理路由跳转,怎么设置路由路径。博主采用的是前后端分离的概念,所以在前端发送到后台请求时,由于前后端端口不同,有本地跨域的问题,这时候需要用到vue-cli的跨域设置,下面将会说到。

直接贴代码,反馈页面的逻辑方法实现:

<pre> import { queryFeedbacks, deleteFeedback } from 'api/api';
export default {
data() {
return {
formInline: {
name: '',
keywords: '',
daterange: ''
},
pickerOptions2: {
shortcuts: [{
text: '最近一周',
onClick(picker) {
const end = new Date();
const start = new Date();
start.setTime(start.getTime() - 3600 * 1000 * 24 * 7);
picker.$emit('pick', [start, end]);
}
}, {
text: '最近一个月',
onClick(picker) {
const end = new Date();
const start = new Date();
start.setTime(start.getTime() - 3600 * 1000 * 24 * 30);
picker.$emit('pick', [start, end]);
}
}, {
text: '最近三个月',
onClick(picker) {
const end = new Date();
const start = new Date();
start.setTime(start.getTime() - 3600 * 1000 * 24 * 90);
picker.$emit('pick', [start, end]);
}
}]
},
feedbacks: [],
filtr: {
name: '',
kyw: '',
bdate: '',
edate: '',
page: 1,
pageSize: 10
},
page: {
total: 0,
sizes: [10, 20, 30],
}
}
},
mounted() {
this.getFeedbacks();
},
methods: {
onSubmit() {
this.filtr.name = this.formInline.name;
this.filtr.kyw = this.formInline.keywords;
this.filtr.bdate = this.formInline.daterange[0] ? new Date(this.formInline.daterange[0]).Format("yyyy-MM-dd") : '';
this.filtr.edate = this.formInline.daterange[1] ? new Date(this.formInline.daterange[1]).Format("yyyy-MM-dd") : '';
console.log(this.filtr)
this.getFeedbacks();
},
getFeedbacks() {
let that = this;
let param = this.filtr;
queryFeedbacks(param).then(response => {
let res = response.data;
console.log(res)
if (res.code) {
that.page.total = res.data.total;
that.feedbacks = res.data.feedbacks;
} else {
this.$message({
type: 'error',
message: res.message
})
}
})
},
handleDelete(index, row) {
this.$confirm('此操作将永久删除该文件, 是否继续?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
removeFeedback(row.id).then(res => {
this.$message({
type: 'success',
message: '删除成功!'
});
}).catch(() => {
this.$message({
type: 'alert',
message: '删除失败'
});
})
});
},
handleSizeChange(val) {
this.filtr.pageSize = val;
this.getFeedbacks();
},
handleCurrentChange(val) {
this.filtr.page = val;
this.getFeedbacks();
}
}
}
Date.prototype.Format = function (fmt) {
var o = {
"M+": this.getMonth() + 1, //月份
"d+": this.getDate(), //日
"h+": this.getHours(), //小时
"m+": this.getMinutes(), //分
"s+": this.getSeconds(), //秒
"q+": Math.floor((this.getMonth() + 3) / 3), //季度
"S": this.getMilliseconds() //毫秒
};
if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length));
for (var k in o)
if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length)));
return fmt;
}
</pre>

发送请求的插件使用axios ,vue-source已不再被维护

与vue-source相比,axios可以说要简洁干净一些,后来,发现axios确实不错,vue官方就开始推荐使用它,vue2以后,vue-source已经停止了维护。axios对http请求的封装使得get也可以写得看上去和post请求一样,当然了,也可以把参数直接附在url后面来发送get请求。详细看官网,比较简单,不再赘述。
传送门:https://github.com/mzabriskie/axios
系统用到axios:
<pre>
import axios from 'axios'; let apiUrl='/api'; export const removeFeedback=params=>{ return axios.get(\${apiUrl}/feedbacks/remove`,{params:params})}
export const queryFeedbacks=params=>{ return axios.get(`${apiUrl}/feedbacks/query`,{params:params})}`
</pre>

本地跨域使用proxytable解决

如果你在启动看到如下的错误,那你就得用这个设置代理了,位置在config文件夹的index.js(前提是你是用vue-cli构建的项目)。
``xhr.js?14ed:177 XMLHttpRequest cannot load localhost:3000/feedbacks/query?name=&kyw=&bdate=&edate=. Cross origin requests are only supported for protocol schemes: http, data, chrome, chrome-extension, https.`
proxytable传送门:https://vuejs-templates.github.io/webpack/proxy.html
系统中设置的代理:
<pre>
dev: {
env: require('./dev.env'),
port: 8080,
autoOpenBrowser: true,
assetsSubDirectory: 'static',
assetsPublicPath: '/',
proxyTable: {//插件会虚拟一个引起跨域的服务器,代替8080端口发送请求,从而避免跨域
'/api':{//这个吧表的意思就是只要连接里有/api就转为http://127.0.0.1:3000
target:'http://127.0.0.1:3000',
changeOrigin:true,
pathRewrite:{
'^/api':''//这里意思就是转化后不用添加/api,比如/api/1就是http://127.0.01:3000/1
}
}
},
</pre>

express只负责查数据,逻辑处理还给前端

express很简单,只是进行查询,删除操作,这里查询,博主把条件查询,分页查询,全部查询都融合到同一个方法中
<pre>router.get('/remove', function (req, res, next) {
let _feedback = req.body;
console.log(_feedback.id);
let id = connection.escape(_feedback.id);
let sql = 'DELETE FROM feedbacks WHERE id=' + id;
connection.query(sql, function (err, results) {
if (results) {
res.json({ 'code': 1, message: '删除成功!' })
} else {
res.json({ 'code': 0, message: '删除失败!' })
}
})
})
router.get('/query', function (req, res, next) {
let _fb = req.query;
let page = (_fb.page - 1) * 10;//计算查询的起点
let ps = Number(_fb.pageSize);//计算结束点
console.log(_fb.name || _fb.kyw ? 1:0)
console.log(_fb.bdate ? 1:0)
console.log((_fb.bdate ? 1:0) & (_fb.name || _fb.kyw ? 1:0))
console.log(_fb.name || _fb.kyw)
console.log(_fb.bdate & (_fb.name || _fb.kyw))
let flitr = (_fb.name || _fb.kyw || _fb.bdate ? ' WHERE '//如果存在一条条件查询就加where,如果没有直接为空字符串
+ (_fb.name ? 'name="' + _fb.name + '"' : '')//判断name是否需要查询
+ (_fb.name & _fb.kyw ? ' AND ' : '')//如果两者都存在就加上and
+ (_fb.kyw ? ' info like "%' + _fb.kyw + '%"' : '') : '')//判断关键字是否需要查询
+ ((_fb.bdate ? 1:0) & (_fb.name || _fb.kyw ? 1:0) ? ' AND ' : '')//如果时间区间存在并且前面两个条件查询有一个存在就加上and
+ (_fb.bdate ? 'date BETWEEN "' + _fb.bdate + '" AND "' + _fb.edate + '"' : '')
let sql1 = 'SELECT count(*) as total FROM feedbacks ' + flitr + ' ; '
let sql2 = 'SELECT id, name,DATE_FORMAT(date,"%Y-%m-%d") date,info FROM feedbacks ' + flitr + ' LIMIT ' + page + ' , ' + ps
console.log(sql1 + sql2)
connection.query(sql1 + sql2, function (err, results) {
if (err) {
console.log(err);
res.json({
code: 0,
message: '[查询失败]' + err
})
} else {
res.json({
data: {
total: JSON.parse(JSON.stringify(results[0]))[0].total,//这里要特别注意,mysql查出来的数据是object但不是可以直接键值对就可以获得值得,需要json转化。
feedbacks: results[1]
},
code: 1,
message: '查询成功'
})
console.log(results[1])
}
})
})</pre>

最终效果图:

反馈页面模块效果

托管的代码地址:
<pre>
前端:https://github.com/githubziven/font
后台:https://github.com/githubziven/service
</pre>

好了,差不多了,终于把第三篇比较完整的写出来,有很多不完善的地方,将在后面几篇文章发布的同时,进行同步更新。下篇也就是第四篇,是关于框架的上传下载,就是全景图页面的实现。不足之处,评论指出,万分感谢!!

系列文章:

1、Vue2.0+Element+express+mysql 内容管理系统的搭建(一)
2、Vue2.0+Element+express+mysql 内容管理系统的搭建(二)
3、Vue2.0+Element+express+mysql 内容管理系统的搭建(三)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 01.{ 换行: Opening Brace Can't Be Placed on a Separate Lin...
    码农不器阅读 2,400评论 0 14
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • 文|糖小果 如今,微信拉近了我们上的距离,互联网上一分钟,几百人便可以组织成一个小团体,天南地北。 不出半个小时,...
    杉杉杂谈阅读 616评论 7 5
  • 昨晚概率论有个活动,写着写着就超了,原来自己还能记起这么多故事。 最早认识他的时候,觉得自己不可能喜欢上他。可是时...
    那夜最美的北京城阅读 732评论 0 0
  • 昨天早上先生准备出门上班,看到昨晚写的几个毛笔字,情不自已又欣赏了一番,末了又看了看少年养的蚕,没有食粮瘦...
    岁月清瘦好好胖着阅读 565评论 0 0