react-draft-wysiwyg富文本编辑器 图片上传到oss

react项目中 使用react-draft-wysiwyg这个富文本组件,因为公司项目图片都是上传到阿里云oss服务器上,废话不说上代码,代码如下:

// 安装组件

yarn add react-draft-wysiwyg draftjs-to-html  html-to-draftjs draft-js  js-md5 axios

或者用npm install ** 来安装

// 引用组件

import axios from 'axios';

import {EditorState, convertToRaw, ContentState} from 'draft-js';

import 'react-draft-wysiwyg/dist/react-draft-wysiwyg.css';

import{Editor} from 'react-draft-wysiwyg';

import draftToHtml from 'draftjs-to-html';

import htmlToDraft from 'html-to-draftjs';

import md5 from 'js-md5';

export default class Wysiwyg extends Component{

    constructor(props) {

        super(props);

       // 编辑器里的默认值

        const html = '<p>Hey this <strong>editor</strong> rocks 😀</p>';

        const contentBlock = htmlToDraft(html);

        if (contentBlock) {

          const contentState = ContentState.createFromBlockArray(contentBlock.contentBlocks);

          const editorState = EditorState.createWithContent(contentState);

          this.state = {

            editorState,

          };

        }

    }

    componentWillMount(){

        axios({                               

            url: '**/frontend_sts_token',  //后台接口的链接,主要返回上传到oss要用到的key, accesskeyid等

            method: 'GET',

            timeout: 4000,

            headers: {

                'content-type': 'application/json',

                'Authorization': "Bearer token"  //用到的token是从后台获取到, 这看后端要怎么传给他们,

            },

            params: ''

        }).then((response)=>{

            console.log(response.data.data)

            this.setState({

                frontend_sts_token: response.data.data

            })

        })

    }

//编辑器输入方法

    onEditorStateChange = Function = (editorState)=>{

        this.setState({

            editorState

        })

    }

//图片上传到oss的方法

    uploadImageCallBack = Function = (file)=>{

        console.log(file)

        const {frontend_sts_token} = this.state;

        return new Promise((resolve, reject) => {

            let name = md5(file.name) +".jpg";  //生成md5文件名

            let formData = new FormData();

            let _key = frontend_sts_token.dir + name;

            // 文件名字,可设置路径

            console.log(_key)

            formData.append('key',  frontend_sts_token.dir + name);

            // policy规定了请求的表单域的合法性

            formData.append('policy', frontend_sts_token.policy)

            // Bucket 拥有者的Access Key Id

            formData.append('OSSAccessKeyId', frontend_sts_token.accessid)

            // 让服务端返回200,不然,默认会返回204

            formData.append('success_action_status', '200')

            // 根据Access Key Secret和policy计算的签名信息,OSS验证该签名信息从而验证该Post请求的合法性

            formData.append('callback', frontend_sts_token.callback)

            formData.append('signature', frontend_sts_token.signature)

            formData.append('name', name)

            formData.append('file', file)

            console.log(formData)

            axios({

                url: frontend_sts_token.host,

                method: 'post',

                data: formData,

                headers: { 'Content-Type': 'multipart/form-data' } ,

                // 或者

                // headers: {

                //    'Content-Type': 'application/x-www-form-urlencoded'

                // },

            })

            .then((res) => {

                console.log(res)

                //添加到虚线框

                let imgurl = res.data.filename;

                let imgObj = {

                    data:{

                        link: res.data.oss_domain + imgurl

                    }

                }

                resolve(imgObj)

            })

            .catch((err) => { reject(err) })

        });

    }

    render(){

        console.log(EditorState.createEmpty())

        const { editorState } = this.state;

        return (

            <div>

                <Editor

                    editorState={editorState}

                    wrapperClassName="wrapper-class"

                    editorClassName="editor-class"

                    toolbarClassName="toolbar-class"

                    onEditorStateChange={this.onEditorStateChange}

                    localization={{

                        locale: 'zh',

                    }}

                    toolbar={{

                        options: ['inline', 'colorPicker', 'blockType', 'fontSize', 'fontFamily', 'list', 'textAlign', 'link', 'embedded', 'emoji', 'image', 'remove', 'history'],

                        colorPicker: {

                          className: undefined,

                          component: undefined,

                          popupClassName: undefined,

                          colors: ['rgb(97,189,109)', 'rgb(26,188,156)', 'rgb(84,172,210)', 'rgb(44,130,201)',

                            'rgb(147,101,184)', 'rgb(71,85,119)', 'rgb(204,204,204)', 'rgb(65,168,95)', 'rgb(0,168,133)',

                            'rgb(61,142,185)', 'rgb(41,105,176)', 'rgb(85,57,130)', 'rgb(40,50,78)', 'rgb(0,0,0)',

                            'rgb(247,218,100)', 'rgb(251,160,38)', 'rgb(235,107,86)', 'rgb(226,80,65)', 'rgb(163,143,132)',

                            'rgb(239,239,239)', 'rgb(255,255,255)', 'rgb(250,197,28)', 'rgb(243,121,52)', 'rgb(209,72,65)',

                            'rgb(184,49,47)', 'rgb(124,112,107)', 'rgb(209,213,216)'],

                        },

                        image: {

                            urlEnabled: true,

                            uploadEnabled: true,

                            alignmentEnabled: true,

                            uploadCallback: this.uploadImageCallBack,

                            previewImage: true,

                            inputAccept: 'image/*',

                            alt:{ present: true, mandatory: true }

                        }

                    }}

                />

                <textarea

                    disabled

                    value={draftToHtml(convertToRaw(editorState.getCurrentContent()))}

                />

            </div>

        )

    }

}


github下载地址:react-draft-wysiwyg-oss

参考资料:

https://jpuri.github.io/react-draft-wysiwyg/#/docs

https://jpuri.github.io/react-draft-wysiwyg/#/demo

https://www.cnblogs.com/xufeimei/p/9804659.html

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