Tatoeba Markdown Editor

Adds MarkEdit to Tatoeba and parses markdown comments as HTML

当前为 2015-04-28 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name		   Tatoeba Markdown Editor
// @namespace	   Jakob V. <[email protected]>
// @description	   Adds MarkEdit to Tatoeba and parses markdown comments as HTML
// @include        http://*.tatoeba.org/*
// @include        https://*.tatoeba.org/*
// @match          http://*.tatoeba.org/*
// @match          https://*.tatoeba.org/*
// @require		   http://code.jquery.com/jquery-1.7.js
// @require		   http://code.jquery.com/ui/1.8.20/jquery-ui.min.js
// @require		   http://code.jquery.com/ui/1.11.4/jquery-ui.js
// @require        https://greasyfork.org/scripts/9431-tatoeba-jquery-ui-css-for-require/code/Tatoeba%20jQuery%20UI%20CSS%20for%[email protected]?version=48895
// @require        https://greasyfork.org/scripts/9546-js-pandoc-for-require/code/js-pandoc%20for%[email protected]?version=48894
// @require		   https://greasyfork.org/scripts/9434-markdown-editor-rehost-for-userscript-integration/code/Markdown%20Editor%20rehost%20for%20userscript%20integration.user.js
// @require https://greasyfork.org/scripts/9547-diff-match-patch-for-require/code/diff_match_patch-for-require.js?version=48896
// @version 0.0.1.20150428080829
// ==/UserScript==
// 
// http://userscripts.org/scripts/source/134357.user.js // Tatoeba jQuery UI CSS for @require
// http://userscripts.org/scripts/source/135396.user.js // Showdown.js rehost for userscript integration
// http://userscripts.org/scripts/source/135400.user.js // Markdown Editor rehost for userscript integration
// The MIT License
// 
// Original WMD and Showdwon code copyright (c) 2007 John Fraser
// Toolbar images (c) 2009 Dana Robinson
// MarkEdit jQuery rewrite and modified images (c) 2009 Titus Stone
// 
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// 
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
// 
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
//
$(document).ready(main);

