gremmie@1: // ----------------------------------------------------------------------------
gremmie@1: // markItUp! Universal MarkUp Engine, JQuery plugin
gremmie@1: // v 1.1.5
gremmie@1: // Dual licensed under the MIT and GPL licenses.
gremmie@1: // ----------------------------------------------------------------------------
gremmie@1: // Copyright (C) 2007-2008 Jay Salvat
gremmie@1: // http://markitup.jaysalvat.com/
gremmie@1: // ----------------------------------------------------------------------------
gremmie@1: // Permission is hereby granted, free of charge, to any person obtaining a copy
gremmie@1: // of this software and associated documentation files (the "Software"), to deal
gremmie@1: // in the Software without restriction, including without limitation the rights
gremmie@1: // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
gremmie@1: // copies of the Software, and to permit persons to whom the Software is
gremmie@1: // furnished to do so, subject to the following conditions:
gremmie@1: //
gremmie@1: // The above copyright notice and this permission notice shall be included in
gremmie@1: // all copies or substantial portions of the Software.
gremmie@1: //
gremmie@1: // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
gremmie@1: // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
gremmie@1: // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
gremmie@1: // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
gremmie@1: // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
gremmie@1: // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
gremmie@1: // THE SOFTWARE.
gremmie@1: // ----------------------------------------------------------------------------
gremmie@1: (function($) {
gremmie@1: $.fn.markItUp = function(settings, extraSettings) {
gremmie@1: var options, ctrlKey, shiftKey, altKey;
gremmie@1: ctrlKey = shiftKey = altKey = false;
gremmie@1:
gremmie@1: options = { id: '',
gremmie@1: nameSpace: '',
gremmie@1: root: '',
gremmie@1: previewInWindow: '', // 'width=800, height=600, resizable=yes, scrollbars=yes'
gremmie@1: previewAutoRefresh: true,
gremmie@1: previewPosition: 'after',
gremmie@1: previewTemplatePath: '~/templates/preview.html',
gremmie@1: previewParserPath: '',
gremmie@1: previewParserVar: 'data',
gremmie@1: resizeHandle: true,
gremmie@1: beforeInsert: '',
gremmie@1: afterInsert: '',
gremmie@1: onEnter: {},
gremmie@1: onShiftEnter: {},
gremmie@1: onCtrlEnter: {},
gremmie@1: onTab: {},
gremmie@1: markupSet: [ { /* set */ } ]
gremmie@1: };
gremmie@1: $.extend(options, settings, extraSettings);
gremmie@1:
gremmie@1: // compute markItUp! path
gremmie@1: if (!options.root) {
gremmie@1: $('script').each(function(a, tag) {
gremmie@1: miuScript = $(tag).get(0).src.match(/(.*)jquery\.markitup(\.pack)?\.js$/);
gremmie@1: if (miuScript !== null) {
gremmie@1: options.root = miuScript[1];
gremmie@1: }
gremmie@1: });
gremmie@1: }
gremmie@1:
gremmie@1: return this.each(function() {
gremmie@1: var $$, textarea, levels, scrollPosition, caretPosition, caretOffset,
gremmie@1: clicked, hash, header, footer, previewWindow, template, iFrame, abort;
gremmie@1: $$ = $(this);
gremmie@1: textarea = this;
gremmie@1: levels = [];
gremmie@1: abort = false;
gremmie@1: scrollPosition = caretPosition = 0;
gremmie@1: caretOffset = -1;
gremmie@1:
gremmie@1: options.previewParserPath = localize(options.previewParserPath);
gremmie@1: options.previewTemplatePath = localize(options.previewTemplatePath);
gremmie@1:
gremmie@1: // apply the computed path to ~/
gremmie@1: function localize(data, inText) {
gremmie@1: if (inText) {
gremmie@1: return data.replace(/("|')~\//g, "$1"+options.root);
gremmie@1: }
gremmie@1: return data.replace(/^~\//, options.root);
gremmie@1: }
gremmie@1:
gremmie@1: // init and build editor
gremmie@1: function init() {
gremmie@1: id = ''; nameSpace = '';
gremmie@1: if (options.id) {
gremmie@1: id = 'id="'+options.id+'"';
gremmie@1: } else if ($$.attr("id")) {
gremmie@1: id = 'id="markItUp'+($$.attr("id").substr(0, 1).toUpperCase())+($$.attr("id").substr(1))+'"';
gremmie@1:
gremmie@1: }
gremmie@1: if (options.nameSpace) {
gremmie@1: nameSpace = 'class="'+options.nameSpace+'"';
gremmie@1: }
gremmie@1: $$.wrap('
');
gremmie@1: $$.wrap('');
gremmie@1: $$.wrap('');
gremmie@1: $$.addClass("markItUpEditor");
gremmie@1:
gremmie@1: // add the header before the textarea
gremmie@1: header = $('').insertBefore($$);
gremmie@1: $(dropMenus(options.markupSet)).appendTo(header);
gremmie@1:
gremmie@1: // add the footer after the textarea
gremmie@1: footer = $('').insertAfter($$);
gremmie@1:
gremmie@1: // add the resize handle after textarea
gremmie@1: if (options.resizeHandle === true && $.browser.safari !== true) {
gremmie@1: resizeHandle = $('')
gremmie@1: .insertAfter($$)
gremmie@1: .bind("mousedown", function(e) {
gremmie@1: var h = $$.height(), y = e.clientY, mouseMove, mouseUp;
gremmie@1: mouseMove = function(e) {
gremmie@1: $$.css("height", Math.max(20, e.clientY+h-y)+"px");
gremmie@1: return false;
gremmie@1: };
gremmie@1: mouseUp = function(e) {
gremmie@1: $("html").unbind("mousemove", mouseMove).unbind("mouseup", mouseUp);
gremmie@1: return false;
gremmie@1: };
gremmie@1: $("html").bind("mousemove", mouseMove).bind("mouseup", mouseUp);
gremmie@1: });
gremmie@1: footer.append(resizeHandle);
gremmie@1: }
gremmie@1:
gremmie@1: // listen key events
gremmie@1: $$.keydown(keyPressed).keyup(keyPressed);
gremmie@1:
gremmie@1: // bind an event to catch external calls
gremmie@1: $$.bind("insertion", function(e, settings) {
gremmie@1: if (settings.target !== false) {
gremmie@1: get();
gremmie@1: }
gremmie@1: if (textarea === $.markItUp.focused) {
gremmie@1: markup(settings);
gremmie@1: }
gremmie@1: });
gremmie@1:
gremmie@1: // remember the last focus
gremmie@1: $$.focus(function() {
gremmie@1: $.markItUp.focused = this;
gremmie@1: });
gremmie@1: }
gremmie@1:
gremmie@1: // recursively build header with dropMenus from markupset
gremmie@1: function dropMenus(markupSet) {
gremmie@1: var ul = $(''), i = 0;
gremmie@1: $('li:hover > ul', ul).css('display', 'block');
gremmie@1: $.each(markupSet, function() {
gremmie@1: var button = this, t = '', title, li, j;
gremmie@1: title = (button.key) ? (button.name||'')+' [Ctrl+'+button.key+']' : (button.name||'');
gremmie@1: key = (button.key) ? 'accesskey="'+button.key+'"' : '';
gremmie@1: if (button.separator) {
gremmie@1: li = $(''+(button.separator||'')+'').appendTo(ul);
gremmie@1: } else {
gremmie@1: i++;
gremmie@1: for (j = levels.length -1; j >= 0; j--) {
gremmie@1: t += levels[j]+"-";
gremmie@1: }
gremmie@1: li = $(''+(button.name||'')+'')
gremmie@1: .bind("contextmenu", function() { // prevent contextmenu on mac and allow ctrl+click
gremmie@1: return false;
gremmie@1: }).click(function() {
gremmie@1: return false;
gremmie@1: }).mouseup(function() {
gremmie@1: if (button.call) {
gremmie@1: eval(button.call)();
gremmie@1: }
gremmie@1: markup(button);
gremmie@1: return false;
gremmie@1: }).hover(function() {
gremmie@1: $('> ul', this).show();
gremmie@1: $(document).one('click', function() { // close dropmenu if click outside
gremmie@1: $('ul ul', header).hide();
gremmie@1: }
gremmie@1: );
gremmie@1: }, function() {
gremmie@1: $('> ul', this).hide();
gremmie@1: }
gremmie@1: ).appendTo(ul);
gremmie@1: if (button.dropMenu) {
gremmie@1: levels.push(i);
gremmie@1: $(li).addClass('markItUpDropMenu').append(dropMenus(button.dropMenu));
gremmie@1: }
gremmie@1: }
gremmie@1: });
gremmie@1: levels.pop();
gremmie@1: return ul;
gremmie@1: }
gremmie@1:
gremmie@1: // markItUp! markups
gremmie@1: function magicMarkups(string) {
gremmie@1: if (string) {
gremmie@1: string = string.toString();
gremmie@1: string = string.replace(/\(\!\(([\s\S]*?)\)\!\)/g,
gremmie@1: function(x, a) {
gremmie@1: var b = a.split('|!|');
gremmie@1: if (altKey === true) {
gremmie@1: return (b[1] !== undefined) ? b[1] : b[0];
gremmie@1: } else {
gremmie@1: return (b[1] === undefined) ? "" : b[0];
gremmie@1: }
gremmie@1: }
gremmie@1: );
gremmie@1: // [![prompt]!], [![prompt:!:value]!]
gremmie@1: string = string.replace(/\[\!\[([\s\S]*?)\]\!\]/g,
gremmie@1: function(x, a) {
gremmie@1: var b = a.split(':!:');
gremmie@1: if (abort === true) {
gremmie@1: return false;
gremmie@1: }
gremmie@1: value = prompt(b[0], (b[1]) ? b[1] : '');
gremmie@1: if (value === null) {
gremmie@1: abort = true;
gremmie@1: }
gremmie@1: return value;
gremmie@1: }
gremmie@1: );
gremmie@1: return string;
gremmie@1: }
gremmie@1: return "";
gremmie@1: }
gremmie@1:
gremmie@1: // prepare action
gremmie@1: function prepare(action) {
gremmie@1: if ($.isFunction(action)) {
gremmie@1: action = action(hash);
gremmie@1: }
gremmie@1: return magicMarkups(action);
gremmie@1: }
gremmie@1:
gremmie@1: // build block to insert
gremmie@1: function build(string) {
gremmie@1: openWith = prepare(clicked.openWith);
gremmie@1: placeHolder = prepare(clicked.placeHolder);
gremmie@1: replaceWith = prepare(clicked.replaceWith);
gremmie@1: closeWith = prepare(clicked.closeWith);
gremmie@1: if (replaceWith !== "") {
gremmie@1: block = openWith + replaceWith + closeWith;
gremmie@1: } else if (selection === '' && placeHolder !== '') {
gremmie@1: block = openWith + placeHolder + closeWith;
gremmie@1: } else {
gremmie@1: block = openWith + (string||selection) + closeWith;
gremmie@1: }
gremmie@1: return { block:block,
gremmie@1: openWith:openWith,
gremmie@1: replaceWith:replaceWith,
gremmie@1: placeHolder:placeHolder,
gremmie@1: closeWith:closeWith
gremmie@1: };
gremmie@1: }
gremmie@1:
gremmie@1: // define markup to insert
gremmie@1: function markup(button) {
gremmie@1: var len, j, n, i;
gremmie@1: hash = clicked = button;
gremmie@1: get();
gremmie@1:
gremmie@1: $.extend(hash, { line:"",
gremmie@1: root:options.root,
gremmie@1: textarea:textarea,
gremmie@1: selection:(selection||''),
gremmie@1: caretPosition:caretPosition,
gremmie@1: ctrlKey:ctrlKey,
gremmie@1: shiftKey:shiftKey,
gremmie@1: altKey:altKey
gremmie@1: }
gremmie@1: );
gremmie@1: // callbacks before insertion
gremmie@1: prepare(options.beforeInsert);
gremmie@1: prepare(clicked.beforeInsert);
gremmie@1: if (ctrlKey === true && shiftKey === true) {
gremmie@1: prepare(clicked.beforeMultiInsert);
gremmie@1: }
gremmie@1: $.extend(hash, { line:1 });
gremmie@1:
gremmie@1: if (ctrlKey === true && shiftKey === true) {
gremmie@1: lines = selection.split(/\r?\n/);
gremmie@1: for (j = 0, n = lines.length, i = 0; i < n; i++) {
gremmie@1: if ($.trim(lines[i]) !== '') {
gremmie@1: $.extend(hash, { line:++j, selection:lines[i] } );
gremmie@1: lines[i] = build(lines[i]).block;
gremmie@1: } else {
gremmie@1: lines[i] = "";
gremmie@1: }
gremmie@1: }
gremmie@1: string = { block:lines.join('\n')};
gremmie@1: start = caretPosition;
gremmie@1: len = string.block.length + (($.browser.opera) ? n : 0);
gremmie@1: } else if (ctrlKey === true) {
gremmie@1: string = build(selection);
gremmie@1: start = caretPosition + string.openWith.length;
gremmie@1: len = string.block.length - string.openWith.length - string.closeWith.length;
gremmie@1: len -= fixIeBug(string.block);
gremmie@1: } else if (shiftKey === true) {
gremmie@1: string = build(selection);
gremmie@1: start = caretPosition;
gremmie@1: len = string.block.length;
gremmie@1: len -= fixIeBug(string.block);
gremmie@1: } else {
gremmie@1: string = build(selection);
gremmie@1: start = caretPosition + string.block.length ;
gremmie@1: len = 0;
gremmie@1: start -= fixIeBug(string.block);
gremmie@1: }
gremmie@1: if ((selection === '' && string.replaceWith === '')) {
gremmie@1: caretOffset += fixOperaBug(string.block);
gremmie@1:
gremmie@1: start = caretPosition + string.openWith.length;
gremmie@1: len = string.block.length - string.openWith.length - string.closeWith.length;
gremmie@1:
gremmie@1: caretOffset = $$.val().substring(caretPosition, $$.val().length).length;
gremmie@1: caretOffset -= fixOperaBug($$.val().substring(0, caretPosition));
gremmie@1: }
gremmie@1: $.extend(hash, { caretPosition:caretPosition, scrollPosition:scrollPosition } );
gremmie@1:
gremmie@1: if (string.block !== selection && abort === false) {
gremmie@1: insert(string.block);
gremmie@1: set(start, len);
gremmie@1: } else {
gremmie@1: caretOffset = -1;
gremmie@1: }
gremmie@1: get();
gremmie@1:
gremmie@1: $.extend(hash, { line:'', selection:selection });
gremmie@1:
gremmie@1: // callbacks after insertion
gremmie@1: if (ctrlKey === true && shiftKey === true) {
gremmie@1: prepare(clicked.afterMultiInsert);
gremmie@1: }
gremmie@1: prepare(clicked.afterInsert);
gremmie@1: prepare(options.afterInsert);
gremmie@1:
gremmie@1: // refresh preview if opened
gremmie@1: if (previewWindow && options.previewAutoRefresh) {
gremmie@1: refreshPreview();
gremmie@1: }
gremmie@1:
gremmie@1: // reinit keyevent
gremmie@1: shiftKey = altKey = ctrlKey = abort = false;
gremmie@1: }
gremmie@1:
gremmie@1: // Substract linefeed in Opera
gremmie@1: function fixOperaBug(string) {
gremmie@1: if ($.browser.opera) {
gremmie@1: return string.length - string.replace(/\n*/g, '').length;
gremmie@1: }
gremmie@1: return 0;
gremmie@1: }
gremmie@1: // Substract linefeed in IE
gremmie@1: function fixIeBug(string) {
gremmie@1: if ($.browser.msie) {
gremmie@1: return string.length - string.replace(/\r*/g, '').length;
gremmie@1: }
gremmie@1: return 0;
gremmie@1: }
gremmie@1:
gremmie@1: // add markup
gremmie@1: function insert(block) {
gremmie@1: if (document.selection) {
gremmie@1: var newSelection = document.selection.createRange();
gremmie@1: newSelection.text = block;
gremmie@1: } else {
gremmie@1: $$.val($$.val().substring(0, caretPosition) + block + $$.val().substring(caretPosition + selection.length, $$.val().length));
gremmie@1: }
gremmie@1: }
gremmie@1:
gremmie@1: // set a selection
gremmie@1: function set(start, len) {
gremmie@1: if (textarea.createTextRange){
gremmie@1: // quick fix to make it work on Opera 9.5
gremmie@1: if ($.browser.opera && $.browser.version >= 9.5 && len == 0) {
gremmie@1: return false;
gremmie@1: }
gremmie@1: range = textarea.createTextRange();
gremmie@1: range.collapse(true);
gremmie@1: range.moveStart('character', start);
gremmie@1: range.moveEnd('character', len);
gremmie@1: range.select();
gremmie@1: } else if (textarea.setSelectionRange ){
gremmie@1: textarea.setSelectionRange(start, start + len);
gremmie@1: }
gremmie@1: textarea.scrollTop = scrollPosition;
gremmie@1: textarea.focus();
gremmie@1: }
gremmie@1:
gremmie@1: // get the selection
gremmie@1: function get() {
gremmie@1: textarea.focus();
gremmie@1:
gremmie@1: scrollPosition = textarea.scrollTop;
gremmie@1: if (document.selection) {
gremmie@1: selection = document.selection.createRange().text;
gremmie@1: if ($.browser.msie) { // ie
gremmie@1: var range = document.selection.createRange(), rangeCopy = range.duplicate();
gremmie@1: rangeCopy.moveToElementText(textarea);
gremmie@1: caretPosition = -1;
gremmie@1: while(rangeCopy.inRange(range)) { // fix most of the ie bugs with linefeeds...
gremmie@1: rangeCopy.moveStart('character');
gremmie@1: caretPosition ++;
gremmie@1: }
gremmie@1: } else { // opera
gremmie@1: caretPosition = textarea.selectionStart;
gremmie@1: }
gremmie@1: } else { // gecko
gremmie@1: caretPosition = textarea.selectionStart;
gremmie@1: selection = $$.val().substring(caretPosition, textarea.selectionEnd);
gremmie@1: }
gremmie@1: return selection;
gremmie@1: }
gremmie@1:
gremmie@1: // open preview window
gremmie@1: function preview() {
gremmie@1: if (!previewWindow || previewWindow.closed) {
gremmie@1: if (options.previewInWindow) {
gremmie@1: previewWindow = window.open('', 'preview', options.previewInWindow);
gremmie@1: } else {
gremmie@1: iFrame = $('');
gremmie@1: if (options.previewPosition == 'after') {
gremmie@1: iFrame.insertAfter(footer);
gremmie@1: } else {
gremmie@1: iFrame.insertBefore(header);
gremmie@1: }
gremmie@1: previewWindow = iFrame[iFrame.length-1].contentWindow || frame[iFrame.length-1];
gremmie@1: }
gremmie@1: } else if (altKey === true) {
gremmie@1: if (iFrame) {
gremmie@1: iFrame.remove();
gremmie@1: }
gremmie@1: previewWindow.close();
gremmie@1: previewWindow = iFrame = false;
gremmie@1: }
gremmie@1: if (!options.previewAutoRefresh) {
gremmie@1: refreshPreview();
gremmie@1: }
gremmie@1: }
gremmie@1:
gremmie@1: // refresh Preview window
gremmie@1: function refreshPreview() {
gremmie@1: if (previewWindow.document) {
gremmie@1: try {
gremmie@1: sp = previewWindow.document.documentElement.scrollTop
gremmie@1: } catch(e) {
gremmie@1: sp = 0;
gremmie@1: }
gremmie@1: previewWindow.document.open();
gremmie@1: previewWindow.document.write(renderPreview());
gremmie@1: previewWindow.document.close();
gremmie@1: previewWindow.document.documentElement.scrollTop = sp;
gremmie@1: }
gremmie@1: if (options.previewInWindow) {
gremmie@1: previewWindow.focus();
gremmie@1: }
gremmie@1: }
gremmie@1:
gremmie@1: function renderPreview() {
gremmie@1: if (options.previewParserPath !== '') {
gremmie@1: $.ajax( {
gremmie@1: type: 'POST',
gremmie@1: async: false,
gremmie@1: url: options.previewParserPath,
gremmie@1: data: options.previewParserVar+'='+encodeURIComponent($$.val()),
gremmie@1: success: function(data) {
gremmie@1: phtml = localize(data, 1);
gremmie@1: }
gremmie@1: } );
gremmie@1: } else {
gremmie@1: if (!template) {
gremmie@1: $.ajax( {
gremmie@1: async: false,
gremmie@1: url: options.previewTemplatePath,
gremmie@1: success: function(data) {
gremmie@1: template = localize(data, 1);
gremmie@1: }
gremmie@1: } );
gremmie@1: }
gremmie@1: phtml = template.replace(//g, $$.val());
gremmie@1: }
gremmie@1: return phtml;
gremmie@1: }
gremmie@1:
gremmie@1: // set keys pressed
gremmie@1: function keyPressed(e) {
gremmie@1: shiftKey = e.shiftKey;
gremmie@1: altKey = e.altKey;
gremmie@1: ctrlKey = (!(e.altKey && e.ctrlKey)) ? e.ctrlKey : false;
gremmie@1:
gremmie@1: if (e.type === 'keydown') {
gremmie@1: if (ctrlKey === true) {
gremmie@1: li = $("a[accesskey="+String.fromCharCode(e.keyCode)+"]", header).parent('li');
gremmie@1: if (li.length !== 0) {
gremmie@1: ctrlKey = false;
gremmie@1: li.triggerHandler('mouseup');
gremmie@1: return false;
gremmie@1: }
gremmie@1: }
gremmie@1: if (e.keyCode === 13 || e.keyCode === 10) { // Enter key
gremmie@1: if (ctrlKey === true) { // Enter + Ctrl
gremmie@1: ctrlKey = false;
gremmie@1: markup(options.onCtrlEnter);
gremmie@1: return options.onCtrlEnter.keepDefault;
gremmie@1: } else if (shiftKey === true) { // Enter + Shift
gremmie@1: shiftKey = false;
gremmie@1: markup(options.onShiftEnter);
gremmie@1: return options.onShiftEnter.keepDefault;
gremmie@1: } else { // only Enter
gremmie@1: markup(options.onEnter);
gremmie@1: return options.onEnter.keepDefault;
gremmie@1: }
gremmie@1: }
gremmie@1: if (e.keyCode === 9) { // Tab key
gremmie@1: if (shiftKey == true || ctrlKey == true || altKey == true) { // Thx Dr Floob.
gremmie@1: return false;
gremmie@1: }
gremmie@1: if (caretOffset !== -1) {
gremmie@1: get();
gremmie@1: caretOffset = $$.val().length - caretOffset;
gremmie@1: set(caretOffset, 0);
gremmie@1: caretOffset = -1;
gremmie@1: return false;
gremmie@1: } else {
gremmie@1: markup(options.onTab);
gremmie@1: return options.onTab.keepDefault;
gremmie@1: }
gremmie@1: }
gremmie@1: }
gremmie@1: }
gremmie@1:
gremmie@1: init();
gremmie@1: });
gremmie@1: };
gremmie@1:
gremmie@1: $.fn.markItUpRemove = function() {
gremmie@1: return this.each(function() {
gremmie@1: $$ = $(this).unbind().removeClass('markItUpEditor');
gremmie@1: $$.parent('div').parent('div.markItUp').parent('div').replaceWith($$);
gremmie@1: }
gremmie@1: );
gremmie@1: };
gremmie@1:
gremmie@1: $.markItUp = function(settings) {
gremmie@1: var options = { target:false };
gremmie@1: $.extend(options, settings);
gremmie@1: if (options.target) {
gremmie@1: return $(options.target).each(function() {
gremmie@1: $(this).focus();
gremmie@1: $(this).trigger('insertion', [options]);
gremmie@1: });
gremmie@1: } else {
gremmie@1: $('textarea').trigger('insertion', [options]);
gremmie@1: }
gremmie@1: };
gremmie@1: })(jQuery);