|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
( function( tinymce, setTimeout ) { |
|
if ( tinymce.Env.ie && tinymce.Env.ie < 9 ) { |
|
return; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function escapeRegExp( string ) { |
|
return string.replace( /[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, '\\$&' ); |
|
} |
|
|
|
tinymce.PluginManager.add( 'wptextpattern', function( editor ) { |
|
var VK = tinymce.util.VK; |
|
var settings = editor.settings.wptextpattern || {}; |
|
|
|
var spacePatterns = settings.space || [ |
|
{ regExp: /^[*-]\s/, cmd: 'InsertUnorderedList' }, |
|
{ regExp: /^1[.)]\s/, cmd: 'InsertOrderedList' } |
|
]; |
|
|
|
var enterPatterns = settings.enter || [ |
|
{ start: '##', format: 'h2' }, |
|
{ start: '###', format: 'h3' }, |
|
{ start: '####', format: 'h4' }, |
|
{ start: '#####', format: 'h5' }, |
|
{ start: '######', format: 'h6' }, |
|
{ start: '>', format: 'blockquote' }, |
|
{ regExp: /^(-){3,}$/, element: 'hr' } |
|
]; |
|
|
|
var inlinePatterns = settings.inline || [ |
|
{ delimiter: '`', format: 'code' } |
|
]; |
|
|
|
var canUndo; |
|
|
|
editor.on( 'selectionchange', function() { |
|
canUndo = null; |
|
} ); |
|
|
|
editor.on( 'keydown', function( event ) { |
|
if ( ( canUndo && event.keyCode === 27 ) || ( canUndo === 'space' && event.keyCode === VK.BACKSPACE ) ) { |
|
editor.undoManager.undo(); |
|
event.preventDefault(); |
|
event.stopImmediatePropagation(); |
|
} |
|
|
|
if ( VK.metaKeyPressed( event ) ) { |
|
return; |
|
} |
|
|
|
if ( event.keyCode === VK.ENTER ) { |
|
enter(); |
|
|
|
} else if ( event.keyCode === VK.SPACEBAR ) { |
|
setTimeout( space ); |
|
} else if ( event.keyCode > 47 && ! ( event.keyCode >= 91 && event.keyCode <= 93 ) ) { |
|
setTimeout( inline ); |
|
} |
|
}, true ); |
|
|
|
function inline() { |
|
var rng = editor.selection.getRng(); |
|
var node = rng.startContainer; |
|
var offset = rng.startOffset; |
|
var startOffset; |
|
var endOffset; |
|
var pattern; |
|
var format; |
|
var zero; |
|
|
|
|
|
if ( ! node || node.nodeType !== 3 || ! node.data.length || ! offset ) { |
|
return; |
|
} |
|
|
|
var string = node.data.slice( 0, offset ); |
|
var lastChar = node.data.charAt( offset - 1 ); |
|
|
|
tinymce.each( inlinePatterns, function( p ) { |
|
|
|
if ( lastChar !== p.delimiter.slice( -1 ) ) { |
|
return; |
|
} |
|
|
|
var escDelimiter = escapeRegExp( p.delimiter ); |
|
var delimiterFirstChar = p.delimiter.charAt( 0 ); |
|
var regExp = new RegExp( '(.*)' + escDelimiter + '.+' + escDelimiter + '$' ); |
|
var match = string.match( regExp ); |
|
|
|
if ( ! match ) { |
|
return; |
|
} |
|
|
|
startOffset = match[1].length; |
|
endOffset = offset - p.delimiter.length; |
|
|
|
var before = string.charAt( startOffset - 1 ); |
|
var after = string.charAt( startOffset + p.delimiter.length ); |
|
|
|
|
|
|
|
|
|
if ( startOffset && /\S/.test( before ) ) { |
|
if ( /\s/.test( after ) || before === delimiterFirstChar ) { |
|
return; |
|
} |
|
} |
|
|
|
|
|
if ( ( new RegExp( '^[\\s' + escapeRegExp( delimiterFirstChar ) + ']+$' ) ).test( string.slice( startOffset, endOffset ) ) ) { |
|
return; |
|
} |
|
|
|
pattern = p; |
|
|
|
return false; |
|
} ); |
|
|
|
if ( ! pattern ) { |
|
return; |
|
} |
|
|
|
format = editor.formatter.get( pattern.format ); |
|
|
|
if ( format && format[0].inline ) { |
|
editor.undoManager.add(); |
|
|
|
editor.undoManager.transact( function() { |
|
node.insertData( offset, '\uFEFF' ); |
|
|
|
node = node.splitText( startOffset ); |
|
zero = node.splitText( offset - startOffset ); |
|
|
|
node.deleteData( 0, pattern.delimiter.length ); |
|
node.deleteData( node.data.length - pattern.delimiter.length, pattern.delimiter.length ); |
|
|
|
editor.formatter.apply( pattern.format, {}, node ); |
|
|
|
editor.selection.setCursorLocation( zero, 1 ); |
|
} ); |
|
|
|
|
|
setTimeout( function() { |
|
canUndo = 'space'; |
|
|
|
editor.once( 'selectionchange', function() { |
|
var offset; |
|
|
|
if ( zero ) { |
|
offset = zero.data.indexOf( '\uFEFF' ); |
|
|
|
if ( offset !== -1 ) { |
|
zero.deleteData( offset, offset + 1 ); |
|
} |
|
} |
|
} ); |
|
} ); |
|
} |
|
} |
|
|
|
function firstTextNode( node ) { |
|
var parent = editor.dom.getParent( node, 'p' ), |
|
child; |
|
|
|
if ( ! parent ) { |
|
return; |
|
} |
|
|
|
while ( child = parent.firstChild ) { |
|
if ( child.nodeType !== 3 ) { |
|
parent = child; |
|
} else { |
|
break; |
|
} |
|
} |
|
|
|
if ( ! child ) { |
|
return; |
|
} |
|
|
|
if ( ! child.data ) { |
|
if ( child.nextSibling && child.nextSibling.nodeType === 3 ) { |
|
child = child.nextSibling; |
|
} else { |
|
child = null; |
|
} |
|
} |
|
|
|
return child; |
|
} |
|
|
|
function space() { |
|
var rng = editor.selection.getRng(), |
|
node = rng.startContainer, |
|
parent, |
|
text; |
|
|
|
if ( ! node || firstTextNode( node ) !== node ) { |
|
return; |
|
} |
|
|
|
parent = node.parentNode; |
|
text = node.data; |
|
|
|
tinymce.each( spacePatterns, function( pattern ) { |
|
var match = text.match( pattern.regExp ); |
|
|
|
if ( ! match || rng.startOffset !== match[0].length ) { |
|
return; |
|
} |
|
|
|
editor.undoManager.add(); |
|
|
|
editor.undoManager.transact( function() { |
|
node.deleteData( 0, match[0].length ); |
|
|
|
if ( ! parent.innerHTML ) { |
|
parent.appendChild( document.createElement( 'br' ) ); |
|
} |
|
|
|
editor.selection.setCursorLocation( parent ); |
|
editor.execCommand( pattern.cmd ); |
|
} ); |
|
|
|
|
|
setTimeout( function() { |
|
canUndo = 'space'; |
|
} ); |
|
|
|
return false; |
|
} ); |
|
} |
|
|
|
function enter() { |
|
var rng = editor.selection.getRng(), |
|
start = rng.startContainer, |
|
node = firstTextNode( start ), |
|
i = enterPatterns.length, |
|
text, pattern, parent; |
|
|
|
if ( ! node ) { |
|
return; |
|
} |
|
|
|
text = node.data; |
|
|
|
while ( i-- ) { |
|
if ( enterPatterns[ i ].start ) { |
|
if ( text.indexOf( enterPatterns[ i ].start ) === 0 ) { |
|
pattern = enterPatterns[ i ]; |
|
break; |
|
} |
|
} else if ( enterPatterns[ i ].regExp ) { |
|
if ( enterPatterns[ i ].regExp.test( text ) ) { |
|
pattern = enterPatterns[ i ]; |
|
break; |
|
} |
|
} |
|
} |
|
|
|
if ( ! pattern ) { |
|
return; |
|
} |
|
|
|
if ( node === start && tinymce.trim( text ) === pattern.start ) { |
|
return; |
|
} |
|
|
|
editor.once( 'keyup', function() { |
|
editor.undoManager.add(); |
|
|
|
editor.undoManager.transact( function() { |
|
if ( pattern.format ) { |
|
editor.formatter.apply( pattern.format, {}, node ); |
|
node.replaceData( 0, node.data.length, ltrim( node.data.slice( pattern.start.length ) ) ); |
|
} else if ( pattern.element ) { |
|
parent = node.parentNode && node.parentNode.parentNode; |
|
|
|
if ( parent ) { |
|
parent.replaceChild( document.createElement( pattern.element ), node.parentNode ); |
|
} |
|
} |
|
} ); |
|
|
|
|
|
setTimeout( function() { |
|
canUndo = 'enter'; |
|
} ); |
|
} ); |
|
} |
|
|
|
function ltrim( text ) { |
|
return text ? text.replace( /^\s+/, '' ) : ''; |
|
} |
|
} ); |
|
} )( window.tinymce, window.setTimeout ); |
|
|