function main() {
	if(window.location.href.split('/')[4] == 'sentences' && window.location.href.split('/')[5] == 'show' ){
		infoIcon = $('.infoIcon:first');
		if(infoIcon.length==0){
			infoIcon = $('<a class="infoIcon"><img width="16" height="16" alt="" src=""></a>');
			$('.sentence.mainSentence').prepend(infoIcon);
		}
	
		infoIcon.click(function(event){
			thetextarea = $('.markedittextarea');
			state = thetextarea.markeditGetState();
	
			// insert sentence citation (the selection will be the proposed change;
			// if it's empty the whole sentence will be copied)
			state.beforeSelect = state.beforeSelect + '"' + $('.mainSentence .sentenceContent .text').text() + '" ? "';
			state.select = (state.select == "" ? $('.mainSentence .sentenceContent .text').text() : state.select);
			state.afterSelect = '"' + state.afterSelect;
	
			thetextarea.markeditSetState(state);
			event.preventDefault();
		});
	}
	/*$('select#SentenceFrom, select#SentenceTo, select#randomLangChoiceInBrowse').selectmenu({style:'dropdown', menuWidth: 250, format: function(text, opt){ value = opt.attr('value'); if(value=='und'){value = 'unknown';} return '<img width="30" height="20" alt="'+value+'" title="'+text+'" class="languageFlag" src="http://flags.tatoeba.org/img/flags/'+value+'.png"> '+text; }});
	$('input[type="submit"]').button();
	$('input:text, input:password')
  .button()
  .css({
		  'font' : 'inherit',
		 'color' : 'inherit',
	'text-align' : 'left',
	   'outline' : 'none',
		'cursor' : 'text'
  });

	$('#randomLink a, #prevSentence a, #nextSentence a').button();
	$('.languageSelect + ul li').css({'background':'transparent', 'padding':'0', 'margin':'0'});
	$('fieldset.submit').prepend($('<label>_</label>'));
	*/
	
	// debugging
	console = unsafeWindow.console;
	GM_log = console.log;
	
	// define default variables
	default_markdownusers = {
		'jakov': true,
		'Esperantostern': true,
		'sacredceltic': true
	};
	default_markdownusers = JSON.stringify(default_markdownusers);

	// get saved variables
	markdownusers = GM_getValue('markdownusers');
	markdownusers = markdownusers || default_markdownusers;
	markdownusers = JSON.parse(markdownusers);
	console.log('markdownusers: ' + JSON.stringify(markdownusers));

	// add CSS
	var css = resourcevar + "\
	.markedit { clear: both; display: inline-block; } \
	.markedit textarea { width: 100%; } \
	.markedit-toolbar { padding: 0.3em; margin: 0; clear: both; height: 22px; border-radius: 8px 8px 0px 0px; -moz-border-radius: 8px 8px 0px 0px;} \
	.markedit-toolbar .toolbar-group { margin-right: 0.5em; padding: 0 0 0 5px; float: left; } \
	.markedit-toolbar .toggle-group {  } \
	.markedit-toolbar .toggle-group button { font-size: 0.85em; font-weight: bold; padding: 0.15em 0.5em; } \
	.markedit-toolbar .toggle-group button:first-child { -moz-border-radius: 11px 0 0 11px; border-radius: 11px 0 0 11px; } \
	.markedit-toolbar .toggle-group button:last-child { -moz-border-radius: 0 11px 11px 0; border-radius: 0 11px 11px 0; } \
	.markedit-toolbar button { height: 22px; outline: 0; cursor: pointer; } \
	.markedit-toolbar button.icon { width: 22px; background-repeat: no-repeat; margin: 0 5px 0 0; } \
	.markedit .light-bg button.icon { background-image: url(images/wmd-buttons.png); } \
	.markedit .dark-bg button.icon { background-image: url(images/wmd-buttons-dark.png); } \
	.markedit-toolbar button.bold { background-position: 0px 0px; } \
	.markedit-toolbar button.italic { background-position: -20px 0px; } \
	.markedit-toolbar button.link { background-position: -40px 1px; } \
	.markedit-toolbar button.quote { background-position: -60px 0px; } \
	.markedit-toolbar button.code { background-position: -80px 1px; } \
	.markedit-toolbar button.image { background-position: -100px 1px; } \
	.markedit-toolbar button.numberlist { background-position: -120px 0px; } \
	.markedit-toolbar button.bulletlist { background-position: -140px 0px; } \
	.markedit-toolbar button.heading { background-position: -160px 0px; } \
	.markedit-toolbar button.line { background-position: -180px 0px; } \
	.markedit-toolbar button.undo { background-position: -200px 0px; } \
	.markedit-toolbar button.redo { background-position: -220px 0px; } \
	.markedit-toolbar button.help { background-position: -240px 0px; } \
	.markedit-dialog { font-size: 0.75em; } \
	.markedit-dialog input { width: 100%; } \
	.markedit-preview { padding: 15px; } \
	\
	/* Selectmenu */ \
	.ui-selectmenu { display: block; display: inline-block; position: relative; height: 2.2em; vertical-align: middle; text-decoration: none; overflow: hidden; zoom: 1; } \
	.ui-selectmenu-icon { position:absolute; right:6px; margin-top:-8px; top: 50%; } \
	.ui-selectmenu-menu { padding:0; margin:0; position:absolute; top: 0; display: none; z-index: 1005;} /* z-index: 1005 to make selectmenu work with dialog */ \
	.ui-selectmenu-menu	 ul { padding:0; margin:0; list-style:none; position: relative; overflow: auto; overflow-y: auto ; overflow-x: hidden; -webkit-overflow-scrolling: touch;}	\
	.ui-selectmenu-open { display: block; } \
	.ui-selectmenu-menu-popup { margin-top: -1px; } \
	.ui-selectmenu-menu li { padding:0; margin:0; display: block; border-top: 1px dotted transparent; border-bottom: 1px dotted transparent; border-right-width: 0 !important; border-left-width: 0 !important; font-weight: normal !important; } \
	.ui-selectmenu-menu li a,.ui-selectmenu-status { line-height: 1.4em; display: block; padding: .405em 2.1em .405em 1em; outline:none; text-decoration:none; } \
	.ui-selectmenu-menu li.ui-state-disabled a, .ui-state-disabled { cursor: default; } \
	.ui-selectmenu-menu li.ui-selectmenu-hasIcon a, \
	.ui-selectmenu-hasIcon .ui-selectmenu-status { padding-left: 20px; position: relative; margin-left: 5px; } \
	.ui-selectmenu-menu li .ui-icon, .ui-selectmenu-status .ui-icon { position: absolute; top: 1em; margin-top: -8px; left: 0; } \
	.ui-selectmenu-status { line-height: 1.4em; } \
	/*.ui-selectmenu-menu li span,.ui-selectmenu-status span { display:block; margin-bottom: .2em; } */\
	.ui-selectmenu-menu li .ui-selectmenu-item-header { font-weight: bold; } \
	.ui-selectmenu-menu li .ui-selectmenu-item-footer { opacity: .8; } \
	/* for optgroups */ \
	.ui-selectmenu-menu .ui-selectmenu-group { font-size: 1em; } \
	.ui-selectmenu-menu .ui-selectmenu-group .ui-selectmenu-group-label { line-height: 1.4em; display:block; padding: .6em .5em 0; font-weight: bold; } \
	.ui-selectmenu-menu .ui-selectmenu-group ul { margin: 0; padding: 0; } \
	/* IE6 workaround (dotted transparent borders) */ \
	* html .ui-selectmenu-menu li { border-color: pink; filter:chroma(color=pink); width:100%; } \
	* html .ui-selectmenu-menu li a { position: relative } \
	/* IE7 workaround (opacity disabled) */ \
	*+html .ui-state-disabled, *+html .ui-state-disabled a { color: silver; } \
	\
	/* selectmenu Tatoeba */ \
	.ui-selectmenu-status { line-height: 0.7em; } \
	.ui-selectmenu { height: 1.5em;} \
	.ui-selectmenu a { width: auto !important; } \
	.ui-button { padding: 0.1em 0.2em !important; } \
	.ui-button-text { padding: 0.1em 0.2em !important; } \
	.ui-selectmenu-status .languageFlag { float: none; margin: 0; vertical-align: text-bottom; } \
	.ui-selectmenu { height: auto !important; width: auto !important; } \
	.ui-selectmenu-menu li a, .ui-selectmenu-status { padding: 0.3em 1.6em 0.3em 0.8em; } \
	/*.search_bar a { color: #257D0C !important; } \
	.search_bar .select, .search_bar .input { width: auto !important; } \
	.search_bar fieldset { float: left; } */ \
	\
	.markedit .light-bg button.icon { background-image: url(''); } .markedit .dark-bg button.icon { background-image: url(''); } \
	\
	textarea, .unparsedmarkdown { font-family: monospace; } \
	\
	pre.unparsedmarkdown { white-space: pre-wrap; } \
	.unparsedmarkdown br { display: none; } \
	.profileDescription .unparsedmarkdown { font-size: 1.3em; } \
	.privateMessage div.body { font-size: 1.1em; } \
	.privateMessage pre.body { font-size: 1.2em; } \
	.sticky .unparsedmarkdown { border: 1px solid grey; } \
	.parsedmarkdown { padding: 15px; } \
	.parsedmarkdown blockquote, .markedit-preview blockquote { border-left: 0.3em solid #BBBBBB; margin:0.5em 0;	padding: 0 1.1em; } \
	.parsedmarkdown > :first-child { margin-top: 0 !important; } \
	.parsedmarkdown > :last-child { margin-bottom: 0 !important; } \
	.parsedmarkdown li { margin: 0.2em 0 !important; } \
	.parsedmarkdown p { margin: 0.2em 0 !important; } \
	#WallContent { /* max-width: 508px;*/ } \
	.parsedmarkdown ul, .parsedmarkdown ol	{ padding-left: 3em; } \
	.parsedmarkdown ul { list-style: disc; } \
	\
	.parsedmarkdown ul, .parsedmarkdown ol	{ padding-left: 3em; } \
	.parsedmarkdown ul { list-style: disc; } \
	\
	#sendMessageForm .markedit textarea, .replyFormDiv .markedit textarea, .markedit textarea { min-width: 388px; margin: 3px 0; } \
	#SentenceCommentSaveForm textarea, #sendMessageForm textarea { width: 508px; } \
	.replyFormDiv textarea { width: 508px; width:100%; } \
	#WallSaveForm textarea { width: 508px; } \
	#PrivateMessageContent { width: 560px; } \
	#PrivateMessageSendForm .input.text { clear:both; display: inline-block; width: 100%; } \
	#PrivateMessageSendForm .input.text label { width:auto; } \
	#PrivateMessageSendForm .input.text input { float: right; width: 455px; } \
	#PrivateMessageSendForm { /* display: table; */ } \
	label:empty { display: none; } \
	 #SentenceCommentSaveForm textarea { width: auto; } \
	.markedit-preview { margin-bottom: 3px; } \
	\
	a[href^='/tags/show_sentences_with_tag/'] { background: none repeat scroll 0 0 #FFE684; border-bottom: 1px solid #DDDDDD; border-radius: 5px 5px 5px 5px; border-right: 1px solid #DDDDDD; padding: 2px 8px; color: #333333 !important; font-size: 1.2em;} \
	\
	a[href^='/user/profile/'] { } \
	a[href^='/sentences/show/'] { } \
	\
	a[href^='/tags/show_sentences_with_tag/']:before { content:'#'; } \
	a[href^='/user/profile/']:before { content:'@'; } \
	a[href^='/sentences/show/']:before { content:'?'; } \
	\
	.comp { padding:2px 8px; line-height:25px; background-color: #FBFFC9; border-bottom: 1px solid #DDDDDD; border-right: 1px solid #DDDDDD; border-radius:3px; } \
	.comp del, .patch del {color:red; background-color: #FFE6E6;} \
	.comp ins, .patch ins {color:green; background-color: #E6FFE6;} \
	.comp .before ins, .patch .before ins {display:none;} \
	.comp .after del, .patch .after del {display:none;} \
	\
	.autocorrect .ui-selectmenu { display: block; } \
	.autocorrect { font-size: 0.8em; margin-bottom: 3px; } \
	.autocorrect .patch { width: auto; width: -moz-available; display: inline-table !important; font-family: monospace; white-space:pre-wrap; } \
	.autocorrect .patch > span { display: table-cell !important; text-align:left;} \
	.autocorrect .patch .gets { font-size: x-large !important; } \
	.dropdown { border: 1px solid red; height: 100%; margin-top: -1px; position: absolute; right: -1px; top: 0; width: 26px; } \
	\
	a { -moz-transition: text-shadow 0.5s ease 0s; text-shadow: none;} \
	a:hover { text-shadow: 0 0 2px #89C140; } \
	\
	.parsedmarkdown table, .markedit-preview table { width: auto; } \
	.parsedmarkdown th, .parsedmarkdown td, .markedit-preview th, .markedit-preview td { border: 1px solid black; } \
	a:hover .markdownmark.solid path { fill: #FB6B10 !important; } \
	a:hover .markdownmark.normal path { fill: #FB6B10 !important; } \
	a:hover .markdownmark.normal rect { stroke: #FB6B10 !important; } \
	";
	GM_addStyle(css);


	// prepare wall replies for live preview (work in progress)
	$('.replyLink').click(function () {
		$('.replyFormDiv').find('.markedit').find('.markedit-toolbar, .ac_select, .markedit-preview').remove();
		$('.replyFormDiv').find('textarea').markedit({
		'preview': 'below',
		'toolbar': {
			'backgroundMode': 'light',
			'layout': 'bold italic | link quote code image | numberlist bulletlist heading line | cite-sentence',
			'buttons': [{
				// prepare the link toggle button
				// this should toggle between endnote and inline link
				// doesn'T work at all yet
				'id': 'linktoggle',
				'css': 'icon ui-state-default ui-corner-all',
				//ui-icon ui-icon-transferthick-e-w
				'tip': 'click to toggle between endnote style and inline style',
				'click': function () {
					//markedit = $(this).parentsUntil('.markedit').parent();
					//markedit.markeditSetLinkOrImage('http://example.com');
				},
				'mouseover': function () {},
				'mouseout': function () {}
			}, {
				'id': 'cite-sentence',
				'css': 'icon ui-state-default ui-corner-all',
				//ui-icon ui-icon-transferthick-e-w
				'tip': 'cite the main sentence',
				'click': function () {
					//find the textarea
					thetextarea = $('.markedittextarea');
					state = thetextarea.markeditGetState();

					// insert sentence citation (the selection will be the proposed change;
					// if it's empty the whole sentence will be copied)
					state.beforeSelect = state.beforeSelect + '"' + $('.mainSentence .sentenceContent .text').text() + '" ? "';
					state.select = (state.select == "" ? $('.mainSentence .sentenceContent .text').text() : state.select);
					state.afterSelect = '"' + state.afterSelect;

					thetextarea.markeditSetState(state);
				},
				'mouseover': function () {},
				'mouseout': function () {}
			}],
		}
		});
	});

	// diff_match_patch
	// the function that outputs the pretty html for the diffs
	diff_match_patch.prototype.diff_prettyHtml = function (diffs) {
		var html = [];
		var i = 0;
		for (var x = 0; x < diffs.length; x++) {
			var op = diffs[x][0]; // Operation (insert, delete, equal)
			var data = diffs[x][1]; // Text of change.
			var text = data.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\n/g, '&para;<BR>');
			switch (op) {
			case DIFF_INSERT:
				html[x] = '<ins>' + text + '</ins>';
				break;
			case DIFF_DELETE:
				html[x] = '<del>' + text + '</del>';
				break;
			case DIFF_EQUAL:
				html[x] = '<span>' + text + '</span>';
				break;
			}
			if (op !== DIFF_DELETE) {
				i += data.length;
			}
		}
		return html.join('');
	};

	// http://stackoverflow.com/questions/8584098/how-to-change-an-element-type-using-jquery
	$.fn.changeElementType = function(newType) {
		var attrs = {};

		$.each(this[0].attributes, function(idx, attr) {
			attrs[attr.nodeName] = attr.nodeValue;
		});
 
		var newelement = $("<" + newType + "/>", attrs).append($(this).contents());
		this.replaceWith(newelement);
		return newelement;
	};

http://stackoverflow.com/questions/202605/repeat-string-javascript
String.prototype.repeat = function(count) {
	if (count < 1) return '';
	var result = '', pattern = this.valueOf();
	while (count > 0) {
		if (count & 1) result += pattern;
		count >>= 1, pattern += pattern;
	}
	return result;
};

function linum2int(input) { //jakob
	input = input.replace(/[^A-Za-z]/, '');
	output = 0;
	for (i = 0; i < input.length; i++) {
		output = output * 26 + parseInt(input.substr(i, 1), 26 + 10) - 9;
	}
	console.log('linum', output);
	return output;
}

function int2linum(input) { //jakob
	// There's a quicker function that does the same on stackoverflow, but i wrote this one myself and im not sure about the license of the other one
	// http://stackoverflow.com/questions/8603480/how-to-create-a-function-that-converts-a-number-to-a-bijective-hexavigesimal/11506042#11506042
	var zeros = 0;
	var next = input;
	var generation = 0;
	while (next >= 27) {
		next = (next - 1) / 26 - (next - 1) % 26 / 26;
		zeros += next * Math.pow(27, generation);
		generation++;
	}
	output = (input + zeros).toString(27).replace(/./g, function ($0) {
		return '_abcdefghijklmnopqrstuvwxyz'.charAt(parseInt($0, 27));
	});
	return output;
}

function roman2int(input) { //jakob

	romans = {
		'm': 1000,
		'd': 500,
		'c': 100,
		'l': 50,
		'x': 10,
		'v': 5,
		'i': 1
	};
	input = input.replace(/[^A-Za-z]/, '').toLowerCase();
	output = 0;
	highest = false;
	for (i = input.length - 1; i >= 0; i--) {
		num = romans[input.substr(i, 1)] || 0;
		highest = (num > highest ? num : highest);
		output = (num < highest ? (output - num) : (output + num));
	}
	return output;
}

