cthep-ui-vue3
Version:
Vue 3 试题组件库
466 lines (427 loc) • 22 kB
JavaScript
import IntegrationModel from '@wiris/mathtype-html-integration-devkit/src/integrationmodel';
import Configuration from '@wiris/mathtype-html-integration-devkit/src/configuration';
import Parser from '@wiris/mathtype-html-integration-devkit/src/parser';
import Util from '@wiris/mathtype-html-integration-devkit/src/util';
import Listeners from '@wiris/mathtype-html-integration-devkit/src/listeners';
import packageInfo from './package.json';
/**
* TinyMCE integration class. This class extends IntegrationModel class.
*/
export class TinyMceIntegration extends IntegrationModel {
constructor(integrationModelProperties) {
super(integrationModelProperties);
/**
* Indicates if the content of the TinyMCe editor has
* been parsed.
* @type {Boolean}
*/
this.initParsed = integrationModelProperties.initParsed;
/**
* Indicates if the TinyMCE is integrated in Moodle.
* @type {Boolean}
*/
this.isMoodle = integrationModelProperties.isMoodle;
/**
* Indicates if the plugin is loaded as an external plugin by TinyMCE.
* @type {Boolean}
*/
this.isExternal = integrationModelProperties.isExternal;
}
/**
* Returns the absolute path of the integration script. Depends on
* TinyMCE integration (Moodle or standard).
* @returns {Boolean} - Absolute path for the integration script.
*/
getPath() {
if (this.isMoodle) {
// return '/lib/editor/tinymce/plugins/tiny_mce_wiris/tinymce/';
const search = 'lib/editor/tinymce';
const pos = tinymce.baseURL.indexOf(search);
const baseURL = tinymce.baseURL.substr(0, pos + search.length);
return `${baseURL}/plugins/tiny_mce_wiris/tinymce/`;
} if (this.isExternal) {
const externalUrl = this.editorObject.getParam('external_plugins').tiny_mce_wiris;
return externalUrl.substring(0, externalUrl.lastIndexOf('/') + 1);
}
return `${tinymce.baseURL}/plugins/tiny_mce_wiris/`;
}
/**
* Returns the absolute path of plugin icons. A set of different
* icons is needed for TinyMCE and Moodle 2.5-
* @returns {String} - Absolute path of the icons folder.
*/
getIconsPath() {
if (this.isMoodle && Configuration.get('versionPlatform') < 2013111800) {
return `${this.getPath()}icons/tinymce3/`;
}
return `${this.getPath()}icons/`;
}
/**
* Returns the integration language. TinyMCE language is inherited.
* When no language is set, TinyMCE sets the toolbar to english.
* @returns {String} - Integration language.
*/
getLanguage() {
const editorSettings = this.editorObject.settings;
try {
return editorSettings.mathTypeParameters.editorParameters.language;
} catch (e) { console.error(); }
// Get the deprecated wirisformulaeditorlang
if (editorSettings.wirisformulaeditorlang) {
console.warn('Deprecated property wirisformulaeditorlang. Use mathTypeParameters on instead.');
return editorSettings.wirisformulaeditorlang;
}
const langParam = this.editorObject.getParam('language');
return langParam || super.getLanguage();
}
/**
* Callback function called before 'onTargetLoad' is fired. All the logic here is to
* avoid TinyMCE change MathType formulas.
*/
callbackFunction() {
const dataImgFiltered = [];
super.callbackFunction();
// Avoid to change class of image formulas.
const imageClassName = Configuration.get('imageClassName');
if (this.isIframe) {
// Attaching observers to wiris images.
if (typeof Parser.observer !== 'undefined') {
Array.prototype.forEach.call(this.target.contentDocument.getElementsByClassName(imageClassName), (wirisImages) => {
Parser.observer.observe(wirisImages);
});
}
} else { // Inline.
// Attaching observers to wiris images.
Array.prototype.forEach.call(document.getElementsByClassName(imageClassName), (wirisImages) => {
Parser.observer.observe(wirisImages);
});
}
// When a formula is updated TinyMCE 'Change' event must be fired.
// See https://www.tiny.cloud/docs/advanced/events/#change for further information.
const listener = Listeners.newListener('onAfterFormulaInsertion', () => {
if (typeof this.editorObject.fire !== 'undefined') {
this.editorObject.fire('Change');
}
});
this.getCore().addListener(listener);
// Avoid filter formulas with performance enabled.
dataImgFiltered[this.editorObject.id] = this.editorObject.settings.images_dataimg_filter;
this.editorObject.settings.images_dataimg_filter = (img) => {
if (img.hasAttribute('class') && img.getAttribute('class').indexOf(Configuration.get('imageClassName')) !== -1) {
return img.hasAttribute('internal-blob');
}
// If the client put an image data filter, run. Otherwise default behaviour (put blob).
if (typeof dataImgFiltered[this.editorObject.id] !== 'undefined') {
return dataImgFiltered[this.editorObject.id](img);
}
return true;
};
}
/**
* Fires the event ExecCommand and transform a MathML into an image formula.
* @param {string} mathml - MathML to generate the formula and can be caught with the event.
*/
updateFormula(mathml) {
if (typeof this.editorObject.fire !== 'undefined') {
this.editorObject.fire('ExecCommand', { command: 'updateFormula', value: mathml });
}
super.updateFormula(mathml);
}
}
/**
* Object containing all TinyMCE integration instances. One for each TinyMCE editor.
* @type {Object}
*/
export const instances = {};
/**
* TinyMCE integration current instance. The current instance
* is the instance related with the focused editor.
* @type {TinyMceIntegration}
*/
export const currentInstance = null;
/* Plugin integration */
(function () {
tinymce.create('tinymce.plugins.tiny_mce_wiris', {
init(editor) {
// Array with MathML valid alements.
const validMathML = [
'math[*]',
'maction[*]]',
'malignmark[*]',
'maligngroup[*]',
'menclose[*]',
'merror[*]',
'mfenced[*]',
'mfrac[*]',
'mglyph[*]',
'mi[*]',
'mlabeledtr[*]',
'mlongdiv[*]',
'mmultiscripts[*]',
'mn[*]',
'mo[*]',
'mover[*]',
'mpadded[*]',
'mphantom[*]',
'mprescripts[*]',
'none[*]',
'mroot[*]',
'mrow[*]',
'ms[*]',
'mscarries[*]',
'mscarry[*]',
'msgroup[*]',
'msline[*]',
'mspace[*]',
'msqrt[*]',
'msrow[*]',
'mstack[*]',
'mstyle[*]',
'msub[*]',
'msubsup[*]',
'msup[*]',
'mtable[*]',
'mtd[*]',
'mtext[*]',
'mtr[*]',
'munder[*]',
'munderover[*]',
'semantics[*]',
'annotation[*]',
];
editor.settings.extended_valid_elements += `,${validMathML.join()}`; // eslint-disable-line no-param-reassign
const callbackMethodArguments = {};
/**
* Integration model properties
* @type {Object}
* @property {Object} target - Integration DOM target.
* @property {String} configurationService - Configuration integration service.
* @property {String} version - Plugin version.
* @property {String} scriptName - Integration script name.
* @property {Object} environment - Integration environment properties.
* @property {String} editor - Editor name.
*/
const integrationModelProperties = {};
integrationModelProperties.serviceProviderProperties = {};
integrationModelProperties.serviceProviderProperties.URI = '/wirispluginengine/integration/';
integrationModelProperties.serviceProviderProperties.server = 'ruby';
integrationModelProperties.version = packageInfo.version;
integrationModelProperties.isMoodle = (!!((typeof M === 'object' && M !== null))); // eslint-disable-line no-undef
if (integrationModelProperties.isMoodle) {
// eslint-disable-next-line no-undef
integrationModelProperties.configurationService = M.cfg.wwwroot + '/filter/wiris/integration/configurationjs.php'; // eslint-disable-line prefer-template
}
if (typeof (editor.getParam('wiriscontextpath')) !== 'undefined') {
integrationModelProperties.configurationService = Util.concatenateUrl(editor.getParam('wiriscontextpath'), integrationModelProperties.configurationService);
`${editor.getParam('wiriscontextpath')}/${integrationModelProperties.configurationService}`; // eslint-disable-line no-unused-expressions
console.warn('Deprecated property wiriscontextpath. Use mathTypeParameters on instead.', editor.opts.wiriscontextpath);
}
// Overriding MathType integration parameters.
if (typeof (editor.getParam('mathTypeParameters')) !== 'undefined') {
integrationModelProperties.integrationParameters = editor.getParam('mathTypeParameters');
}
integrationModelProperties.scriptName = 'plugin.min.js';
integrationModelProperties.environment = {};
let editorVersion = '4';
if (tinymce.majorVersion === '5') {
editorVersion = '5';
}
integrationModelProperties.environment.editor = `TinyMCE ${editorVersion}.x`;
integrationModelProperties.environment.editorVersion = `${tinymce.majorVersion}.${tinymce.minorVersion}`;
integrationModelProperties.callbackMethodArguments = callbackMethodArguments;
integrationModelProperties.editorObject = editor;
integrationModelProperties.initParsed = false;
// We need to create the instance before TinyMce initialization in order to register commands.
// However, as TinyMCE is not initialized at this point the HTML target is not created.
// Here we create the target as null and onInit object the target is updated.
integrationModelProperties.target = null;
const isExternalPlugin = typeof (editor.getParam('external_plugins')) !== 'undefined' && 'tiny_mce_wiris' in editor.getParam('external_plugins');
integrationModelProperties.isExternal = isExternalPlugin;
integrationModelProperties.rtl = (editor.getParam('directionality') === 'rtl');
// GenericIntegration instance.
const tinyMceIntegrationInstance = new TinyMceIntegration(integrationModelProperties);
tinyMceIntegrationInstance.init();
WirisPlugin.instances[tinyMceIntegrationInstance.editorObject.id] = tinyMceIntegrationInstance;
WirisPlugin.currentInstance = tinyMceIntegrationInstance;
const onInit = function (editor) { // eslint-disable-line no-shadow
const integrationInstance = WirisPlugin.instances[tinyMceIntegrationInstance.editorObject.id];
if (!editor.inline) {
integrationInstance.setTarget(editor.getContentAreaContainer().firstChild);
} else {
integrationInstance.setTarget(editor.getElement());
}
integrationInstance.setEditorObject(editor);
integrationInstance.listeners.fire('onTargetReady', {});
if ('wiriseditorparameters' in editor.settings) {
Configuration.update('editorParameters', editor.settings.wiriseditorparameters);
}
// Prevent TinyMCE attributes insertion.
// TinyMCE insert attributes only when a new node is inserted.
// For this reason, the mutation observer only acts on addedNodes.
const mutationInstance = new MutationObserver(function (editor, mutations) { // eslint-disable-line no-shadow
Array.prototype.forEach.call(mutations, function (editor, mutation) { // eslint-disable-line no-shadow
Array.prototype.forEach.call(mutation.addedNodes, function (editor, node) { // eslint-disable-line no-shadow
if (node.nodeType === 1) {
// Act only in our own formulas.
Array.prototype.forEach.call(node.querySelectorAll(`.${WirisPlugin.Configuration.get('imageClassName')}`), ((editor, image) => { // eslint-disable-line no-shadow
// This only is executed due to init parse.
image.removeAttribute('data-mce-src');
image.removeAttribute('data-mce-style');
}).bind(this, editor));
}
}.bind(this, editor));
}.bind(this, editor));
}.bind(this, editor));
mutationInstance.observe(editor.getBody(), {
attributes: true,
childList: true,
characterData: true,
subtree: true,
});
const content = editor.getContent();
// We set content in html because other tiny plugins need data-mce
// and this is not possible with raw format.
editor.setContent(Parser.initParse(content, editor.getParam('language')), { format: 'html' });
// This clean undoQueue for prevent onChange and Dirty state.
editor.undoManager.clear();
// Init parsing OK. If a setContent method is called
// wrs_initParse is called again.
// Now if source code is edited the returned code is parsed.
// PLUGINS-1070: We set this variable out of condition to parse content after.
WirisPlugin.instances[editor.id].initParsed = true;
};
if ('onInit' in editor) {
editor.onInit.add(onInit);
} else {
editor.on('init', () => {
onInit(editor);
});
}
if ('onActivate' in editor) {
editor.onActivate.add((editor) => { // eslint-disable-line no-unused-vars, no-shadow
WirisPlugin.currentInstance = WirisPlugin.instances[tinymce.activeEditor.id];
});
} else {
editor.on('focus', (event) => { // eslint-disable-line no-unused-vars, no-shadow
WirisPlugin.currentInstance = WirisPlugin.instances[tinymce.activeEditor.id];
});
}
const onSave = function (editor, params) { // eslint-disable-line no-shadow
params.content = Parser.endParse(params.content, editor.getParam('language'));
};
if ('onSaveContent' in editor) {
editor.onSaveContent.add(onSave);
} else {
editor.on('saveContent', (params) => {
onSave(editor, params);
});
}
if ('onGetContent' in editor) {
editor.onGetContent.add(onSave);
} else {
editor.on('getContent', (params) => {
onSave(editor, params);
});
}
if ('onBeforeSetContent' in editor) {
editor.onBeforeSetContent.add((e, params) => {
if (WirisPlugin.instances[editor.id].initParsed) {
params.content = Parser.initParse(params.content, editor.getParam('language'));
}
});
} else {
editor.on('beforeSetContent', (params) => {
if (WirisPlugin.instances[editor.id].initParsed) {
params.content = Parser.initParse(params.content, editor.getParam('language'));
}
});
}
function openFormulaEditorFunction() {
const tinyMceIntegrationInstance = WirisPlugin.instances[editor.id]; // eslint-disable-line no-shadow
// Disable previous custom editors.
tinyMceIntegrationInstance.core.getCustomEditors().disable();
tinyMceIntegrationInstance.openNewFormulaEditor();
}
let commonEditor;
const mathTypeIcon = 'mathtypeicon';
const chemTypeIcon = 'chemtypeicon';
const mathTypeIconSvg = '<svg width="16" height="16" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"viewBox="0 0 300 261.7" style="enable-background:new 0 0 300 261.7;" xml:space="preserve"><g id=icon-wirisformula stroke="none" stroke-width="1" fill-rule="evenodd"><g><path class="st1" d="M90.2,257.7c-11.4,0-21.9-6.4-27-16.7l-60-119.9c-7.5-14.9-1.4-33.1,13.5-40.5c14.9-7.5,33.1-1.4,40.5,13.5l27.3,54.7L121.1,39c5.3-15.8,22.4-24.4,38.2-19.1c15.8,5.3,24.4,22.4,19.1,38.2l-59.6,179c-3.9,11.6-14.3,19.7-26.5,20.6C91.6,257.7,90.9,257.7,90.2,257.7"/></g></g><g><g><path class="st2" d="M300,32.8c0-16.4-13.4-29.7-29.9-29.7c-2.9,0-7.2,0.8-7.2,0.8c-37.9,9.1-71.3,14-112,14c-0.3,0-0.6,0-1,0c-16.5,0-29.9,13.3-29.9,29.7c0,16.4,13.4,29.7,29.9,29.7l0,0c45.3,0,83.1-5.3,125.3-15.3h0C289.3,59.5,300,47.4,300,32.8"/></g></g></svg>'; // eslint-disable-line max-len
const chemTypeIconSvg = '<svg width="16" height="16" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" width="16px" height="16px" viewBox="0 0 16 16" enable-background="new 0 0 16 16" xml:space="preserve"> <image id="image0" width="16" height="16" x="0" y="0"href="" /></svg>'; // eslint-disable-line max-len
if (tinymce.majorVersion === '5') {
commonEditor = editor.ui.registry;
commonEditor.addIcon(mathTypeIcon, mathTypeIconSvg);
commonEditor.addIcon(chemTypeIcon, chemTypeIconSvg);
// The next two blocks create menu items to give the possibility
// of add MathType in the menubar.
commonEditor.addMenuItem('tiny_mce_wiris_formulaEditor', {
text: 'MathType',
icon: mathTypeIcon,
onAction: openFormulaEditorFunction,
});
// Dynamic customEditors buttons.
const customEditors = WirisPlugin.instances[editor.id].getCore().getCustomEditors();
Object.keys(customEditors.editors).forEach((customEditor) => {
if (customEditors.editors[customEditor].confVariable) {
commonEditor.addMenuItem(`tiny_mce_wiris_formulaEditor${customEditors.editors[customEditor].name}`, {
text: customEditors.editors[customEditor].title,
icon: chemTypeIcon, // Parametrize when other custom editors are added.
onAction: () => {
customEditors.enable(customEditor);
WirisPlugin.instances[editor.id].openNewFormulaEditor();
},
});
}
});
} else {
commonEditor = editor;
commonEditor.addCommand('tiny_mce_wiris_openFormulaEditor', openFormulaEditorFunction);
}
// MathType button.
// Cmd Parameter is needed in TinyMCE4 and onAction parameter is needed in TinyMCE5.
// For more details see TinyMCE migration page: https://www.tiny.cloud/docs-preview/migration-from-4.x/
commonEditor.addButton('tiny_mce_wiris_formulaEditor', {
tooltip: '数学公式', // TinyMCE3
title: '数学公式',
cmd: 'tiny_mce_wiris_openFormulaEditor',
image: `${WirisPlugin.instances[editor.id].getIconsPath()}formula.png`,
onAction: openFormulaEditorFunction,
icon: mathTypeIcon,
});
// Dynamic customEditors buttons.
const customEditors = WirisPlugin.instances[editor.id].getCore().getCustomEditors();
for (const customEditor in customEditors.editors) {
if (customEditors.editors[customEditor].confVariable) {
const cmd = `tiny_mce_wiris_openFormulaEditor${customEditors.editors[customEditor].name}`;
// eslint-disable-next-line no-inner-declarations, no-loop-func
function commandFunction() {
customEditors.enable(customEditor);
// eslint-disable-next-line no-undef
WirisPlugin.instances[editor.id].openNewFormulaEditor();
}
editor.addCommand(cmd, commandFunction);
// Cmd Parameter is needed in TinyMCE4 and onAction parameter is needed in TinyMCE5.
// For more details see TinyMCE migration page: https://www.tiny.cloud/docs-preview/migration-from-4.x/
commonEditor.addButton(`tiny_mce_wiris_formulaEditor${customEditors.editors[customEditor].name}`, {
title: customEditors.editors[customEditor].tooltip, // TinyMCE3
tooltip: customEditors.editors[customEditor].tooltip,
onAction: commandFunction,
cmd,
image: WirisPlugin.instances[editor.id].getIconsPath() + customEditors.editors[customEditor].icon,
icon: chemTypeIcon, // At the moment only chemTypeIcon because of the provisional solution for TinyMCE5.
});
}
}
},
// All versions.
getInfo() {
return {
longname: 'tiny_mce_wiris',
author: 'Maths for More',
authorurl: 'http://www.wiris.com',
infourl: 'http://www.wiris.com',
version: packageInfo.version,
};
},
});
tinymce.PluginManager.add('tiny_mce_wiris', tinymce.plugins.tiny_mce_wiris);
}());