123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472 |
- /**
- * @license Copyright (c) 2003-2017, CKSource - Frederico Knabben. All rights reserved.
- * For licensing, see LICENSE.md or http://ckeditor.com/license
- */
- /**
- * @fileOverview The [Mathematical Formulas](http://ckeditor.com/addon/mathjax) plugin that allows you to create and modify mathematical equations written in TeX directly in CKEditor..
- */
- 'use strict';
- ( function() {
- CKEDITOR.plugins.add( 'mathjax', {
- lang: 'af,ar,az,bg,ca,cs,cy,da,de,de-ch,el,en,en-gb,eo,es,eu,fa,fi,fr,gl,he,hr,hu,id,it,ja,km,ko,ku,lt,nb,nl,no,oc,pl,pt,pt-br,ro,ru,sk,sl,sq,sv,tr,tt,ug,uk,vi,zh,zh-cn', // %REMOVE_LINE_CORE%
- requires: 'widget,dialog',
- icons: 'mathjax',
- hidpi: true, // %REMOVE_LINE_CORE%
- init: function( editor ) {
- var cls = editor.config.mathJaxClass || 'math-tex';
- if ( !editor.config.mathJaxLib ) {
- CKEDITOR.error( 'mathjax-no-config' );
- }
- editor.widgets.add( 'mathjax', {
- inline: true,
- dialog: 'mathjax',
- button: editor.lang.mathjax.button,
- mask: true,
- allowedContent: 'span(!' + cls + ')',
- // Allow style classes only on spans having mathjax class.
- styleToAllowedContentRules: function( style ) {
- var classes = style.getClassesArray();
- if ( !classes )
- return null;
- classes.push( '!' + cls );
- return 'span(' + classes.join( ',' ) + ')';
- },
- pathName: editor.lang.mathjax.pathName,
- template: '<span class="' + cls + '" style="display:inline-block" data-cke-survive=1></span>',
- parts: {
- span: 'span'
- },
- defaults: {
- math: '\\(x = {-b \\pm \\sqrt{b^2-4ac} \\over 2a}\\)'
- },
- init: function() {
- var iframe = this.parts.span.getChild( 0 );
- // Check if span contains iframe and create it otherwise.
- if ( !iframe || iframe.type != CKEDITOR.NODE_ELEMENT || !iframe.is( 'iframe' ) ) {
- iframe = new CKEDITOR.dom.element( 'iframe' );
- iframe.setAttributes( {
- style: 'border:0;width:0;height:0',
- scrolling: 'no',
- frameborder: 0,
- allowTransparency: true,
- src: CKEDITOR.plugins.mathjax.fixSrc
- } );
- this.parts.span.append( iframe );
- }
- // Wait for ready because on some browsers iFrame will not
- // have document element until it is put into document.
- // This is a problem when you crate widget using dialog.
- this.once( 'ready', function() {
- // Src attribute must be recreated to fix custom domain error after undo
- // (see iFrame.removeAttribute( 'src' ) in frameWrapper.load).
- if ( CKEDITOR.env.ie )
- iframe.setAttribute( 'src', CKEDITOR.plugins.mathjax.fixSrc );
- this.frameWrapper = new CKEDITOR.plugins.mathjax.frameWrapper( iframe, editor );
- this.frameWrapper.setValue( this.data.math );
- } );
- },
- data: function() {
- if ( this.frameWrapper )
- this.frameWrapper.setValue( this.data.math );
- },
- upcast: function( el, data ) {
- if ( !( el.name == 'span' && el.hasClass( cls ) ) )
- return;
- if ( el.children.length > 1 || el.children[ 0 ].type != CKEDITOR.NODE_TEXT )
- return;
- data.math = CKEDITOR.tools.htmlDecode( el.children[ 0 ].value );
- // Add style display:inline-block to have proper height of widget wrapper and mask.
- var attrs = el.attributes;
- if ( attrs.style )
- attrs.style += ';display:inline-block';
- else
- attrs.style = 'display:inline-block';
- // Add attribute to prevent deleting empty span in data processing.
- attrs[ 'data-cke-survive' ] = 1;
- el.children[ 0 ].remove();
- return el;
- },
- downcast: function( el ) {
- el.children[ 0 ].replaceWith( new CKEDITOR.htmlParser.text( CKEDITOR.tools.htmlEncode( this.data.math ) ) );
- // Remove style display:inline-block.
- var attrs = el.attributes;
- attrs.style = attrs.style.replace( /display:\s?inline-block;?\s?/, '' );
- if ( attrs.style === '' )
- delete attrs.style;
- return el;
- }
- } );
- // Add dialog.
- CKEDITOR.dialog.add( 'mathjax', this.path + 'dialogs/mathjax.js' );
- // Add MathJax script to page preview.
- editor.on( 'contentPreview', function( evt ) {
- evt.data.dataValue = evt.data.dataValue.replace(
- /<\/head>/,
- '<script src="' + CKEDITOR.getUrl( editor.config.mathJaxLib ) + '"><\/script><\/head>'
- );
- } );
- editor.on( 'paste', function( evt ) {
- // Firefox does remove iFrame elements from pasted content so this event do the same on other browsers.
- // Also iFrame in paste content is reason of "Unspecified error" in IE9 (#10857).
- var regex = new RegExp( '<span[^>]*?' + cls + '.*?<\/span>', 'ig' );
- evt.data.dataValue = evt.data.dataValue.replace( regex, function( match ) {
- return match.replace( /(<iframe.*?\/iframe>)/i, '' );
- } );
- } );
- }
- } );
- /**
- * @private
- * @singleton
- * @class CKEDITOR.plugins.mathjax
- */
- CKEDITOR.plugins.mathjax = {};
- /**
- * A variable to fix problems with `iframe`. This variable is global
- * because it is used in both the widget and the dialog window.
- *
- * @private
- * @property {String} fixSrc
- */
- CKEDITOR.plugins.mathjax.fixSrc =
- // In Firefox src must exist and be different than about:blank to emit load event.
- CKEDITOR.env.gecko ? 'javascript:true' : // jshint ignore:line
- // Support for custom document.domain in IE.
- CKEDITOR.env.ie ? 'javascript:' + // jshint ignore:line
- 'void((function(){' + encodeURIComponent(
- 'document.open();' +
- '(' + CKEDITOR.tools.fixDomain + ')();' +
- 'document.close();'
- ) + '})())' :
- // In Chrome src must be undefined to emit load event.
- 'javascript:void(0)'; // jshint ignore:line
- /**
- * Loading indicator image generated by http://preloaders.net.
- *
- * @private
- * @property {String} loadingIcon
- */
- CKEDITOR.plugins.mathjax.loadingIcon = CKEDITOR.plugins.get( 'mathjax' ).path + 'images/loader.gif';
- /**
- * Computes predefined styles and copies them to another element.
- *
- * @private
- * @param {CKEDITOR.dom.element} from Copy source.
- * @param {CKEDITOR.dom.element} to Copy target.
- */
- CKEDITOR.plugins.mathjax.copyStyles = function( from, to ) {
- var stylesToCopy = [ 'color', 'font-family', 'font-style', 'font-weight', 'font-variant', 'font-size' ];
- for ( var i = 0; i < stylesToCopy.length; i++ ) {
- var key = stylesToCopy[ i ],
- val = from.getComputedStyle( key );
- if ( val )
- to.setStyle( key, val );
- }
- };
- /**
- * Trims MathJax value from '\(1+1=2\)' to '1+1=2'.
- *
- * @private
- * @param {String} value String to trim.
- * @returns {String} Trimed string.
- */
- CKEDITOR.plugins.mathjax.trim = function( value ) {
- var begin = value.indexOf( '\\(' ) + 2,
- end = value.lastIndexOf( '\\)' );
- return value.substring( begin, end );
- };
- /**
- * FrameWrapper is responsible for communication between the MathJax library
- * and the `iframe` element that is used for rendering mathematical formulas
- * inside the editor.
- * It lets you create visual mathematics by using the
- * {@link CKEDITOR.plugins.mathjax.frameWrapper#setValue setValue} method.
- *
- * @private
- * @class CKEDITOR.plugins.mathjax.frameWrapper
- * @constructor Creates a class instance.
- * @param {CKEDITOR.dom.element} iFrame The `iframe` element to be wrapped.
- * @param {CKEDITOR.editor} editor The editor instance.
- */
- if ( !( CKEDITOR.env.ie && CKEDITOR.env.version == 8 ) ) {
- CKEDITOR.plugins.mathjax.frameWrapper = function( iFrame, editor ) {
- var buffer, preview, value, newValue,
- doc = iFrame.getFrameDocument(),
- // Is MathJax loaded and ready to work.
- isInit = false,
- // Is MathJax parsing Tex.
- isRunning = false,
- // Function called when MathJax is loaded.
- loadedHandler = CKEDITOR.tools.addFunction( function() {
- preview = doc.getById( 'preview' );
- buffer = doc.getById( 'buffer' );
- isInit = true;
- if ( newValue )
- update();
- // Private! For test usage only.
- CKEDITOR.fire( 'mathJaxLoaded', iFrame );
- } ),
- // Function called when MathJax finish his job.
- updateDoneHandler = CKEDITOR.tools.addFunction( function() {
- CKEDITOR.plugins.mathjax.copyStyles( iFrame, preview );
- preview.setHtml( buffer.getHtml() );
- editor.fire( 'lockSnapshot' );
- iFrame.setStyles( {
- height: 0,
- width: 0
- } );
- // Set iFrame dimensions.
- var height = Math.max( doc.$.body.offsetHeight, doc.$.documentElement.offsetHeight ),
- width = Math.max( preview.$.offsetWidth, doc.$.body.scrollWidth );
- iFrame.setStyles( {
- height: height + 'px',
- width: width + 'px'
- } );
- editor.fire( 'unlockSnapshot' );
- // Private! For test usage only.
- CKEDITOR.fire( 'mathJaxUpdateDone', iFrame );
- // If value changed in the meantime update it again.
- if ( value != newValue )
- update();
- else
- isRunning = false;
- } );
- iFrame.on( 'load', load );
- load();
- function load() {
- doc = iFrame.getFrameDocument();
- if ( doc.getById( 'preview' ) )
- return;
- // Because of IE9 bug in a src attribute can not be javascript
- // when you undo (#10930). If you have iFrame with javascript in src
- // and call insertBefore on such element then IE9 will see crash.
- if ( CKEDITOR.env.ie )
- iFrame.removeAttribute( 'src' );
- doc.write( '<!DOCTYPE html>' +
- '<html>' +
- '<head>' +
- '<meta charset="utf-8">' +
- '<script type="text/x-mathjax-config">' +
- // MathJax configuration, disable messages.
- 'MathJax.Hub.Config( {' +
- 'showMathMenu: false,' +
- 'messageStyle: "none"' +
- '} );' +
- // Get main CKEDITOR form parent.
- 'function getCKE() {' +
- 'if ( typeof window.parent.CKEDITOR == \'object\' ) {' +
- 'return window.parent.CKEDITOR;' +
- '} else {' +
- 'return window.parent.parent.CKEDITOR;' +
- '}' +
- '}' +
- // Run MathJax.Hub with its actual parser and call callback function after that.
- // Because MathJax.Hub is asynchronous create MathJax.Hub.Queue to wait with callback.
- 'function update() {' +
- 'MathJax.Hub.Queue(' +
- '[ \'Typeset\', MathJax.Hub, this.buffer ],' +
- 'function() {' +
- 'getCKE().tools.callFunction( ' + updateDoneHandler + ' );' +
- '}' +
- ');' +
- '}' +
- // Run MathJax for the first time, when the script is loaded.
- // Callback function will be called then it's done.
- 'MathJax.Hub.Queue( function() {' +
- 'getCKE().tools.callFunction(' + loadedHandler + ');' +
- '} );' +
- '</script>' +
- // Load MathJax lib.
- '<script src="' + ( editor.config.mathJaxLib ) + '"></script>' +
- '</head>' +
- '<body style="padding:0;margin:0;background:transparent;overflow:hidden">' +
- '<span id="preview"></span>' +
- // Render everything here and after that copy it to the preview.
- '<span id="buffer" style="display:none"></span>' +
- '</body>' +
- '</html>' );
- }
- // Run MathJax parsing Tex.
- function update() {
- isRunning = true;
- value = newValue;
- editor.fire( 'lockSnapshot' );
- buffer.setHtml( value );
- // Set loading indicator.
- preview.setHtml( '<img src=' + CKEDITOR.plugins.mathjax.loadingIcon + ' alt=' + editor.lang.mathjax.loading + '>' );
- iFrame.setStyles( {
- height: '16px',
- width: '16px',
- display: 'inline',
- 'vertical-align': 'middle'
- } );
- editor.fire( 'unlockSnapshot' );
- // Run MathJax.
- doc.getWindow().$.update( value );
- }
- return {
- /**
- * Sets the TeX value to be displayed in the `iframe` element inside
- * the editor. This function will activate the MathJax
- * library which interprets TeX expressions and converts them into
- * their representation that is displayed in the editor.
- *
- * @param {String} value TeX string.
- */
- setValue: function( value ) {
- newValue = CKEDITOR.tools.htmlEncode( value );
- if ( isInit && !isRunning )
- update();
- }
- };
- };
- } else {
- // In IE8 MathJax does not work stable so instead of using standard
- // frame wrapper it is replaced by placeholder to show pure TeX in iframe.
- CKEDITOR.plugins.mathjax.frameWrapper = function( iFrame, editor ) {
- iFrame.getFrameDocument().write( '<!DOCTYPE html>' +
- '<html>' +
- '<head>' +
- '<meta charset="utf-8">' +
- '</head>' +
- '<body style="padding:0;margin:0;background:transparent;overflow:hidden">' +
- '<span style="white-space:nowrap;" id="tex"></span>' +
- '</body>' +
- '</html>' );
- return {
- setValue: function( value ) {
- var doc = iFrame.getFrameDocument(),
- tex = doc.getById( 'tex' );
- tex.setHtml( CKEDITOR.plugins.mathjax.trim( CKEDITOR.tools.htmlEncode( value ) ) );
- CKEDITOR.plugins.mathjax.copyStyles( iFrame, tex );
- editor.fire( 'lockSnapshot' );
- iFrame.setStyles( {
- width: Math.min( 250, tex.$.offsetWidth ) + 'px',
- height: doc.$.body.offsetHeight + 'px',
- display: 'inline',
- 'vertical-align': 'middle'
- } );
- editor.fire( 'unlockSnapshot' );
- }
- };
- };
- }
- } )();
- /**
- * Sets the path to the MathJax library. It can be both a local resource and a location different than the default CDN.
- *
- * Please note that this must be a full or absolute path.
- *
- * Read more in the [documentation](#!/guide/dev_mathjax)
- * and see the [SDK sample](http://sdk.ckeditor.com/samples/mathjax.html).
- *
- * config.mathJaxLib = '//cdn.mathjax.org/mathjax/2.6-latest/MathJax.js?config=TeX-AMS_HTML';
- *
- * **Note:** Since CKEditor 4.5 this option does not have a default value, so it must
- * be set in order to enable the MathJax plugin.
- *
- * @since 4.3
- * @cfg {String} mathJaxLib
- * @member CKEDITOR.config
- */
- /**
- * Sets the default class for `span` elements that will be
- * converted into [Mathematical Formulas](http://ckeditor.com/addon/mathjax)
- * widgets.
- *
- * If you set it to the following:
- *
- * config.mathJaxClass = 'my-math';
- *
- * The code below will be recognized as a Mathematical Formulas widget.
- *
- * <span class="my-math">\( \sqrt{4} = 2 \)</span>
- *
- * Read more in the [documentation](#!/guide/dev_mathjax)
- * and see the [SDK sample](http://sdk.ckeditor.com/samples/mathjax.html).
- *
- * @cfg {String} [mathJaxClass='math-tex']
- * @member CKEDITOR.config
- */
|