mirror of
https://github.com/BookStackApp/BookStack.git
synced 2024-10-01 01:36:00 -04:00
232 lines
7.8 KiB
JavaScript
232 lines
7.8 KiB
JavaScript
|
import {register as registerShortcuts} from "./shortcuts";
|
||
|
import {listen as listenForCommonEvents} from "./common-events";
|
||
|
import {scrollToQueryString, fixScrollForMobile} from "./scrolling";
|
||
|
import {listenForDragAndPaste} from "./drop-paste-handling";
|
||
|
|
||
|
import {getPlugin as getCodeeditorPlugin} from "./plugin-codeeditor";
|
||
|
import {getPlugin as getDrawioPlugin} from "./plugin-drawio";
|
||
|
import {getPlugin as getCustomhrPlugin} from "./plugins-customhr";
|
||
|
import {getPlugin as getImagemanagerPlugin} from "./plugins-imagemanager";
|
||
|
|
||
|
const style_formats = [
|
||
|
{title: "Header Large", format: "h2", preview: 'color: blue;'},
|
||
|
{title: "Header Medium", format: "h3"},
|
||
|
{title: "Header Small", format: "h4"},
|
||
|
{title: "Header Tiny", format: "h5"},
|
||
|
{title: "Paragraph", format: "p", exact: true, classes: ''},
|
||
|
{title: "Blockquote", format: "blockquote"},
|
||
|
{title: "Inline Code", inline: "code"},
|
||
|
{
|
||
|
title: "Callouts", items: [
|
||
|
{title: "Info", format: 'calloutinfo'},
|
||
|
{title: "Success", format: 'calloutsuccess'},
|
||
|
{title: "Warning", format: 'calloutwarning'},
|
||
|
{title: "Danger", format: 'calloutdanger'}
|
||
|
]
|
||
|
},
|
||
|
];
|
||
|
|
||
|
const formats = {
|
||
|
codeeditor: {selector: 'p,h1,h2,h3,h4,h5,h6,td,th,div'},
|
||
|
alignleft: {selector: 'p,h1,h2,h3,h4,h5,h6,td,th,div,ul,ol,li,table,img', classes: 'align-left'},
|
||
|
aligncenter: {selector: 'p,h1,h2,h3,h4,h5,h6,td,th,div,ul,ol,li,table,img', classes: 'align-center'},
|
||
|
alignright: {selector: 'p,h1,h2,h3,h4,h5,h6,td,th,div,ul,ol,li,table,img', classes: 'align-right'},
|
||
|
calloutsuccess: {block: 'p', exact: true, attributes: {class: 'callout success'}},
|
||
|
calloutinfo: {block: 'p', exact: true, attributes: {class: 'callout info'}},
|
||
|
calloutwarning: {block: 'p', exact: true, attributes: {class: 'callout warning'}},
|
||
|
calloutdanger: {block: 'p', exact: true, attributes: {class: 'callout danger'}}
|
||
|
};
|
||
|
|
||
|
function file_picker_callback(callback, value, meta) {
|
||
|
|
||
|
// field_name, url, type, win
|
||
|
if (meta.filetype === 'file') {
|
||
|
window.EntitySelectorPopup.show(entity => {
|
||
|
callback(entity.link, {
|
||
|
text: entity.name,
|
||
|
title: entity.name,
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
|
||
|
if (meta.filetype === 'image') {
|
||
|
// Show image manager
|
||
|
window.ImageManager.show(function (image) {
|
||
|
callback(image.url, {alt: image.name});
|
||
|
}, 'gallery');
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param {WysiwygConfigOptions} options
|
||
|
* @return {string}
|
||
|
*/
|
||
|
function buildToolbar(options) {
|
||
|
const textDirPlugins = options.textDirection === 'rtl' ? 'ltr rtl' : '';
|
||
|
|
||
|
const toolbar = [
|
||
|
'undo redo',
|
||
|
'styleselect',
|
||
|
'bold italic underline strikethrough superscript subscript',
|
||
|
'forecolor backcolor',
|
||
|
'alignleft aligncenter alignright alignjustify',
|
||
|
'bullist numlist outdent indent',
|
||
|
textDirPlugins,
|
||
|
'table imagemanager-insert link hr codeeditor drawio media',
|
||
|
'removeformat code ${textDirPlugins} fullscreen'
|
||
|
];
|
||
|
|
||
|
return toolbar.filter(row => Boolean(row)).join(' | ');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param {WysiwygConfigOptions} options
|
||
|
* @return {string}
|
||
|
*/
|
||
|
function gatherPlugins(options) {
|
||
|
const plugins = [
|
||
|
"image",
|
||
|
"imagetools",
|
||
|
"table",
|
||
|
"paste",
|
||
|
"link",
|
||
|
"autolink",
|
||
|
"fullscreen",
|
||
|
"code",
|
||
|
"customhr",
|
||
|
"autosave",
|
||
|
"lists",
|
||
|
"codeeditor",
|
||
|
"media",
|
||
|
"imagemanager",
|
||
|
options.textDirection === 'rtl' ? 'directionality' : '',
|
||
|
];
|
||
|
|
||
|
window.tinymce.PluginManager.add('codeeditor', getCodeeditorPlugin(options));
|
||
|
window.tinymce.PluginManager.add('customhr', getCustomhrPlugin(options));
|
||
|
window.tinymce.PluginManager.add('imagemanager', getImagemanagerPlugin(options));
|
||
|
|
||
|
if (options.drawioUrl) {
|
||
|
window.tinymce.PluginManager.add('drawio', getDrawioPlugin(options));
|
||
|
plugins.push('drawio');
|
||
|
}
|
||
|
|
||
|
return plugins.filter(plugin => Boolean(plugin)).join(' ');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Load custom HTML head content from the settings into the editor.
|
||
|
* TODO: We should be able to get this from current parent page?
|
||
|
* @param {Editor} editor
|
||
|
*/
|
||
|
function loadCustomHeadContent(editor) {
|
||
|
window.$http.get(window.baseUrl('/custom-head-content')).then(resp => {
|
||
|
if (!resp.data) return;
|
||
|
let head = editor.getDoc().querySelector('head');
|
||
|
head.innerHTML += resp.data;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param {WysiwygConfigOptions} options
|
||
|
* @return {function(Editor)}
|
||
|
*/
|
||
|
function getSetupCallback(options) {
|
||
|
return function(editor) {
|
||
|
editor.on('ExecCommand change input NodeChange ObjectResized', editorChange);
|
||
|
listenForCommonEvents(editor);
|
||
|
registerShortcuts(editor);
|
||
|
listenForDragAndPaste(editor, options);
|
||
|
|
||
|
editor.on('init', () => {
|
||
|
editorChange();
|
||
|
scrollToQueryString(editor);
|
||
|
fixScrollForMobile(editor);
|
||
|
window.editor = editor;
|
||
|
});
|
||
|
|
||
|
function editorChange() {
|
||
|
const content = editor.getContent();
|
||
|
if (options.darkMode) {
|
||
|
editor.contentDocument.documentElement.classList.add('dark-mode');
|
||
|
}
|
||
|
window.$events.emit('editor-html-change', content);
|
||
|
}
|
||
|
|
||
|
// TODO - Update to standardise across both editors
|
||
|
// Use events within listenForBookStackEditorEvents instead (Different event signature)
|
||
|
window.$events.listen('editor-html-update', html => {
|
||
|
editor.setContent(html);
|
||
|
editor.selection.select(editor.getBody(), true);
|
||
|
editor.selection.collapse(false);
|
||
|
editorChange(html);
|
||
|
});
|
||
|
|
||
|
// Custom handler hook
|
||
|
window.$events.emitPublic(options.containerElement, 'editor-tinymce::setup', {editor});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param {WysiwygConfigOptions} options
|
||
|
* @return {Object}
|
||
|
*/
|
||
|
export function build(options) {
|
||
|
|
||
|
return {
|
||
|
width: '100%',
|
||
|
height: '100%',
|
||
|
selector: '#html-editor',
|
||
|
content_css: [
|
||
|
window.baseUrl('/dist/styles.css'),
|
||
|
],
|
||
|
branding: false,
|
||
|
skin: options.darkMode ? 'oxide-dark' : 'oxide',
|
||
|
body_class: 'page-content',
|
||
|
browser_spellcheck: true,
|
||
|
relative_urls: false,
|
||
|
directionality: options.textDirection,
|
||
|
remove_script_host: false,
|
||
|
document_base_url: window.baseUrl('/'),
|
||
|
end_container_on_empty_block: true,
|
||
|
statusbar: false,
|
||
|
menubar: false,
|
||
|
paste_data_images: false,
|
||
|
extended_valid_elements: 'pre[*],svg[*],div[drawio-diagram]',
|
||
|
automatic_uploads: false,
|
||
|
valid_children: "-div[p|h1|h2|h3|h4|h5|h6|blockquote],+div[pre],+div[img]",
|
||
|
plugins: gatherPlugins(options),
|
||
|
imagetools_toolbar: 'imageoptions',
|
||
|
contextmenu: false,
|
||
|
toolbar: buildToolbar(options),
|
||
|
content_style: `html, body, html.dark-mode {background: ${options.darkMode ? '#222' : '#fff'};} body {padding-left: 15px !important; padding-right: 15px !important; margin:0!important; margin-left:auto!important;margin-right:auto!important;}`,
|
||
|
style_formats,
|
||
|
style_formats_merge: false,
|
||
|
media_alt_source: false,
|
||
|
media_poster: false,
|
||
|
formats,
|
||
|
file_picker_types: 'file image',
|
||
|
file_picker_callback,
|
||
|
paste_preprocess(plugin, args) {
|
||
|
let content = args.content;
|
||
|
if (content.indexOf('<img src="file://') !== -1) {
|
||
|
args.content = '';
|
||
|
}
|
||
|
},
|
||
|
init_instance_callback(editor) {
|
||
|
loadCustomHeadContent(editor);
|
||
|
},
|
||
|
setup: getSetupCallback(options),
|
||
|
};
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @typedef {Object} WysiwygConfigOptions
|
||
|
* @property {Element} containerElement
|
||
|
* @property {boolean} darkMode
|
||
|
* @property {string} textDirection
|
||
|
* @property {string} drawioUrl
|
||
|
* @property {int} pageId
|
||
|
* @property {Object} translations
|
||
|
*/
|