持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第20天,点击查看活动详情
一眨眼礼拜五了,说啥再水一篇博文,之后的话,小爷就可以去玩几把游戏了,嘿嘿~。
那么今天带来的主要是使用这个Vue整合到Markdown一个组件实现这个前端博客的编辑,然后是咱们的后端,实现这个图片上传,之后的话是咱们的这个文本差异对比的一个组件的使用吧(其实之所以要说这个是因为接下来要用的到的这两个组件间有差异冲突,先前也是搞了好久才发现这个问题的)
那么本文要做的呢就是三个事情嘛。
由于咱们需要使用到OSS服务,所以的话,还没有的同学可以查看这篇博文:懒人系列--文件上传之OSS使用案例 这个应该也是我写很详细的一篇博文了吧。
废话不多说,咱们先来看看咱们这个实现的效果吧。
在这里的话,我设计的玩意有两个东西,一个是博文,一个是问答,他们都是基于Markdown进行编辑的。 咱们这里只是测试哈(值得一提的是,咱们的话其实也是有这个文本的一些过滤的,敏感词汇之类的,当然实现的算法目前是很简单就是一个对比过滤,木有上机器学习,一个木有数据训练,一个是基于java的比较少咱们还得是用python,最后是资源消耗有点大)
之后是回答页面:
这个就简单了 这个图不好看,可以看这个:
(问题的回答这个暂时我是不想做限制的,因为确实有些问题的回答就是很复杂的)
至于图片上传的演示,这个都是哈。
这个主要是这个玩意:
(这个博文部分的功能还没写,其实和问答部分的一样,我在等这个社区功能做好) (这个是大家都可以看到的,但是只有作者才能够去进行操作,所以我们博文的展示完全是靠这个来看的,以内容为驱动(当然这个平台上线可能也没人,反正我做着玩))
那么为什么要说这个呢,原因很简单,这个和我们要用的Markdown组件之间存在一定的兼容问题。这个是啥情况我们待会再说。
ok,这个就是咱们的效果,那么咱们一个一个来:
这个组件的话有很多,那么在我们这里的话我是选择了 mavon-editor
这个使用的话其实非常简单,官方文档说的也还详细,主要我觉得最爽的是那个自己集成了代码高亮。
css
复制代码
npm install mavon-editor --save
为了方便,我这里选择全局注册使用。那么在main.js里面进行注册
javascript
复制代码
`//全局注册
import mavonEditor from 'mavon-editor'
import 'mavon-editor/dist/css/index.css' //解决编辑器的功能显示问题
Vue.use(mavonEditor)`
使用,使用的话很简单,在一个页面使用就好了
ini
复制代码
`<mavon-editor
class="md"
:value="webDataString"
:subfield="false"
:defaultOpen="'preview'"
:toolbarsFlag="false"
:editable="false"
:scrollStyle="true"
:ishljs="true"
/>`
但是这个的话,其实还是最基础的使用,这里是没有集成到图片上传的。
那么图片上传的话,咱们其实是有两个部分的,一个是前端,还有一个是后端。
我们先来说说前端吧。
ini
复制代码
`<mavon-editor
v-model="form.value"
ref="md"
@imgAdd="imgAdd"
@change="change"
style="min-height: 400px;width: 100%"
/>`
首先的话还是引入,但是这里的话绑定了两个函数 这两个一个就是上传图片的,还有一个就是拿到咱们Markdown文本的。
那么我们先来看到这个。
这个的话,咱们是有这个OSS的,所以上传也是到OSS里面的(其实主要是为了使用那个图片扫描,毕竟不是所有的图片都是可以看的,总有用户上传不友好的图片,文字))
javascript
复制代码
`toOss(pos,$file){
//组装数据
let formData = new FormData();
Object.keys(this.dataObj).forEach(key => {
formData.append(key, this.dataObj[key]);
});
formData.append('file',$file)
//此时发送请求去给到OSS
this.axios({
url: this.dataObj.host,
method: 'post',
data: formData
}).then((res) => {
let imgpath = this.dataObj.host + '/' + this.dataObj.key;
//把这个给到我们的编辑器
this.$refs.md.$img2Url(pos,imgpath)
})
},
imgAdd(pos, $file){
/**
let filename = $file.name
let _self = this;
// 获取认证码
this.axios
.get('/third-part/oss/quizWriteAnsImgPolicy')
.then(response => {
response = response.data;
_self.dataObj.policy = response.data.policy;
_self.dataObj.signature = response.data.signature;
_self.dataObj.ossaccessKeyId = response.data.accessid;
_self.dataObj.key = response.data.dir +getUUID()+"_"+filename;
_self.dataObj.dir = response.data.dir;
_self.dataObj.host = response.data.host;
//推送到OSS
this.toOss(pos,$file);
}).catch(function (error) {
alert(error)
console.log("出错了...",err)
})
},`
之后的话是获取我们这个文本。
javascript
复制代码
`change(value, render){
//value为编辑器中的实际内容,即markdown的内容,render为被解析成的html的内容
this.form.html = render;
},`
我这里的话,绑定了这个数据:
javascript
复制代码
this.form.html
通过这个就可以上传了:
javascript
复制代码
`submit(){
//点击提交后既可以获取html内容,又可以获得markdown的内容,之后存入到服务端就可以了
this.fullscreenLoading = true;
// console.log(this.form.value);
// console.log(this.form.html);
//将Markdown文档提交到服务器
let flag = true;
if(!this.form.value){
flag = false;
}else {
if(this.form.value.length<10){
flag = false;
alert("回答不能低于10个字符")
}
}
if(flag){
//此时对用户回答进行提交
this.axios({
url: "/quiz/quiz/base/baseUpAns",
method: 'post',
headers: {
"userid": this.userid,
"loginType": "PcType",
"loginToken": this.loginToken,
},
data:{
"userid": this.userid,
"quizid": this.Messages.quizid,
"quizTitle": this.Messages.quizTitle,
"context": this.form.value
}
}).then((res)=>{
res = res.data;
if(res.code===0){
alert(res.msg)
}else {
this.$message.error(res.msg);
}
this.fullscreenLoading = false;
this.editFlag = false;
});
}else {
alert("您的回答为空!")
}
},`
我的这部分代码是这样的(这里也没有去封装,天知道还要不要改接口,写好再封)
那么这个是我们前端比较重要的事情,那么接下来是我们的后端。
后端的话,其实有两个接口,第一个是授权OSS的,还有一个是保存咱们的这个回答的,也就是Markdown的。
那么我们这边的话,授权的话,这个已经说烂了,在那个OSS整合里面。
那么我们这边主要说说这个,敏感词过滤吧。
毕竟咱们这边上传其实就老三样。
首先是咱们的这个限制,就是防止恶意刷提交,由于是咱们的这个文本,里面的内容是有可能修改的,修改了多少其实也不太好评判,所以的话,不太好保证这个接口幂等,只能限制,这个限制的话咱们就是直接使用这个redis做的。
javascript
复制代码
`if(redisUtils.hasKey(RedisTransKey.getBaseUpQuizKey(entity.getUserid()))){
return R.error(BizCodeEnum.HAS_UPANS.getCode(), BizCodeEnum.HAS_UPANS.getMsg());
}
......
你的业务
成功之后
设置标志
redisUtils.set(RedisTransKey.setBaseUpAnsKey(entity.getUserid())
,1,5, TimeUnit.MINUTES
);`
反正就这样吧。然后这个redisUtils这个咱们先前也给出过是这样的:
java
复制代码
`public class RedisUtils {
@Autowired
private RedisTemplate<String,Object> redisTemplate;
/**
*/
public boolean expire(String key, long time) {
if (time > 0) {
redisTemplate.expire(key, time, TimeUnit.SECONDS);
return true;
} else {
throw new RuntimeException("超时时间小于0");
}
}
/**
*/
public long getExpire(String key) {
return redisTemplate.getExpire(key, TimeUnit.SECONDS);
}
/**
*/
public long getExpire(String key,TimeUnit tiemtype) {
return redisTemplate.getExpire(key, tiemtype);
}
/**
*/
public boolean hasKey(String key) {
return Boolean.TRUE.equals(redisTemplate.hasKey(key));
}
/**
*/
@SuppressWarnings("unchecked")
public void del(String... key) {
if (key != null && key.length > 0) {
if (key.length == 1) {
redisTemplate.delete(key[0]);
} else {
redisTemplate.delete(CollectionUtils.arrayToList(key));
}
}
}
// ============================String=============================
/**
*/
public Object get(String key) {
return key == null ? null : redisTemplate.opsForValue().get(key);
}
/**
*/
public boolean set(String key, Object value) {
redisTemplate.opsForValue().set(key, value);
return true;
}
/**
*/
public boolean set(String key, Object value, long time) {
if (time > 0) {
redisTemplate.opsForValue().set(key, value, time, TimeUnit.SECONDS);
} else {
this.set(key, value);
}
return true;
}
/**
*/
public boolean set(String key, Object value, long time,TimeUnit tiemtype) {
if (time > 0) {
redisTemplate.opsForValue().set(key, value, time, tiemtype);
} else {
this.set(key, value);
}
return true;
}
/**
*/
public long incr(String key, long delta) {
if (delta < 0) {
throw new RuntimeException("递增因子必须大于0");
}
return redisTemplate.opsForValue().increment(key, delta);
}
/**
*/
public long decr(String key, long delta) {
if (delta < 0) {
throw new RuntimeException("递减因子必须大于0");
}
return redisTemplate.opsForValue().increment(key, -delta);
}
// ================================Map=================================
/**
*/
public Object hget(String key, String item) {
return redisTemplate.opsForHash().get(key, item);
}
/**
*/
public Map<Object, Object> hmget(String key) {
return redisTemplate.opsForHash().entries(key);
}
/**
*/
public boolean hmset(String key, Map<String, Object> map) {
redisTemplate.opsForHash().putAll(key, map);
return true;
}
/**
*/
public boolean hmset(String key, Map<String, Object> map, long time) {
redisTemplate.opsForHash().putAll(key, map);
if (time > 0) {
expire(key, time);
}
return true;
}
/**
*/
public boolean hset(String key, String item, Object value) {
redisTemplate.opsForHash().put(key, item, value);
return true;
}
/**
*/
public boolean hset(String key, String item, Object value, long time) {
redisTemplate.opsForHash().put(key, item, value);
if (time > 0) {
expire(key, time);
}
return true;
}
/**
*/
public void hdel(String key, Object... item) {
redisTemplate.opsForHash().delete(key, item);
}
/**
*/
public boolean hHasKey(String key, String item) {
return redisTemplate.opsForHash().hasKey(key, item);
}
/**
*/
public double hincr(String key, String item, double by) {
return redisTemplate.opsForHash().increment(key, item, by);
}
/**
*/
public double hdecr(String key, String item, double by) {
return redisTemplate.opsForHash().increment(key, item, -by);
}
// ============================set=============================
/**
*/
public Set<Object> sGet(String key) {
return redisTemplate.opsForSet().members(key);
}
/**
*/
public boolean sHasKey(String key, Object value) {
return redisTemplate.opsForSet().isMember(key, value);
}
/**
*/
public long sSet(String key, Object... values) {
return redisTemplate.opsForSet().add(key, values);
}
/**
*/
public long sSetAndTime(String key, long time, Object... values) {
final Long count = redisTemplate.opsForSet().add(key, values);
if (time > 0)
expire(key, time);
return count;
}
/**
*/
public long sGetSetSize(String key) {
return redisTemplate.opsForSet().size(key);
}
/**
*/
public long setRemove(String key, Object... values) {
final Long count = redisTemplate.opsForSet().remove(key, values);
return count;
}
// ===============================list=================================
/**
*/
public List<Object> lGet(String key, long start, long end) {
return redisTemplate.opsForList().range(key, start, end);
}
/**
*/
public long lGetListSize(String key) {
return redisTemplate.opsForList().size(key);
}
/**
*/
public Object lGetIndex(String key, long index) {
return redisTemplate.opsForList().index(key, index);
}
/**
*/
public boolean lSet(String key, Object value) {
redisTemplate.opsForList().rightPush(key, value);
return true;
}
/**
*/
public boolean lSet(String key, Object value, long time) {
redisTemplate.opsForList().rightPush(key, value);
if (time > 0) {
expire(key, time);
}
return true;
}
/**
*/
public boolean lSetList(String key, List<Object> value) {
redisTemplate.opsForList().rightPushAll(key, value);
return true;
}
/**
*/
public boolean lSetList(String key, List<Object> value, long time) {
redisTemplate.opsForList().rightPushAll(key, value);
if (time > 0) {
expire(key, time);
}
return true;
}
/**
*/
public boolean lUpdateIndex(String key, long index, Object value) {
redisTemplate.opsForList().set(key, index, value);
return true;
}
}`
这个的话可以查看咱们的这篇文章: Java铭感词过滤
那么到这里咱们的这个Markdown的组件编辑就好了
之后就是展示,我们这里的话为了方便后面修改存储的都是Markdown的文本,所以的话,也是非常方便的。那么的话得益于这个组件,我们不需要那么麻烦,直接这样用就好了。
java
复制代码
`<!-- 这个是我们的回答的具体内容-->
<div style="width: 96%;margin: 0 auto">
<mavon-editor
class="md"
:boxShadow="false"
:value="Messages.context"
:subfield="false"
defaultOpen="preview"
:toolbarsFlag="true"
/>
</div>`
如果不想要这个工具栏的话,还可以:toolbarsFlag="false" 但是的话,咱们是要的主要是有目录,而且方便观看。
之后是代码的风格,我们现在这个: 这个是GitHub的一个风格,我们可以通过这个codeStyle等去设置。
更加完整的咱们去官网可以看到: github.com/hinesboy/ma…
我这里就不阐述了,前端的事情咋也不敢问。
之后就是咱们的代码差异对比了。
首先在组件上有很多的选择,一般很多博文,教程啥的用的都是这个: vue-code-diff
但是问题就在这里,首先这个玩意的话,其实也是有代码高亮的,没错就是这个:highlight 来实现的。 我们的那个编辑器组件的高亮也是这个来实现的。本来是没啥问题的,但是问题在于这个是9.x版本的,我们那个组件是10.x版本的,这个是我翻找依赖和官方文档和说明已经源码发现的。那么如果你使用的是这个那么这个玩意就会存在一定的冲突,后果就是你的代码显示区域很奇怪,全部压缩到了一行里面。
所我们需要使用另一个代替品。刚好有这个玩意: v-code-diff 这个是在 vue-code-diff 基础上开发的。
java
复制代码
npm i v-code-diff
之后的话是这个:
java
复制代码
npm i @vue/composition-api
因为我的是vue2,vue3不用,现在好多都是vue3,下次我再换vue3吧,没办法一开始学的时候是vue2后来没管了,一直用的就是2.
也是可以全局,可以局部使用的。 全局就是在main.js这样
java
复制代码
`import Vue from 'vue';
import CodeDiff from 'v-code-diff'
Vue.use(CodeDiff);`
然后直接使用
java
复制代码
`<template>
<code-diff
:old-string="'12345'"
:new-string="'3456'"
file-name="test.txt"
output-format="side-by-side"/>
</template>`
单独局部的话就是这样:
java
复制代码
`<template>
<code-diff
:old-string="'12345'"
:new-string="'3456'"
file-name="test.txt"
output-format="side-by-side"/>
</template>
<script>
import {CodeDiff} from 'v-code-diff'
export default {
name: 'App',
components: {
CodeDiff
}
}
</script>`
临时加水一篇~
原网址: 访问
创建于: 2024-05-14 14:23:11
目录: default
标签: 无
未标明原创文章均为采集,版权归作者所有,转载无需和我联系,请注明原出处,南摩阿彌陀佛,知识,不只知道,要得到
最新评论