@veroni1337

Как добавить новый функционал в редактор prosemirror?

Здравствуйте. Стоит задача добавить функционал чекбоксов для формирования todo листа в редакторе prosemirror в проекте на HumHub. HumHub написан на YII без использования js фреймворков на фронтенде. Пытаюсь все сделать как на примере ссылка, но на humhub все выглядит иначе чем на примерах.

В проекте по пути /protected/humhub/modules/content/resources/js/humhub.ui.richtext.prosemirror.js

Внутри:
/*
 * @link https://www.humhub.org/
 * @copyright Copyright (c) 2018 HumHub GmbH & Co. KG
 * @license https://www.humhub.com/licences
 * @since 1.8
 */
humhub.module('ui.richtext.prosemirror', function(module, require, $) {

    var object = require('util').object;
    var client = require('client');
    var Widget = require('ui.widget').Widget;
    var additions = require('ui.additions');

    var MarkdownEditor = prosemirror.MarkdownEditor;
    var MentionProvider = prosemirror.MentionProvider;

    var RichTextEditor = Widget.extend();

    RichTextEditor.component = 'humhub-ui-richtexteditor';

    RichTextEditor.prototype.getDefaultOptions = function() {
        return {
            attributes:  {
                'class': 'atwho-input form-control humhub-ui-richtext',
                'data-ui-markdown': true,
            },
            mention: {
                provider: new HumHubMentionProvider(module.config.mention)
            },
            link: {
              validate: module.config.validate
            },
            emoji: module.config.emoji,
            oembed: module.config.oembed,
            translate: function(key) {
                return module.text(key);
            }
        };
    };

    RichTextEditor.prototype.init = function() {
        if(this.options.placeholder) {
            this.options.placeholder = {
                text: this.options.placeholder,
                'class' : 'placeholder atwho-placeholder'
            };
        }

        if(this.options.disabled) {
            setTimeout($.proxy(this.disable, this), 50);
        }

        //var options = $.extend({}, this.options, {exclude: ['blockquote', 'bullet_list', 'strong', 'code', 'code_block', 'em', 'image', 'list_item', 'ordered_list', 'heading', 'link', 'clipboard']});

        this.editor = new MarkdownEditor(this.$, this.options);
        this.editor.init(this.getInitValue());

        if(this.options.focus) {
            this.editor.view.focus();
        }

        var that = this;
        this.$.on('focusout', function() {
            that.getInput().val(that.editor.serialize());
        }).on('clear', function() {
            that.editor.clear();
        }).on('focus', function() {
            that.focus();
        });

        this.$.find('.humhub-ui-richtext').on('focus', function() {
            that.focus();
        })

        if (this.options.backupInterval) {
            setInterval(function () {
                that.backup();
            }, this.options.backupInterval * 1000);
        }
    };

    RichTextEditor.prototype.getInitValue = function() {
        var inputId = this.getInput().attr('id');
        var backup = this.getBackup();

        if (typeof backup[inputId] === 'string' && backup[inputId] !== '') {
            return backup[inputId];
        }

        return this.$.find('[data-ui-richtext]').text();
    }

    RichTextEditor.prototype.getBackup = function() {
        var backup = sessionStorage.getItem(this.options.backupCookieKey);

        if (typeof backup === 'string' && backup !== '') {
            return JSON.parse(backup);
        }

        return {};
    }

    RichTextEditor.prototype.backup = function() {
        var inputId = this.getInput().attr('id');
        var currentValue = this.editor.serialize();
        var isBackuped = typeof this.backupedValue !== 'undefined';

        if (!isBackuped && currentValue === '') {
            // Don't back up first empty value
            return;
        }

        if (isBackuped && currentValue === this.backupedValue) {
            // Don't back up same content twice
            return;
        }

        this.backupedValue = currentValue;

        var backup = this.getBackup();
        if (this.backupedValue === '' && typeof backup[inputId] !== 'undefined') {
            delete backup[inputId];
        } else {
            backup[inputId] = this.backupedValue;
        }

        if (Object.keys(backup).length) {
            sessionStorage.setItem(this.options.backupCookieKey, JSON.stringify(backup));
        } else {
            sessionStorage.removeItem(this.options.backupCookieKey);
        }
    };

    RichTextEditor.prototype.focus = function() {
        this.editor.view.focus();
    };

    RichTextEditor.prototype.disable = function(tooltip) {
        tooltip = tooltip || this.options.disabledText;
        $(this.editor.view.dom).removeAttr('contenteditable').attr({
            disabled: 'disabled',
            title: tooltip,
        }).tooltip({
            placement: 'bottom'
        });
    };

    RichTextEditor.prototype.getInput = function() {
        return $('#'+this.$.attr('id')+'_input');

    };

    var RichText = Widget.extend();

    RichText.component = 'humhub-ui-richtext';

    RichText.prototype.init = function() {
        // If in edit mode we do not actually render, we just hold the content
        if(!this.options.edit) {
            this.editor = new MarkdownEditor(this.$, this.options);
            this.$.html(this.editor.render());
            additions.applyTo(this.$, {filter: ['highlightCode']});
            this.$.find('table').wrap('<div class="table-responsive"></div>');
            this.$.trigger('afterRender');
        }

        // See https://github.com/ProseMirror/prosemirror/issues/432
        document.execCommand('enableObjectResizing', false, 'false');
        document.execCommand('enableInlineTableEditing', false, 'false');
    };

    HumHubMentionProvider = function(options) {
        MentionProvider.call(this, options);
    };

    object.inherits(HumHubMentionProvider, MentionProvider);

    HumHubMentionProvider.prototype.find = function(query, node) {
        if(this.xhr) {
            this.xhr.abort();
        }

        var that = this;
        var $editor = Widget.closest(node);

        return new Promise(function(resolve, reject) {
            client.get($editor.options.mentioningUrl, {
                data: {keyword: query},
                beforeSend: function(jqXHR) {
                    that.xhr = jqXHR;
                }
            }).then(function(response) {
                resolve(response.data);
            }).catch(function(err) {
                reject(reject)
            });
        });

    };

    /**
     * Builds mentioning string from container link
     * @param $containerLink
     * @returns {string}
     */
    var buildMentioning = function($containerLink) {
        var username = $containerLink.text();
        var guid = $containerLink.data('guid');
        var url = $containerLink.attr('href');
        return '['+username+'](mention:'+guid+' "'+url+'")';
    };

    module.export({
        initOnPjaxLoad: true,
        unload: function(pjax) {
            $('.humhub-richtext-provider').remove();
            $('.ProseMirror-prompt').remove();
        },
        RichTextEditor: RichTextEditor,
        RichText: RichText,
        buildMentioning: buildMentioning,
        api: prosemirror
    });
});