function int2roman(number) {
	// http://www.blackwasp.co.uk/NumberToRoman_2.aspx
	result = "";
	values = [1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1];
	numerals = ["m", "cm", "d", "cd", "c", "xc", "l", "xl", "x", "ix", "v", "iv", "i"];

	for (i = 0; i < 13; i++) {
		while (number >= values[i]) {
			number -= values[i];
			result += numerals[i];
		}
	}
	return result;
}

function tatoebaundo(text) {
		  //console.log(text);

		// undo the link shortening and fix its bugs
		// replace shortened link within inline markdown links or images
		// text = text.replace(/\(<a href="(.*?)">(.*?)<\/a>\)/gi, '\($1\)');
		
// repair markdown forced links <http://example.com>
		text = text.replace(/&lt;<a href="(.*?)%3E">(.*?)&gt;<\/a>;/gim, function($0, $1, $2){ return '<'+unescape($1)+'>';});
		
		// repair link detection errors
		text = text.replace(/<a href="(.*?&amp;)">(.*?)<\/a>;/gim, function($0, $1, $2){ return '<'+unescape($1)+'>';});

// replace shortened link with its URL
	   text = text.replace(/<a href="(.*?)">(.*?)<\/a>/gi, function($0, $1, $2){ return '<'+unescape($1)+'>';});
	   
		// remove forced linebreaks (<br>)
		text = text.replace(/(<br\/*>)+/gim, "");

		// replace shortened link within reference markdown links or images
		text = text.replace(/\[(.*)\]\: <a href="(.*)">(.*)<\/a>/gi, '[$1]: $2');

	
		return text;
	}

	function tatoedown(text) {


		// mimik Tatoeba's HTML denial by displaying angle brackets as they are
		// this will be later reenabled further down, but in the end we maybe don't even want html to be parsed!!!
		text = text.replace(/</gim, "&lt;").replace(/>/gim, "&gt;");

		// reenable html (Tatoeba doesn't allow HTML, so it displays the angle brackets as &lt; and &gt;)
		// however in the end we maybe don't even want html to be parsed!!!
		text = text.replace(/&lt;/gim, "<").replace(/&gt;/gim, ">");

		// "[blah](http://example.com/) asdf asdf asdf asdf http://blah.com deng asdf asdf asd".replace(/[^\(\<]((mailto\:|(news|(ht|f)tp(s?))\:\/\/){1}\S+)[^\)\>]/, 'xxx')
		// this is the linkification pattern taken from tatoeba's php (views > helpers > clickable_links.php)
	//	text = text.replace(/((ht|f)tps?:\/\/([\w\.]+\.)?[\w-]+(\.[a-zA-Z]{2,4})?[^\s\r\n\(\)"\'\!<]+)/g, '$1');
		
		//convert links to sentences (like nº642253)
		text = text.replace(/()(?:\[(?:n°|?)([0-9]+)\]|(?:n°|?)([0-9]+))/ig, '$1[$2$3](/sentences/show/$2$3)');
		//text = text.replace(/(\s)\[n°([0-9]+)\]/ig, '$1[n°$2](/sentences/show/$2 )');
		// convert simple #hashtags to links to the respective tag		 
		text = text.replace(/(\s)(?:\#([@0-9A-Za-z\u00A1-\uFFFF]+)|\[\#(.+)\])/ig, '$1[$2$3](/tags/show_sentences_with_tag/$2$3)');
		// convert @username to a link to the respective user profile
		text = text.replace(/(\s)(?:\@([A-Za-z0-9_]+)|\[\@([A-Za-z0-9_]+)\])/ig, '$1[$2$3](/user/profile/$2$3)');

		// convert [reference]-style links to "[reference][]" to activate [reference] (not only [reference][]) js-markdown-extra
		// i found out that this isnt even intended in markdown, but showdown seems to tolerate it
		// i think it disables the useage of [square brackets] in other intentions, which is bad
		// therefore i will not bypass this
		// and i think the code below isnt working well, but i'll let it here if needed later
		//text = text.replace(/\[(.+)\](?=[^\[\(])/ig, '[$1][$1]');
		//text = text.replace(/\[([^\]\^]+)\]\[([^\]\^]+)\]|\[([^\]\^]+)\](?=[^\[\(]\:)/ig, '[$1$3][$2]');

		//console.log(text);

		/*
		// convert "1) " lists to "1. " lists; this is the natural behaviour of some users
		text = text.replace(/^(\s*)(\d+)\) /img, '$1$2. ');
		*/
		
		// convert wiki-style headers "== header ==" with markdown style headers
		text = text.replace(/^\s*= (.*?) =\s*$/gim, '# $1 #').replace(/^\s*== (.*?) ==\s*$/gim, '## $1 ##').replace(/^\s*=== (.*) ===\s*$/gim, '### $1 ###').replace(/^\s*==== (.*?) ====\s*$/gim, '#### $1 ####').replace(/^\s*====== (.*?) ======\s*$/gim, '###### $1 ######').replace(/^\s*======= (.*?) =======\s*$/gim, '####### $1 #######');

		// prepare for easier matching of text-change patterns
		// e.g. “this” => “that” or «this» => «that» both gets {this}=>{that}
		text = text.replace(/([„\"{«»“`´'‘’"'])[ ]?(.*?)[ ]?([“\"}»«”`´'‘’"'])\s*(?:-+&gt;|=+&gt;|-+>|=+>|?|?|?|?|?|?|?)\s*\1(.*?)\3/ig, "{$2}=>{$4}");

		// match unified change proposals like {this}=>{that}
		if (text.match(/{(.*?)}=>{(.*?)}/gim) != null) {
			// each of them gets its own diff
			$.each(text.match(/{(.*?)}=>{(.*?)}/gim), function (index, value) {
				// reference is the old one, compareto is the new one
				reference = value.replace(/{(.*?)}=>{(.*?)}/im, "$1");
				compareto = value.replace(/{(.*?)}=>{(.*?)}/im, "$2");
				// initialize the diff_match_patch
				var dmp = new diff_match_patch();
				dmp.Diff_Timeout = 1;
				// compare them
				var d = dmp.diff_main(reference, compareto);
				// this makes it readable
				dmp.diff_cleanupSemantic(d);
				// when clicking on such a proposal, copy the text to the sentence if editable by you
				noins = (dmp.diff_prettyHtml(d).replace(/<ins>.*?<\/ins>/g,'')==dmp.diff_prettyHtml(d) ? ' noins':'');
				nodel = (dmp.diff_prettyHtml(d).replace(/<del>.*?<\/del>/g,'')==dmp.diff_prettyHtml(d) ? ' nodel':'');
				
				var ds = '<a onclick=\'$(".mainSentence .editableSentence").click();$(".mainSentence .editableSentence input").val($(".mainSentence .editableSentence input").val().replace("'+reference+'", "' + compareto + '"))\' class="comp'+noins+nodel+'"><span class="before">' + dmp.diff_prettyHtml(d).replace(/<ins>.*?<\/ins>/g,'') + '</span><span class="gets"> ? </span><span class="after">' + dmp.diff_prettyHtml(d).replace(/<del>.*?<\/del>/g,'') + '</span></a>';
				// now put it into the text
				text = text.replace(value, ds);
			});
		}
		
		return text;
	}

	//
	//	$.markeditGetHtml
	//
	$.fn.markeditGetHtml = function () {

		/*
		// Load Showdown if it's not loaded
		if (typeof(MarkEditShowDown) === 'undefined') {
			if (typeof(Attacklab) === 'undefined') {
				throw 'Showdown.js (Attacklab.showdown) must be loaded before MarkEdit.';
			}
			MarkEditShowDown = new Attacklab.showdown.converter();
		}*/

		// Render the preview
		var textarea = MarkEdit.getTextArea(this);
		var text = $(textarea).val();
		if (typeof (text) !== 'undefined') {

			var text = $(this).val();

			var html = Pandoc(tatoedown(tatoebaundo(text)));
			// html = html.replace(/\r/g, '');
			/* //this seems to be unneccessary for Tatoeba purposes
			// Convert newlines to <br/> inside a <p>
			var lineBreakInP = /(<p>(?:[\S\s](?!<\/p>))*)\n([\S\s]*?<\/p>)/g;
			var lineBreaksRemaining = lineBreakInP.exec(html);

			while (lineBreaksRemaining !== null) {
				html = html.replace(lineBreakInP, '$1<br />$2');
				lineBreaksRemaining = lineBreakInP.exec(html);
			}
			*/

			return html;
		} else {
			return '';
		}

	};

	//
	//	$.markeditBindAutoPreview
	//
	$.fn.markeditBindAutoPreview = function (element) {

		// Convert out of jQuery & find Textarea
		if (element.get) {
			element = element.get(0);
		}
		var textarea = MarkEdit.getTextArea(this);
		
		autocorrect = $('<div class="autocorrect"></div>');
		first = $('<div id="first"></div>');
		apply = $('<a id="apply"></a>').button().css({'display':'table-cell', 'width':'95%'});
		dropdown = $('<span id="dropdown"></span>').button({
					text: false,
					icons: {
						primary: "ui-icon-triangle-1-s"
					}
				}).css({'display':'table-cell'});
		dropdown.find('.ui-button-icon-primary').css({'position':'inherit'});
		dropdown.find('.ui-button-text').css({'display':'none'});
		first.append(apply).append(dropdown).css({'width':'100%', 'display':'table-row'});
		autocorrect.append(first).css({'display':'table'});
		$(textarea).after(autocorrect);
		
		default_markeditlang = 'eng';
		// get saved variables
		markeditlang = GM_getValue('markeditlang'); 
		markeditlang = markeditlang || default_markeditlang;
		console.log('markeditlang: ' + markeditlang);
		
		langchooser = $('fieldset.select:first select:first').clone().removeAttr('id').removeAttr('name').css({'width':'68px'});
		langchooser.attr('id', 'langchooser');
		langchooser.val(markeditlang);
		//langchooser.find('option:not([value="'+markeditlang+'"])').removeAttr('selected');
		//langchooser.find('option[value="'+markeditlang+'"]').attr('selected', true);
		
		langchooser.change(function(){
			markeditlang = $(this).find('option:selected').val();
			// save the new state
			GM_setValue('markeditlang', markeditlang);
			console.log('markeditlang: ' + markeditlang);
		});
		//langchooser.find('label:first').remove();
		//langchooser.find('select:first').removeAttr('id').removeAttr('name');
		//langchooser.find('select:first').change(function(){
		//	lang = $(this).find('option:selected').val();
		//});
		
		langchoosergroup = $('<div class="langchooser toolbar-group"></div>');
		langchoosergroup.append(langchooser);
		$('.toolbar-group:last').after(langchoosergroup);

		// Enable events for auto-preview
		$(textarea).bind('keyup', function (event) { //input paste change 

			// get the current state of the text
			thismarkeditor = $(this);

			state = thismarkeditor.markeditGetState();
			if (markeditlang != 'false') {
				/*
				RegExp.escape = function(text) {
				  if (!arguments.callee.sRE) {
					var specials = [
					  '/', '.', '*', '+', '?', '|',
					  '(', ')', '[', ']', '{', '}', '\\'
					];
					arguments.callee.sRE = new RegExp(
					  '(\\' + specials.join('|\\') + ')', 'g'
					);
				  }
				  return text.replace(arguments.callee.sRE, '\\$1');
				}
				
				text = 'at the end at the end';
				text.replace(new RegExp(RegExp.escape('at the end')+'$'), 'x');
				
				GESPERRTER TEXT IST BETONTER TEXT:
				
				"Here is an example of how to  e m p h a s i z e  words. You can add `*` for  * s t r o n g *  or `**` for	* * s t r o n g e m p h a s i s * *	 . <= Here you can see the downside (you can't use punktuation around it). It would however work with line breaks:	d o n a u d a m \np f s c h i f f  ".replace(/[ ][ ](\S(?:[ ]\n?\S\n?)*)[ ][ ]/g, function($1, $2){console.log($1+',', $2+',');return ' *'+$2.replace(/\s(\n?\S\n?)/g, function($x, $y){return $y;})+'* ';});
				
				
				TABELLENSPALTEN ANWÄHLEN
				
				$('.markedit-preview').find('table').prepend($('<colgroup/>').css({'background-color':'red'}));
				
				*/

				replacementarray = [
					['all', /(?:^|[ ][ ])(\S(?:[ ]\n?\S\n?)*)[ ]?[ ]?$/g, function($1, $2){console.log($1+',', $2+',');return ' *'+$2.replace(/\s(\n?\S\n?)/g, function($x, $y){return $y;})+'* ';}, 'gesperrter Text'],
					
					//['all', /([„\"{«»“`´'‘’"'])[ ]?(.*?)[ ]?([“\"}»«”`´'‘’"'])(\s*(?:-+&gt;|=+&gt;|-+>|=+>|?|?|?|?|?|?|?)\s*)\1?[ ]?$/, "$1$2$3$4$1$2$3", 'sentence changes'],
					['deu', /([„"{«»“`´'‘’"'])[ ]?(.*?)[ ]?([“"}»«”`´'‘’"'])(\s*(?:-+&gt;|=+&gt;|-+>|=+>|?|?|?|?|?|?|?)\s*)\1[ ]?(.*?)[ ]?\3$/, "„$2“ ? „$5“", 'sentence changes'],
					['eng', /([„"{«»“`´'‘’"'])[ ]?(.*?)[ ]?([“"}»«”`´'‘’"'])(\s*(?:-+&gt;|=+&gt;|-+>|=+>|?|?|?|?|?|?|?)\s*)\1[ ]?(.*?)[ ]?\3$/, "“$2” ? “$5”", 'sentence changes'],
					['fra', /([„"{«»“`´'‘’"'])[ ]?(.*?)[ ]?([“"}»«”`´'‘’"'])(\s*(?:-+&gt;|=+&gt;|-+>|=+>|?|?|?|?|?|?|?)\s*)\1[ ]?(.*?)[ ]?\3$/, "« $2 » ? « $5 »", 'sentence changes'],
					
					['deu', /("|„)(.*?)"$/, '„$2“', 'Anführungszeichen'],
					['deu', /('|‚)(.*?)'$/, '‚$2‘'],
					['deu', /(>>|»)(.*?)<<$/, '»$2«'],
					['deu', /(>|›)(.*?)<$/, '›$2‹'],

					['eng', /("|“|``)(.*?)("|'')$/, '“$2”'],
					['eng', /('|‘|`)(.*?)'$/, '‘$2’'],

					['fra', /~c$/, 'ç', 'c cedilla'],
					['fra', /\\?oe$/, 'œ', 'oe'],
					['fra', /\\?Oe|OE$/, 'Œ', 'OE'],
					['fra', /\\?ae$/, 'æ', 'ae'],
					['fra', /\\?Ae|AE$/, 'Æ', 'AE'],
					['fra', /"\s?([^"\n]*?)\s?"$|<<\s?([^"\n]*?)\s?>>$/, '« $1$2 »'],
					['fra', /'\s?([^'\n]*?)\s?'$|<\s?([^<\n]*?)\s?>$/, '‹ $1$2 ›'],
					['fra', / ([\;\:\?\!»›])$/, ' $1'], // automatically convert spaces before :;?!»›
					['fra', /([«‹]) $/, '$1 '], // automatically convert spaces after «‹
				
					['epo', /\^s|sx$/, 's', 's ^', 'auto'],
					['epo', /\^S|Sx|SX$/, 'S', 'S ^', 'auto'],
					['epo', /\^c|cx$/, 'c', 'c ^', 'auto'],
					['epo', /\^C|Cx|C$/, 'C', 'C ^', 'auto'],
					['epo', /\^g|gx$/, 'g', 'g ^', 'auto'],
					['epo', /\^G|Gx|GX$/, 'G', 'G ^', 'auto'],
					['epo', /\^h|hx$/, 'h', 'h ^', 'auto'],
					['epo', /\^H|Hx|HX$/, 'H', 'H ^', 'auto'],
					['epo', /\^j|jx$/, 'j', 'j ^', 'auto'],
					['epo', /\^J|Jx|JX$/, 'J', 'J ^', 'auto'],
					['epo', /\^u|ux$/, 'u', 'u ^', 'auto'],
					['epo', /\^U|Ux|UX$/, 'U', 'U ^', 'auto'],
					['epo', /sx$/, 's', 's'],
					['epo', /Sx|SX$/, 'S', 'S'],
					['epo', /cx$/, 'c', 'c'],
					['epo', /Cx|CX$/, 'C', 'C'],
					['epo', /gx$/, 'g', 'g'],
					['epo', /Gx|GX$/, 'G', 'G'],
					['epo', /hx$/, 'h', 'h'],
					['epo', /Hx|HX$/, 'H', 'H'],
					['epo', /jx$/, 'j', 'j'],
					['epo', /Jx|JX$/, 'J', 'J'],
					['epo', /ux$/, 'u', 'u'],
					['epo', /Ux|UX$/, 'U', 'U'],
					['epo', /(\S*)(?:\^s|sx)(\S*)$/, '$1s$2', 's ^'],
					['epo', /(\S*)(?:\^S|Sx|SX)(\S*)$/, '$1S$2', 'S ^'],
					['epo', /(\S*)(?:\^c|cx)(\S*)$/, '$1c$2', 'c ^'],
					['epo', /(\S*)(?:\^C|Cx|CX)(\S*)$/, '$1C$2', 'C ^'],
					['epo', /(\S*)(?:\^g|gx)(\S*)$/, '$1g$2', 'g ^'],
					['epo', /(\S*)(?:\^G|Gx|GX)(\S*)$/, '$1G$2', 'G ^'],
					['epo', /(\S*)(?:\^h|hx)(\S*)$/, '$1h$2', 'h ^'],
					['epo', /(\S*)(?:\^H|Hx|HX)(\S*)$/, '$1H$2', 'H ^'],
					['epo', /(\S*)(?:\^j|jx)(\S*)$/, '$1j$2', 'j ^'],
					['epo', /(\S*)(?:\^J|Jx|JX)(\S*)$/, '$1J$2', 'J ^'],
					['epo', /(\S*)(?:\^u|ux)(\S*)$/, '$1u$2', 'u ^'],
					['epo', /(\S*)(?:\^U|Ux|UX)(\S*)$/, '$1U$2', 'U ^'],
					['epo', /(\S*)(?:sx)(\S*)$/, '$1s$2', 's'],
					['epo', /(\S*)(?:Sx|SX)(\S*)$/, '$1S$2', 'S'],
					['epo', /(\S*)(?:cx)(\S*)$/, '$1c$2', 'c'],
					['epo', /(\S*)(?:Cx|CX)(\S*)$/, '$1C$2', 'C'],
					['epo', /(\S*)(?:gx)(\S*)$/, '$1g$2', 'g'],
					['epo', /(\S*)(?:Gx|GX)(\S*)$/, '$1G$2', 'G'],
					['epo', /(\S*)(?:hx)(\S*)$/, '$1h$2', 'h'],
					['epo', /(\S*)(?:Hx|HX)(\S*)$/, '$1H$2', 'H'],
					['epo', /(\S*)(?:jx)(\S*)$/, '$1j$2', 'j'],
					['epo', /(\S*)(?:Jx|JX)(\S*)$/, '$1J$2', 'J'],
					['epo', /(\S*)(?:ux)(\S*)$/, '$1u$2', 'u'],
					['epo', /(\S*)(?:Ux|UX)(\S*)$/, '$1U$2', 'U'],
			
					['all',
						/(?:^|\n)((?:[ ]|\t)*)(\d+|[a-zA-Z]+|[mdclxviMDCLXVI]+)([.\)][ ])(.*?)$/,
						function($0, $s, $1, $2, $3){ 
							if($1.match(/\d+/)){
								return $0+'\n'+$s+($1*1+1)+$2; 
							}
							else if($2==') ') {
								return $0+'\n'+$s+int2linum(linum2int($1)*1+1)+$2;
							} else {
								return $0+'\n'+$s+int2roman(roman2int($1)*1+1)+$2;
							}							
						},
						'numbered lists'
					],
					//['all', /((\d+)[.]\s.*?)$/, function($0, $1, $2){ return $1+"\n"+($2*1+1)+". ";}, 'simple numbered lists'],
					['all', /(^|\n)((?:[ ]|\t)*[*+->][ ])(.*?)$/, '$1$2$3\n$2', 'bulletlists or blockquotes'],
					['all', /((?:^|\n)[^\n]*?)\n(=+|-+)$/, function($0, $1, $2){ return $1+"\n"+$2.charAt(0).repeat($1.length-($1.charAt(0)=='\n'?1:0));}, 'Setext-style headers'],
					['all', /([_*`]{1,2})(?!\s)([^_*`]+)$/, '$1$2$1', 'markdown emphasis or codespan'],
					['all', /[\<]?((https?\:|ftps?\:|mailto\:|[-.\w]+\@)[^'">\s]*)$/, '<$1>', 'markdown auto url'],
					['all', /([!]?)\[([^\]]+)$/, '$1[$2]()', 'markdown (image)url complete'],
					['all', /<(a|abbr|address|area|article|aside|audio|b|base|bdi|bdo|blink|blockquote|body|br|button|canvas|caption|cite|code|col|colgroup|command|datalist|dd|del|details|dfn|div|dl|dt|em|embed|fieldset|figcaption|figure|footer|form|h[1-6]|head|header|hgroup|hr|html|i|iframe|img|input|ins|kbd|keygen|label|legend|li|link|map|mark|menu|meta|meter|nav|noscript|object|ol|optgroup|option|output|p|param|pre|progress|q|samp|script|section|select|small|source|span|strong|style|sub|summary|sup|table|tbody|td|textarea|tfoot|th|thead|time|title|tr|track|ul|var|video|wbr)([ ].*?)?(?:>(?:([^<>]*?)(<(\/((\1|\w+)(>)?)?)?)?)?)?$/, '<$1$2>$3</$1>', 'html tag complete'],
					['all',
						/(.*?[|].*?)\n(.*?[|].*?)$/,
						function ($0, $1, $2) {
							firstline = $1.split(/[|]/).reverse();
							secondline = $2.split(/[|]/).reverse();
						
							firstsmaller = firstline.length < secondline.length;
							while ((firstsmaller ? firstline : secondline).length < (!firstsmaller ? firstline : secondline).length) {
								(firstsmaller ? firstline : secondline).unshift('');
							}
							firstuebertrag = seconduebertrag = 0;
							firsttrenner = $1.match(/^[\|\-\:]*$/);
							secondtrenner = $2.match(/^[\|\-\:]*$/);
						
							$.each((firstline.length > secondline.length ? firstline : secondline), function (index, value) {
								firstline[index] = (typeof firstline[index] === "undefined" ? (firsttrenner ? '-' : ' ') : firstline[index].trim());
								secondline[index] = (typeof secondline[index] === "undefined" ? (firsttrenner ? '-' : ' ') : secondline[index].trim());
								letterdiff = (firstline[index].length + seconduebertrag) - (secondline[index].length + firstuebertrag);
						
								firstuebertrag = (letterdiff < 0 ? letterdiff * (-1) : 0);
								seconduebertrag = (letterdiff > 0 ? letterdiff * (+1) : 0);
								if (firstline[index].length > 0) {
									firstline[index] = firstline[index] + (firsttrenner ? '-' : ' ').repeat(firstuebertrag);
									firstuebertrag = 0;
								}
								if (secondline[index].length > 0) {
									secondline[index] = secondline[index] + (secondtrenner ? '-' : ' ').repeat(seconduebertrag);
									seconduebertrag = 0;
								}
							});
						
							return firstline.reverse().join('|') + '\n' + secondline.reverse().join('|');
						},
						'Table'
					],
					
					['all', /(\w+)'(\w+)$/, '$1’$2', 'apostrophe'],
					['all', /\.\.\.|\\ldots|\\mathellipsis$/, '…', 'ellipsis', 'auto'],
					['all', /–-|---|\\textemdash$/, '—', 'em-dash'],
					['all', /--|\\textendash$/, '–', 'en-dash'],
					['all', /\^a\^|\\textordfeminine$/, 'ª', 'feminine ordinal indicator'],
					['all', /\^o\^|\\textordmasculine$/, 'º', 'masculin ordinal indicator'],
					['all', /\^0\^$/, '°', 'superscript zero'],
					['all', /\^1\^$/, '¹', 'superscript one'],
					['all', /\^2\^$/, '²', 'superscript two'],
					['all', /\^3\^$/, '³', 'superscript three'],
					['all', /\^4\^$/, '4', 'superscript four'],
					['all', /\^5\^$/, '5', 'superscript five'],
					['all', /\^6\^$/, '6', 'superscript six'],
					['all', /\^7\^$/, '7', 'superscript seven'],
					['all', /\^8\^$/, '8', 'superscript eight'],
					['all', /\^9\^$/, '?', 'superscript nine'],
					['all', /\^-\^$/, '?', 'superscript minus'],
					['all', /\^=\^$/, '?', 'superscript equal'],
					['all', /\^\(\^$/, '?', 'superscript left parenthesis'],
					['all', /\^\)\^$/, '?', 'superscript right parenthesis'],
					['all', /\^+\^$/, '?', 'superscript plus sign'],
					['all', /\^n\^$/, 'n', 'superscript n'],

					['all', /\~0\~$/, '0', 'subscript zero'],
					['all', /\~1\~$/, '1', 'subscript one'],
					['all', /\~2\~$/, '2', 'subscript two'],
					['all', /\~3\~$/, '3', 'subscript three'],
					['all', /\~4\~$/, '4', 'subscript four'],
					['all', /\~5\~$/, '5', 'subscript five'],
					['all', /\~6\~$/, '6', 'subscript six'],
					['all', /\~7\~$/, '7', 'subscript seven'],
					['all', /\~8\~$/, '8', 'subscript eight'],
					['all', /\~9\~$/, '9', 'subscript nine'],
					['all', /\~i\~$/, '?', 'subscript i'],
					['all', /\~r\~$/, '?', 'subscript r'],
					['all', /\~u\~$/, '?', 'subscript u'],
					['all', /\~v\~$/, '?', 'subscript v'],
					['all', /\~a\~$/, '?', 'subscript a'],
					['all', /\~e\~$/, '?', 'subscript e'],
					['all', /\~o\~$/, '?', 'subscript o'],
					['all', /\~x\~$/, '?', 'subscript x'],
					['all', /\~schwa\~$/, '?', 'subscript schwa'],
					['all', /\~j\~$/, '?', 'subscript j'],
					['all', /\~beta\~$/, '?', 'subscript beta'],
					['all', /\~gamma\~$/, '?', 'subscript gamma'],
					['all', /\~rho\~$/, '?', 'subscript rho'],
					['all', /\~chi\~$/, '?', 'subscript chi'],
					['all', /\~\+\~$/, '?', 'subscript plus'],
					['all', /\~\-\~$/, '?', 'subscript minus'],
					['all', /\~\=\~$/, '?', 'subscript equals'],
					['all', /\~\(\~$/, '?', 'subscript left parenthesis'],
					['all', /\~\)\~$/, '?', 'subscript right parenthesis'],

					// http://theoval.cmp.uea.ac.uk/~nlct/latex/novices/symbols.html
					['all', /\~\!\~|\!`|\\textexclamdown$/, '¡', 'inverted exclamation mark'],
					['all', /\~\?\~|\?`|\\textquestiondown$/, '¿', 'inverted question mark'],

					['all', /1\/2$/, '½', ''],
					['all', /(-+>|\\rightarrow)$/, '?', 'rightwards arrow'],
					['all', /(=>|\\Rightarrow)$/, '?', 'rightwards double arrow'],
					['all', /==>$/, '?', 'long rightwards double arrow'],
					['all', /(‰%|%‰|%%%|\\textpertenthousand)$/i, '?', 'per ten thousand sign'],
					['all', /(%%|\\textperthousand)$/i, '‰', 'per mille sign'],
					['all', /(\(c\)|\\copyright)$/i, '©', 'copyright sign'],
					['all', /\(R\)$/i, '®', 'registered sign'],
					['all', /\(P\)$/i, '?', 'sound recording copyright'],
					['all', /\(tm\)|\\texttrademark$/i, '™', 'trade mark'],
					['all', /(N°|\\textnumero)$/i, '?', 'number sign'],
					['all', /\!\!$/, '?', ''],
					['all', /\?\?$/, '?', ''],
					['all', /\?\!$/, '?', ''],
					['all', /\!\?$/, '?', ''],
					['all', /(\!\?\!|\\textinterrobang)$/, '?', 'interrogbang'],
					['all', /(\!\?\!`|\\textinterrobangdown)$/, '?', 'inverted interrogbang'],
					['all', /\$\*$/, '·', 'middle dot'],
					['all', /(\$x|\\times)$/, '×', 'multiplication sign'],
					['all', /(\$:|\\div)$/, '÷', 'division sign'],
					['all', /\$-$/, '-', 'minus sign'],
					['all', /(\$?\+-|\\pm)$/, '±', 'plus-minus sign'],
					['all', /(\$?-\+|\\mp)$/, '±', 'minus-or-plus sign'],
					['all', /(\$?<=|\\leq)$/, '=', 'less-than or equal to'],
					['all', /(\$?>=|\\geq)$/, '=', 'greater-than or equal to'],
					['all', /\^_$/, '¯', 'macron'],
					['all', /\\alpha$/, 'a', 'alpha'],
					['all', /\\beta$/, 'ß', ''],
					['all', /\\gamma$/, '?', ''],
					['all', /\\delta$/, 'd', ''],
					['all', /\\epsilon$/, '', ''],
					['all', /\\varepsilon$/, 'e', ''],
					['all', /\\zeta$/, '?', ''],
					['all', /\\eta$/, '?', ''],
					['all', /\\theta$/, '?', ''],
					['all', /\\vartheta$/, '', ''],
					['all', /\\kappa$/, '?', ''],
					['all', /\\lambda$/, '?', ''],
					['all', /(\\mu|\\micro)$/, 'µ', ''],
					['all', /\\nu$/, '?', ''],
					['all', /\\xi$/, '?', ''],
					['all', /\\pi$/, 'p', ''],
					['all', /\\varpi$/, '', ''],
					['all', /\\rho$/, '?', ''],
					['all', /\\varrho$/, '', ''],
					['all', /\\sigma$/, 's', ''],
					['all', /\\varsigma$/, '', ''],
					['all', /\\tau$/, 't', ''],
					['all', /\\upsilon$/, '?', ''],
					['all', /\\phi$/, 'f', ''],
					['all', /\\varphi$/, '', ''],
					['all', /\\chi$/, '?', ''],
					['all', /\\psi$/, '?', ''],
					['all', /\\omega$/, '?', ''],
					['all', /\\Gamma$/, 'G'],
					['all', /\\Delta$/, '?'],
					['all', /\\Theta$/, 'T'],
					['all', /\\Lambda$/, '?'],
					['all', /\\Xi$/, '?'],
					['all', /\\Pi$/, '?'],
					['all', /\\Sigma$/, 'S'],
					['all', /\\Upsilon$/, '?'],
					['all', /\\Phi$/, 'F'],
					['all', /\\Psi$/, '?'],
					['all', /(\\Omega|\\ohm)$/, 'O'],
					
					['all', /\\aa$/, 'å'],
					['all', /\\AA$/, 'Å'],
					['all', /\\DH$/, 'Ð'],
					['all', /\\dh$/, 'ð'],
					['all', /\\DJ$/, 'Ð'],
					['all', /\\dj$/, 'd'],
					['all', /\\o$/, 'ø'],
					['all', /\\O$/, 'Ø'],
					['all', /\\ss$/, 'ß'],
					['all', /\\NG$/, '?'],
					['all', /\\ng$/, '?'],
					['all', /\\TH$/, 'Þ'],
					['all', /\\th$/, 'þ'],
					['all', /\\L$/, 'L'],
					['all', /\\l$/, 'l'],
					['all', /\\(`|grave){a}$/, 'à',		'grave accent'],
					['all', /\\('|acute){a}$/, 'á', 'acute accent'],
					['all', /\\(\^|hat){a}$/, 'â', 'circumflex'],
					['all', /(\\("|ddot){a}|a¨)$/, 'ä', 'umlaut or dieresis'],	
					['all', /\\(~|tilde){a}$/, 'ã', 'tilde'],
					['all', /\\k{a}$/, 'a', 'ogonek'],
					['all', /\\(=|bar){a}$/, 'a', 'macron accent (a bar over the letter)'],
					['all', /\\(\.|dot){a}$/, '?', 'dot over the letter'],
					['all', /\\d{a}$/, '?', 'dot under the letter'],
					['all', /\\(r{a}|aa|{aa})$/, 'å', 'ring over the letter (for å there is also the special command \{aa})'],
					['all', /\\(u|breve){a}$/, 'a', 'breve over the letter'],
					['all', /\\(v|check){a}$/, 'a', 'caron/hacek ("v") over the letter'],
					
					['all', /\\b{b}$/, '?', 'b line below'],
					['all', /\\(\.|dot){b}$/, '?', 'b dot above'],
					['all', /\\d{b}$/, '?', 'b dot below'],
					
					/*
					['all', /\\(`|grave){o}$/, 'ò',		'grave accent'],
					['all', /\\('|acute){o}$/, 'ó', 'acute accent'],
					['all', /\\(\^|hat){o}$/, 'ô', 'circumflex'],
					['all', /\\("|ddot){o}$/, 'ö', 'umlaut or dieresis'],
					['all', /\\H{o}$/, 'o', 'long Hungarian umlaut (double acute)'],
					['all', /\\(~|tilde){o}$/, 'õ', 'tilde'],
					['all', /\\c{c}$/, 'ç', 'cedilla'],
					['all', /\\k{a}$/, 'a', 'ogonek'],
					['all', /\\l$/,		'l', 'l with stroke'],
					['all', /\\(=|bar){o}$/, 'o', 'macron accent (a bar over the letter)'],
					['all', /\\b{o}$/, 'o', 'bar under the letter'],
					['all', /\\(\.|dot){o}$/, '?', 'dot over the letter'],
					['all', /\\d{u}$/, '?', 'dot under the letter'],
					['all', /\\r{a}$/, 'å', 'ring over the letter (for å there is also the special command \{aa})'],
					['all', /\\(u|breve){o}$/, 'o', 'breve over the letter'],
					['all', /\\(v|check){s}$/, 'š', 'caron/hacek ("v") over the letter'],
					*/
					['all', /"\+E$/, 'Ë', 'E diaeresis'],
					['all', /"\+e$/, 'ë', 'e diaeresis'],
					['all', /"\+a$/, 'ä', 'a diaeresis'],
					['all', /"\+A$/, 'Ä', 'A diaeresis'],
					['all', /"\+o$/, 'ö', 'a diaeresis'],
					['all', /"\+O$/, 'Ö', 'A diaeresis'],
					['all', /"\+u$/, 'ü', 'u diaeresis'],
					['all', /"\+U$/, 'Ü', 'U diaeresis'],
					['all', /\\t{(.)(.)}$/, '$1?$2', '"tie" (inverted u) over the two letters'], 

					['eng', /dont$/, 'don’t'],				  
					['eng', /doesnt$/, 'doesn’t'],
					['eng', /hasnt$/, 'hasn’t'],
					['eng', /havent$/, 'haven’t'],
					['eng', /isnt$/, 'isn’t'],
					['eng', /arent$/, 'aren’t'],
					['eng', /wasnt$/, 'wasn’t'],
					['eng', /werent$/, 'weren’t'],
					['eng', /hadnt$/, 'hadn’t'],
					['eng', /cant$/, 'can’t'],
					['eng', /couldnt$/, 'couldn’t'],
					['eng', /wouldnt$/, 'wouldn’t'],
					
					['fra', /francais$/, 'français'],
					
					// thanks to [freefighter](http://tatoeba.org/deu/user/profile/freefighter) for the turkish replacement patterns!
					["tur", /aksam üstü$/, "aksamüstü"],
					["tur", /ali koymak$/, "alikoymak"],
					["tur", /anbar$/, "ambar"],
					["tur", /anfi$/, "amfi"],
					["tur", /antreman$/, "antrenman"],
					["tur", /ara söz$/, "arasöz"],
					["tur", /ara yön$/, "arayön"],
					["tur", /arasira$/, "ara sira"],
					["tur", /ardisira$/, "ardi sira"],
					["tur", /arzetmek$/, "arz etmek"],
					["tur", /ayak üstü$/, "ayaküstü"],
					["tur", /ayirdetmek$/, "ayirt etmek"],
					["tur", /ayidat$/, "aidat"],
					["tur", /bas vurma$/, "basvurmak"],
					["tur", /bas vurmak$/, "basvurmak"],
					["tur", /bilimum$/, "bilumum"],
					["tur", /bir az$/, "biraz"],
					["tur", /bir çok sey$/, "birçok sey"],
					["tur", /bir kaç sey$/, "birkaç sey"],
					["tur", /birçok$/, "bir çok"],
					["tur", /birgün$/, "bir gün"],
					["tur", /birsey$/, "bir sey"],
					["tur", /canbaz$/, "cambaz"],
					["tur", /çenber$/, "çember"],
					["tur", /deynek$/, "degnek"],
					["tur", /direk$/, "direkt"],
					["tur", /egsoz$/, "egzoz"],
					["tur", /eksoz$/, "egzoz"],
					["tur", /entellektüel$/, "entelektüel"],
					["tur", /esortman$/, "esofman"],
					["tur", /farketmek$/, "fark etmek"],
					["tur", /haketmek$/, "hak etmek"],
					["tur", /harfiyat$/, "hafriyat"],
					["tur", /herbiri$/, "her biri"],
					["tur", /hergün$/, "her gün"],
					["tur", /herhangibiri$/, "herhangi biri"],
					["tur", /herkez$/, "herkes"],
					["tur", /hersey$/, "her sey"],
					["tur", /heryer$/, "her yer"],
					["tur", /hiç bir sey$/, "hiçbir sey"],
					["tur", /hiçbirsey$/, "hiçbir sey"],
					["tur", /hiçkimse$/, "hiç kimse"],
					["tur", /hosçakal$/, "hosça kal"],
					["tur", /izdirap$/, "istirap"],
					["tur", /insiyatif$/, "inisiyatif"],
					["tur", /kiprik$/, "kirpik"],
					["tur", /kirbit$/, "kibrit"],
					["tur", /kordalye$/, "kurdele"],
					["tur", /laboratuar$/, "laboratuvar"],
					["tur", /makina$/, "makine"],
					["tur", /matba$/, "matbaa"],
					["tur", /metod$/, "metot"],
					["tur", /meyva$/, "meyve"],
					["tur", /muhafiyet$/, "muafiyet"],
					["tur", /muhattap$/, "muhatap"],
					["tur", /muhtacolmak$/, "muhtaç olmak"],
					["tur", /mütevazi$/, "mütevazi"],
					["tur", /orjinal$/, "orijinal"],
					["tur", /ön görmek$/, "öngörmek"],
					["tur", /pardesü$/, "pardösü"],
					["tur", /pekaz$/, "pek az"],
					["tur", /pekçok$/, "pek çok"],
					["tur", /pesisira$/, "pesi sira"],
					["tur", /pilaj$/, "plaj"],
					["tur", /poaça$/, "pogaça"],
					["tur", /sandoviç$/, "sandviç"],
					["tur", /siril siklam$/, "sirilsiklam"],
					["tur", /suç üstü$/, "suçüstü"],
					["tur", /süpriz$/, "sürpriz"],
					["tur", /sartel$/, "salter"],
					["tur", /sarz$/, "sarj"],
					["tur", /sehidolmak$/, "sehit olmak"],
					["tur", /tazik$/, "tazyik"],
					["tur", /tazzik$/, "tazyik"],
					["tur", /terketmek$/, "terk etmek"],
					["tur", /tras$/, "tiras"],
					["tur", /vaadetmek$/, "vaat etmek"],
					["tur", /var saymak$/, "varsaymak"],
					["tur", /vaz geçmek$/, "vazgeçmek"],
					["tur", /ve ya$/, "veya"],
					["tur", /vijdan$/, "vicdan"],
					["tur", /ya hut$/, "yahut"],
					["tur", /yada$/, "ya da"],
					["tur", /yalnis$/, "yanlis"],
					["tur", /yanisira$/, "yani sira"],
					["tur", /yanliz$/, "yalniz"],
					["tur", /zatüre$/, "zatürree"],
					["tur", /zatürre$/, "zatürree"],
					["tur", /allah$/, "Allah"],
					["tur", /af et$/, "affet"],
					["tur", /aferim$/, "aferin"],
					["tur", /bisey$/, "bir sey"],
					["tur", /bi$/, "bir"],
					["tur", /biliyo musun$/, "biliyor musun"],
					["tur", /biliyomusun$/, "biliyor musun"],
					["tur", /burda$/, "burada"],
					["tur", /buda$/, "bu da"],
					["tur", /bugun$/, "bugün"],
					["tur", /bu gün$/, "bugün"],
					["tur", /based$/, "bas ed"],
					["tur", /baset$/, "bas et"],
					["tur", /bir çok$/, "birçok"],
					["tur", /bir çog$/, "birçog"],
					["tur", /bukadar$/, "bu kadar"],
					["tur", /belkide$/, "belki de"],
					["tur", /bayaga$/, "bayagi"],
					["tur", /bikaç$/, "bir kaç"],
					["tur", /biryer$/, "bir yer"],
					["tur", /birazda$/, "biraz da"],
					["tur", /miyeceg$/, "meyeceg"],
					["tur", /birkere$/, "bir kere"],
					["tur", /baris sever$/, "barissever"],
					["tur", /çünki$/, "çünkü"],
					["tur", /cagim$/, "cagim"],
					["tur", /cegim$/, "cegim"],
					["tur", /çokta$/, "çok da"],
					["tur", /çoksatan$/, "çok satan"],
					["tur", /cok$/, "çok"],
					["tur", /dahami$/, "daha mi"],
					["tur", /dahada$/, "daha da"],
					["tur", /degermi$/, "deger mi"],
					["tur", /degermi$/, "deger mi"],
					["tur", /dakka$/, "dakika"],
					["tur", /dogru$/, "dogru"],
					["tur", /degil$/, "degil"],
					["tur", /etmeliyimki$/, "etmeliyim ki"],
					["tur", /eger$/, "eger"],
					["tur", /elbetteki$/, "elbette ki"],
					["tur", /geceyarisi$/, "gece yarisi"],
					["tur", /gördünmü$/, "gördün mü"],
					["tur", /görmiy$/, "görmey"],
					["tur", /hiçbiryere$/, "hiçbir yere"],
					["tur", /her hangi$/, "harhangi"],
					["tur", /herne$/, "her ne"],
					["tur", /haket$/, "hak et"],
					["tur", /hadi$/, "haydi"],
					["tur", /hazir misin$/, "hazir misin"],
					["tur", /haftasonu$/, "hafta sonu"],
					["tur", /hayir$/, "hayir"],
					["tur", /hiçbirsey$/, "hiçbir sey"],
					["tur", /hicbirsey$/, "hiçbir sey"],
					["tur", /hersey$/, "her sey"],
					["tur", /hatirla$/, "hatirla"],
					["tur", /herzaman$/, "her zaman"],
					["tur", /hastene$/, "hastane"],
					["tur", /hiçbirseye$/, "hiçbir seye"],
					["tur", /hicbir$/, "hiçbir"],
					["tur", /hzt$/, "hz"],
					["tur", /hiristiyan$/, "Hristiyan"],
					["tur", /hiristiyan$/, "Hristiyan"],
					["tur", /istermiydin$/, "ister miydin"],
					["tur", /içisler$/, "iç isler"],
					["tur", /lslak$/, "Islak"],
					["tur", /istesende$/, "istesen de"],
					["tur", /Isden$/, "Isten"],
					["tur", /istermisin$/, "ister misin"],
					["tur", /ikisinide$/, "ikisini de"],
					["tur", /Isaya$/, "Isa'ya"],
					["tur", /icin$/, "için"],
					["tur", /Ilerde$/, "Ileride"],
					["tur", /ikimizde$/, "ikimiz de"],
					["tur", /kiz kardes$/, "kiz kardes"],
					["tur", /kizkardes$/, "kiz kardes"],
					["tur", /misdi$/, "misti"],
					["tur", /misdi$/, "misti"],
					["tur", /mahfol$/, "mahvol"],
					["tur", /mahfet$/, "mahvet"],
					["tur", /magruz$/, "maruz"],
					["tur", /mütesekkir$/, "mütesekkir"],
					["tur", /mutesekkir$/, "mütesekkir"],
					["tur", /mr$/, "Mr."],
					["tur", /nerde$/, "nerede"],
					["tur", /naber$/, "n'aber"],
					["tur", /orda$/, "orada"],
					["tur", /oto yol$/, "otoyol"],
					["tur", /olcak$/, "olacak"],
					["tur", /onuda$/, "onu da"],
					["tur", /papayi$/, "Papayi"],
					["tur", /planli$/, "planli"],
					["tur", /super$/, "süper"],
					["tur", /suanda$/, "su anda"],
					["tur", /suanda$/, "su anda"],
					["tur", /senle$/, "seninle"],
					["tur", /sindi$/, "simdi"],
					["tur", /secim$/, "seçim"],
					["tur", /sefkat$/, "sefkat"],
					["tur", /sevkat$/, "sefkat"],
					["tur", /sinirdisi$/, "sinir disi"],
					["tur", /tamammi$/, "tamam mi"],
					["tur", /taktir$/, "takdir"],
					["tur", /tabiki$/, "tabii ki"],
					["tur", /tanri$/, "Tanri"],
					["tur", /tanemi$/, "tane mi"],
					["tur", /tabi$/, "tabii"],
					["tur", /tanri$/, "Tanri"],
					["tur", /tesekkürler$/, "tesekkürler"],
					["tur", /tesekkurler$/, "tesekkürler"],
					["tur", /tamam mi$/, "tamam mi"],
					["tur", /tesekkür$/, "tesekkür"],
					["tur", /ulasamak$/, "ulasmak"],
					["tur", /üc$/, "üç"],
					["tur", /vaz geç$/, "vazgeç"],
					["tur", /vaz geçm$/, "vazgeçm"],
					["tur", /, ve$/, " ve"],
					["tur", /yinede$/, "yine de"],
					["tur", /yoket$/, "yok et"],
					["tur", /yoked$/, "yok ed"],
					["tur", /yapmiyacagim$/, "yapmayacagim"],
					["tur", /yapiyim$/, "yapayim"],
					["tur", /ingiltere$/, "Ingiltere"],
					["tur", /almanya$/, "Almanya"],
					["tur", /ivan$/, "Ivan"],
					["tur", /japon$/, "Japon"],
					["tur", /tokyo$/, "Tokyo"],
					
					['all', /variante$/, 'variante eins', 'test'],
					['all', /variante$/, 'variente zwei', 'test'],
				];

				autocorrect.find('tr').remove();
				apply.unbind('click').html('');
				
				//ac_select.append($('<option class="default">_</option>').attr('html', '<span class="before"></span><span class="gets"> ? </span><span class="after"></span>'));
		
				
				// keycodes: http://asquare.net/javascript/tests/KeyCode.html
						
				// http://stackoverflow.com/questions/3977642/how-to-know-if-keyup-is-a-character-key-jquery
				//var c= String.fromCharCode(event.keyCode);
				//var isWordcharacter = c.match(/\w/);
				
				//console.time("replacement");
				replacements = 0;
				$.each(replacementarray, function (index, regexparray) {
					if(regexparray[0]=='all' || regexparray[0]==markeditlang){
						match = (state.beforeSelect + state.select).match(regexparray[1]);
						if (match) {
							replacements ++;
							/*if(regexparray[4]=='auto' && (isWordcharacter || event.keyCode == "32") ){
								state.beforeSelect = (state.beforeSelect + state.select).replace(regexparray[1], regexparray[2]);
								state.select = "";
								
								thismarkeditor.markeditSetState(state);
								thismarkeditor.keyup();
							}
							else*/
							if(regexparray[4]!='never'){
								
								// reference is the old one, compareto is the new one
								reference = match[0];
								compareto = match[0].replace(regexparray[1], regexparray[2]);
								// initialize the diff_match_patch
								var dmp = new diff_match_patch();
								dmp.Diff_Timeout = 1;
								// compare them
								var d = dmp.diff_main(reference, compareto);
								// this makes it readable
								dmp.diff_cleanupSemantic(d);
								
																
								if(replacements>1){
									line = $('<tr></tr>');
									cell = $('<a></a>').css({'display':'table-cell'}).button();
									autocorrect.append(line.append(cell));
								}
								
								(replacements==1 ? apply : cell).html('<span class="patch"><span class="before">'+dmp.diff_prettyHtml(d)+'</span><span class="gets"> ? </span><span class="after">'+dmp.diff_prettyHtml(d)+'</span></span>').bind("click", function () {
									state.beforeSelect = (state.beforeSelect + state.select).replace(regexparray[1], regexparray[2]);
									state.select = "";
									thismarkeditor.markeditSetState(state);
									thismarkeditor.trigger('keyup');
									//thismarkeditor.trigger('input');
								});
							}
						}
					}
				});
				//console.timeEnd("replacement");
				$(element).html($(this).markeditGetHtml());

			}
		});
		$(textarea).bind('updated', function () {
			$(element).html($(this).markeditGetHtml());
		});

		// Load initial value
		$(element).html($(this).markeditGetHtml());
	};

	// prepare the link toggle button
	// this should toggle between endnote and inline link
	// doesn'T work at all yet
	//$.extend($.markedit(), blah : function(){alert('didit');});

	// enable MarkEdit with live preview 
	$('#SentenceCommentText, #WallContent, #UserDescription, #PrivateMessageContent').addClass('markedittextarea').markedit({
		'preview': 'below',
		'toolbar': {
			'backgroundMode': 'light',
			'layout': 'bold italic | link quote code image | numberlist bulletlist heading line | cite-sentence',
			'buttons': [{
				// prepare the link toggle button
				// this should toggle between endnote and inline link
				// doesn'T work at all yet
				'id': 'linktoggle',
				'css': 'icon ui-state-default ui-corner-all',
				//ui-icon ui-icon-transferthick-e-w
				'tip': 'click to toggle between endnote style and inline style',
				'click': function () {
					//markedit = $(this).parentsUntil('.markedit').parent();
					//markedit.markeditSetLinkOrImage('http://example.com');
				},
				'mouseover': function () {},
				'mouseout': function () {}
			}, {
				'id': 'cite-sentence',
				'css': 'icon ui-state-default ui-corner-all',
				//ui-icon ui-icon-transferthick-e-w
				'tip': 'cite the main sentence',
				'click': function () {
					//find the textarea
					thetextarea = $('.markedittextarea');
					state = thetextarea.markeditGetState();

					// insert sentence citation (the selection will be the proposed change;
					// if it's empty the whole sentence will be copied)
					state.beforeSelect = state.beforeSelect + '"' + $('.mainSentence .sentenceContent .text').text() + '" ? "';
					state.select = (state.select == "" ? $('.mainSentence .sentenceContent .text').text() : state.select);
					state.afterSelect = '"' + state.afterSelect;

					thetextarea.markeditSetState(state);
				},
				'mouseover': function () {},
				'mouseout': function () {}
			}],
		}
	});

	profile = (window.location.href.split('/')[4] == 'user' && window.location.href.split('/')[5] == 'profile');

	if (profile == true) {
		user = $('.profileSummary .username').text().trim();
	} else {
		user = '_';
	}

	$('.comments .commentText, .wall .message .body, .profileDescription .content, .privateMessage .body').each(function () {

		// determine the author of a comment or post
		if (profile == true) {
			author = user;
		} else {
			author = $(this).parentsUntil('li').parent().find('.meta .author:first').text().trim();
		}
		GM_log(author);
		text = $(this).html();

		// trim leading and trailing whitespace to avoid interpretation like code-blocks  
		if($(this).is('.commentText')){
			text = text.substring(9, text.length-8);
		}
		else if($(this).parent().is('.root')){
			text = text.substring(16, text.length-14);
		}
		else if($(this).parent().is('.privateMessage')){
			text = text.substring(9, text.length-8);
		}
		else if($(this).parentsUntil('.replies').parent().is('.replies')){
			text = text.substring(17, text.length-12);
		}


		text = text.replace(/<br\/?>/g,'');
		$(this).html(text);
		
		

		// prepare for parsing:
		text = Pandoc(tatoedown(tatoebaundo(text)));

		// old parsing
		// MarkEditShowDown = new Attacklab.showdown.converter();
		// parsedmarkdown = $('<div class="parsedmarkdown"></div>').html(MarkEditShowDown.makeHtml(text));

		// create the new element
		parsedmarkdown = $('<div class="parsedmarkdown"></div>').html(text);
		// copy the CSS classes to the new element
		parsedmarkdown.addClass($(this).attr("class")).hide();
		// hide the new element and put it below the original 
		var pre = $(this);
		pre.after(parsedmarkdown);
		pre = pre.changeElementType("pre");
		pre.addClass('unparsedmarkdown');
		
		// add classes for later control of visibility
		//GM_log(markdownusers[author]);
		if (markdownusers[author] == true) {
			pre.addClass('md_force');
			parsedmarkdown.addClass('md_force');
		} else if (markdownusers[author] == false) {
			pre.addClass('md_deny');
			parsedmarkdown.addClass('md_deny');
		} else {
			pre.addClass('md_normal');
			parsedmarkdown.addClass('md_normal');
		}
	});

	// This mark was designed by Dustin Curtis, a villain. http://twitter.com/dcurtis
	markdownmark_normal = '<?xml version="1.0" encoding="utf-8"?> <svg xmlns="http://www.w3.org/2000/svg" version="1.1" height="14" viewBox="0 0 208 128" style="vertical-align: text-bottom;" class="markdownmark normal"> <rect style="fill:none;stroke:white;stroke-width:10" width="198" height="118" x="5" y="5" ry="10" /> <path style="fill:white" d="m 30,98 0,-68 20,0 20,25 20,-25 20,0 0,68 -20,0 0,-39 -20,25 -20,-25 0,39 z" /> <path style="fill:white" d="m 155,98 -30,-33 20,0 0,-35 20,0 0,35 20,0 z" /> </svg>';
	markdownmark_solid = '<?xml version="1.0" encoding="utf-8"?> <svg xmlns="http://www.w3.org/2000/svg" version="1.1" height="14" viewBox="0 0 208 128" style="vertical-align: text-bottom;" class="markdownmark solid"> <path d="M 15,0 C 6.7764862,0 0,6.7764862 0,15 l 0,98 c 0,8.22351 6.7764862,15 15,15 l 178,0 c 8.22351,0 15,-6.77649 15,-15 l 0,-98 C 208,6.7764862 201.22351,0 193,0 L 15,0 z m 15,30 20,0 20,25 20,-25 20,0 0,68 -20,0 0,-39 -20,25 -20,-25 0,39 -20,0 0,-68 z m 115,0 20,0 0,35 20,0 -30,33 -30,-33 20,0 0,-35 z" style="fill:white;" /> </svg>';
	
	// create a toggle "button" for switching between "on" and "all off"
	md_button = $('<a class="menuSection" id="md_button"><span class="state"></span>md<span id="mduser"></span></a>');
	md_button.html(markdownmark_normal);
	// add it a list called "menu"
	menu = $('<li id="md"></li>').append(md_button);
	// put it before the user menu in the navigation bar
	$('#user_menu > ul').prepend(menu);

	// the toggle function
	md_button.click(function () {
		GM_log('md_button click');
		// get saved variables
		markdownusers = GM_getValue('markdownusers');	
		markdownusers = markdownusers || default_markdownusers;
		markdownusers = JSON.parse(markdownusers);
		console.log('markdownusers: ' + JSON.stringify(markdownusers));
		// the '#' variable represents the on or all-off toggle
		// if the current state was "on" switch to "all-off"
		if (markdownusers['#'] == true) {
			GM_log('all-off');
			// hide all parsed stuff in "all-off" state
			$('.parsedmarkdown, .markedit-preview').hide();
			$('.unparsedmarkdown').show();
			// update the button
			$('.state').html(markdownmark_solid);
			$('#md_button').html(markdownmark_solid);
			// for saving the state
			markdownusers['#'] = false;
		}
		// if the current state was "all-off", or not set, switch to "on"
		else {
			GM_log('on');
			// switch the parsing on according to the current mode
			if (markdownusers['_'] == true) {
				$('.md_force').toggle();
			} else if (markdownusers['_'] == false) {
				$('.md_force, .md_normal, .md_deny').toggle();
			} else {
				$('.md_force, .md_normal').toggle();
			}
			// the preview shows in all modes when state is on
			$('.markedit-preview').show();
			// update the button
			$('.state').html(markdownmark_normal);
			$('#md_button').html(markdownmark_normal);
			// for saving the state
			markdownusers['#'] = true;
		}
		// save the new state
		GM_setValue('markdownusers', JSON.stringify(markdownusers));
	});

	// create the mode switcher links
	md_force = $('<li><a id="md_force"><span class="state"></span> !</a></li>');
	md_normal = $('<li><a id="md_normal"><span class="state"></span> .</a></li>');
	md_deny = $('<li><a id="md_deny"><span class="state"></span> /</a></li>');

	// create the mode switch list
	list = $('<ul class="sub-menu"></ul>');
	// add the mode switcher links to the mode switch list and put it to the menu
	menu.append(list.append(md_force, md_normal, md_deny));

	// switch to "md_force" mode: only contributions by opt-in users will be parsed
	md_force.click(function () {
		GM_log('md_force click');
		if (profile == true) {
			markdownusers[author] = true;
		} else {
			markdownusers['_'] = true;
		}
		// save the new mode
		GM_setValue('markdownusers', JSON.stringify(markdownusers));
		$('#md_normal, #md_normal, #md_deny').parent().show();
		$('#md_force').attr('disabled', 'disabled');
		$('#md_normal, #md_deny').removeAttr('disabled');
		$('#mduser').text("!");
		// click twice to unset and reset, so the navigation menu shows the right state
		md_button.click().click();
	});
	// switch to "md_normal" mode: all contributions except the opt-out users will be parsed
	md_normal.click(function () {
		GM_log('md_normal click');
		if (profile == true) {
			delete markdownusers[author];
		} else {
			delete markdownusers['_'];
		}
		// save the new mode
		GM_setValue('markdownusers', JSON.stringify(markdownusers));
		$('#md_normal, #md_normal, #md_deny').parent().show();
		$('#md_normal').attr('disabled', 'disabled');
		$('#md_force, #md_deny').removeAttr('disabled');
		$('#mduser').text(".");
		// click twice to unset and reset, so the navigation menu shows the right state
		md_button.click().click();
	});
	// switch to "md_deny" mode: all contributions, even the opt-out users, will be parsed
	md_deny.click(function () {
		GM_log('md_deny click');
		if (profile == true) {
			markdownusers[author] = false;
		} else {
			markdownusers['_'] = false;
		}
		// save the new mode
		GM_setValue('markdownusers', JSON.stringify(markdownusers));
		$('#md_normal, #md_normal, #md_deny').parent().show();
		$('#md_deny').attr('disabled', 'disabled');
		$('#md_force, #md_normal').removeAttr('disabled');
		$('#mduser').text("/");
		// click twice to unset and reset, so the navigation menu shows the right state
		md_button.click().click();
	});

	if (profile == true) {
		if (markdownusers[author] === true) {
			md_force.click();
		} else if (markdownusers[author] === false) {
			md_deny.click();
		} else {
			md_normal.click();
		}
		$('#md_normal, #md_normal, #md_deny, #md_button').css({
			'font-style': 'italic'
		});
	} else {
		if (markdownusers['_'] === true) {
			md_force.click();
		} else if (markdownusers['_'] === false) {
			md_deny.click();
		} else {
			md_normal.click();
		}

	}
}