react-quill 富文本编辑器中自定义图片插入(图片上传)

react-quill 富文本编辑器插件是一款基础轻量的编辑器,引入也非常简单,如下:

import ReactQuill, { Quill } from 'react-quill';
class MyComponent extends React.Component {
  constructor(props) {
    super(props)
    this.state = { text: '' } // You can also pass a Quill Delta here
    this.handleChange = this.handleChange.bind(this)
  }
 
  handleChange(value) {
    this.setState({ text: value })
  }
 
  render() {
    return (
      <ReactQuill value={this.state.text}
                  onChange={this.handleChange} />
    )
  }
}

但是,编辑器中插入图片时,编辑器默认的处理是将图片转成base64保存,提交后端时请求数据太大,给数据库带来压力。

因此,进行改善,将图片先上传到数据库,使富文本编辑器里插入的img标签,src属性变成一个链接地址。具体效果见如下图:

react-quill自定义插入图片.gif

这个方案实现需要如下几个步骤:

第一步:

在富文本配置的toolbar中增加handler对象,设置其image属性,用于自定义函数去覆盖默认的方法,此处为点击图片上传按钮时,显示一个弹框。

this.modules={//富文本配置
            toolbar:{
                container:[
                    [{ 'font': fonts }],
                    [{ 'header': [1, 2, 3, 4, 5, 6, false] }],
                    ['bold', 'italic', 'underline','strike', 'blockquote'],
                    [{ 'color': [] }, { 'background': [] }],
                    [{'list': 'ordered'}, {'list': 'bullet'}, {'indent': '-1'}, {'indent': '+1'},{ 'align': [] }],
                    ['link', 'image'],
                    ['clean'],
                ],
                handlers: {
                    'image':this.showUploadBox.bind(this)
                }
            },
            imageDrop: true,
        };

//react组件中定义方法
showUploadBox(){
      this.setState({
            uploadBoxVisible:true
     });
}
//react组件render 中return modal结构
<Modal
        title="上传图片"
        visible={this.state.uploadBoxVisible}
        onCancel={this.hideUploadBox.bind(this)}
        onOk={this.handleUpload.bind(this,this.state.uploadUrl,this.state.file)}
        maskClosable={false}
        width={500}
>
         <div className="top_btn top_btn_upload" >
               <div>
                    <Button onClick={this.selectImage.bind(this)} style={{background:"#18ade4",border:"none",color:"#fff"}}>
                                选择图片
                    </Button>
                    <input ref="uploadInput" type='file' accept='image/*'
                               style={{width:"100px",border:"none",visibility:"hidden"}}
                               onChange={this.changeImageBeforeUpload.bind(this)}
                    />
              </div>
              <div style={{textAlign:"center",margin:"10px 0"}}>
                    {this.state.src?
                                <img src={this.state.src} alt="" style={{maxWidth:"100%",height:"300px"}}/>
                                :
                                <div style={{background:"#f2f2f2",width:"100%",height:"300px"}}>

                                </div>
                     }
               </div>
         </div>
</Modal>

第二步:

点击选择图片按钮时选择本地一张图片,具体实现代码如下:

//组件中定义选择图片的方法
selectImage(){
        this.refs.uploadInput.click();//点击modal的html结构中的input标签
}
//如上modal中input标签
<input ref="uploadInput" type='file' accept='image/*'
           style={{width:"100px",border:"none",visibility:"hidden"}}
           onChange={this.changeImageBeforeUpload.bind(this)}
/>

选择好图片后会调用input的onChange回调函数,这个回调函数可以做一些判断,其内容如下:

changeImageBeforeUpload(e){
        const file = e.target.files[0];
        if (!file) {
            return;
        }
        let src;
        // 匹配类型为image/开头的字符串
        if (file.type==="image/png"||file.type==="image/jpeg") {
            src = URL.createObjectURL(file);
        }else{
            message.error("图片上传只支持JPG/PNG格式,请重新上传!");
            return;
        }
        if (file.size/1024/1024>5) {
            message.error("图片上传大小不要超过5MB,请重新上传!");
            return;
        }
        this.setState({
            src:src,
            file:file
        })
    }

第三步:

选择好图片后,可以再选择图片进行替换,确认好后点击确认,进行图片上传。

/*开始上传图片*/
    handleUpload(uploadUrl,file){
        let this_=this;
        /*调用上传图片的封装方法*/
        uploadFunction.uploadForImage(
            uploadUrl,
            file,
            function (response) {//回调函数处理进度和后端返回值
                if (response&&response.code === 200) {
                    message.success("上传成功!");
                    this_.hideUploadBox();//隐藏弹框
                    this_.imageHandler(response.data.url);//处理插入图片到编辑器
                }else if (response && response.code !== 200) {
                    message.error(response.msg)
                }
            },
            localStorage.getItem("access_token"));
    }

如上使用的图片上传封装方法如下,将后端返回的数据传输回调函数中:

function uploadForImage(url,data,callback,token) {//data是数据列表
    console.log('post-请求接口:' + url);
    console.log('请求参数:' + data);
    let this_=this;
    if (!data) {
        console.log('未选择文件');
        return;
    }

    let xhr = new XMLHttpRequest();
    let form = new FormData();
    form.append('file', data);
    xhr.addEventListener('readystatechange',function(e){
        console.log(e);
        let response=e.target.response?JSON.parse(e.target.response):null;
        console.log(response);
        if (e.target.readyState===4&&response) {
            callback(response);
        }
    },false);
    xhr.open('POST', url, true);  // 第三个参数为async?,异步/同步
    xhr.setRequestHeader("accessToken",token);
    xhr.send(form);
}

第四步:

处理插入图片到编辑器,第三步有一个函数imageHandler(response.data.url),其中,response.data.url为返回的图片地址,imageHandler内容如下:

/*处理图片插入*/
    imageHandler(url){
            let quill=this.refs.reactQuillRef.getEditor();//获取到编辑器本身
            const cursorPosition =quill.getSelection().index;//获取当前光标位置
            quill.insertEmbed(cursorPosition, "image",url, Quill.sources.USER);//插入图片
            quill.setSelection(cursorPosition + 1);//光标位置加1
    }

到此为止,功能实现。
可以查看官方文档了解更多:
react-quill:https://www.npmjs.com/package/react-quill#import-the-component
quill:https://quilljs.com/docs/quickstart/

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