Я так понимаю, это код для управления редактором. Чат гпт помог написать такой код, который выводит кнопку в меню инструментов и выполняет функцию при нажатии на него:
// Функция для добавления кнопки
    RichTextEditor.prototype.addTestButton = function () {
        var that = this;

        // Добавляем кнопку в toolbar ProseMirror
        var itemWrap = document.createElement("div");

        var button = document.createElement("div");
        button.innerHTML = "ToDo";
        button.classList.add("ProseMirror-icon", "ProseMirror-menu-insertToDo");
        button.onclick = function () {
        that.addTestText();
        };

        // Вставляем кнопку в toolbar (настраивается в зависимости от вашего DOM)
        this.$.find(".ProseMirror-menubar > .ProseMirror-menu-group").append(
        button
        );
    };

    // Функция для вставки чекбокса в редактор
    RichTextEditor.prototype.addTestText = function () {
        const state = this.editor.view.state;
        const dispatch = this.editor.view.dispatch;
    
        // Получаем схему редактора
        const { schema } = state;

        // Проверяем, определены ли узлы bullet_list и list_item
        if (!schema.nodes.bullet_list || !schema.nodes.list_item) {
        console.error("Узлы bullet_list или list_item не определены в схеме.");
        return;
        }
    
        // Создаём пустой параграф (контейнер для текстового ввода)
        const paragraphNode = schema.nodes.paragraph.create(null, schema.text("-")); // Параграф с пустым текстом
    
        // Создаём узел list_item с пустым параграфом
        const listItemNode = schema.nodes.list_item.create(
        { done: false }, // Атрибуты узла
        paragraphNode // Дочерний узел - параграф
        );
    
        // Создаём узел bullet_list с элементами списка
        const bulletListNode = schema.nodes.bullet_list.create(
        null, // Атрибуты узла
        listItemNode // Дочерний узел - элемент списка
        );

        const transaction = state.tr.insert(state.selection.from, bulletListNode);
        dispatch(transaction);
    };


Затем он вызывается в функции RichTextEditor.prototype.init
  • Вопрос задан
  • 72 просмотра
Пригласить эксперта
Ваш ответ на вопрос

Войдите, чтобы написать ответ

Похожие вопросы