小程序H5页面使用富文本quill

let speakEditor = {};
if (process.env.TARO_ENV === 'h5') {
    speakEditor = require('./quill-editor');
}
const SpeakEditor = speakEditor.default || <View />;

// 外层back使用相对定位,这样SpeakEditor使用绝对定位时位置就没问题了
<View className={style.back}>
    <SpeakEditor onInput={handleInputChange} value={''} />
</View>


// SpeakEditor组件index.jsx
import {createRef, useCallback, useEffect, useImperativeHandle, useRef} from 'react';
import {Events} from '@tarojs/taro';
import {View} from '@tarojs/components';
import ReactQuill from 'react-quill';
import 'react-quill/dist/quill.snow.css';
import 'tippy.js/dist/tippy.css';
import style from './style.less';

export const events = new Events();

const oneSelfRef = createRef();
export default function SpeakEditor(props) {
    const {value, onInput} = props;
    const reactQuillRef = useRef();
    const quillRef = useRef();

    useImperativeHandle(oneSelfRef, () => ({
        getText() {
            return quillRef.current.getText();
        },
        getContents() {
            return quillRef.current.getContents();
        },
        loadContents(contents) {
            quillRef.current.setContents(contents);
        },
        editorEvent(eventName, handler) {
            quillRef.current.root.addEventListener(eventName, handler);
        },
        editorOffEvent(eventName, handler) {
            quillRef.current.root.removeEventListener(eventName, handler);
        },
        call(method, ...args) {
            return quillRef.current[method](...args);
        },
        focusOnSelection() {
            // 滚动到光标处
            const editor = quillRef.current; // quill编辑器实例
            const selection = editor.getSelection(); // 获取选中范围
            if (selection) {
                const [line] = editor.getLines(selection.index, 1);
                if (line) {
                    editor.scrollIntoView(line);
                }
            }
        }
    }));

    useEffect(() => {
        quillRef.current = reactQuillRef.current.getEditor();
    }, []);

    const handleChange = useCallback(
        (_value, delta, source, editor) => {
            // console.log('Change: ', {_value, delta, source, editor}, editor.getContents());
            onInput({detail: {value: _value}});
        },
        [onInput]
    );

    return (
        <View className={style.quill_editor}>
            <ReactQuill theme="snow" value={value} onChange={handleChange} ref={reactQuillRef} />
        </View>
    );
}

export function event(eventName, handler) {
    if (oneSelfRef.current) {
        oneSelfRef.current.call('on', eventName, handler);
    } else if (event.counter < 20) { 
        event.counter++; 
        setTimeout(()=> {
            event(eventName, handler);
        }, 200);
    }
}
event.counter = 0;

export function offEvent(eventName, handler) {
    oneSelfRef.current?.call('off', eventName, handler);
}

export function editorEvent(eventName, handler) {
    if (oneSelfRef.current) {
        oneSelfRef.current.editorEvent(eventName, handler);
    } else if (editorEvent.counter < 20) { editorEvent.counter++; setTimeout(()=> {
        oneSelfRef.current.editorEvent(eventName, handler);
        }, 200);
    }
}
editorEvent.counter = 0;

export function editorOffEvent(eventName, handler) {
    oneSelfRef.current.editorOffEvent(eventName, handler);
}

export function on(eventName, handler) {
    events.on(eventName, handler);
}

export function off(eventName, handler) {
    if (!eventName || !handler) {
        events.off();
    } else {
        events.off(eventName, handler);
    }
}

export function getText() {
    return oneSelfRef.current?.getText();
}

export function getContents() {
    return oneSelfRef.current?.getContents();
}

export function loadContents(contents) {
    return oneSelfRef.current?.loadContents(contents);
}

export function doBlur() {
    console.log('doBlur');
    return oneSelfRef.current?.call('blur');
}

export function doFocus() {
    console.log('doFocus');
    return oneSelfRef.current?.call('focus');
}

export function focusOnSelection() {
    oneSelfRef.current?.focusOnSelection();
}

// SpeakEditor组件style.less
.quill_editor {
    position: absolute;
    top: 0;
    right: 0;
    bottom: 0;
    left: 0;
    width: 100%;

    :global {
        .quill {
            width: 100%;
            height: 100%;
            user-select: text;
        }

        .ql-toolbar {
            display: none;
        }

        .ql-container {
            font-size: 28px;
            color: white;
            border: none;
        }

        .ql-editor {
            padding: 0;
            line-height: 38px;
            letter-spacing: 4px;
        }

        .pause_embed {
            display: inline-block;
            margin: 6px 0;
            margin-right: 3px;
            font-size: 28px !important;
            color: #468df7;
            pointer-events: auto;
        }

        .reading_method {
            display: inline-block;
            padding: 0;
            padding-left: 4px;
            margin: 6px 0;
            margin-right: 3px;
            font-size: inherit !important;
            font-weight: inherit !important;
            pointer-events: none;
            // background: rgb(66, 44, 54);
            background: rgba(239, 161, 56, 0.32);
            border-radius: 4px;

            &::after {
                display: inline-block;
                height: 100%;
                padding: 3px 4px;
                margin-left: 4px;
                font-size: 22px;
                color: #efa138;
                pointer-events: auto;
                background: rgba(239, 161, 56, 0.16);
                content: attr(data-str);
            }
        }

        .shift_speed {
            display: inline-block;
            margin: 6px 0;
            margin-right: 3px;
            font-size: inherit !important;
            font-weight: inherit !important;
            pointer-events: none;

            &::before {
                color: #468df7;
                content: '【';
            }

            &::after {
                color: #468df7;
                pointer-events: auto;
                content: '】' attr(data-value) 'x';
            }
        }

        .chars_check {
            display: inline-block;
            padding: 0;
            padding-left: 4px;
            margin: 6px 0;
            margin-right: 3px;
            font-size: inherit !important;
            font-weight: inherit !important;
            // pointer-events: none;
            background: rgba(82, 196, 26, 0.16);
            border-radius: 4px;

            &::after {
                display: inline-block;
                padding: 3px 4px;
                margin-bottom: 5px;
                margin-left: 4px;
                font-size: 22px;
                vertical-align: middle;
                // pointer-events: auto;
                content: attr(data-str);
            }

            &[data-str=''] {
                &::after {
                    width: 24px;
                    height: 24px;
                    margin-right: 7px;
                    margin-left: 3px;
                    background: center no-repeat;
                    background-size: contain;
                    background-image: url();
                }
            }
        }
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值