diff --git a/inc/toolbar.php b/inc/toolbar.php index d4a9b3a94c33422f0fca83b1828785835d488714..1f34f3403f0cc38b72d8fe4d25e0ed82f112e25f 100644 --- a/inc/toolbar.php +++ b/inc/toolbar.php @@ -161,19 +161,19 @@ function toolbar_JSdefines($varname){ 'sample' => 'http://example.com|'.$lang['qb_extlink'], ), array( - 'type' => 'format', + 'type' => 'formatln', 'title' => $lang['qb_ol'], 'icon' => 'ol.png', 'open' => ' - ', - 'close' => '\n', + 'close' => '', 'key' => '-', ), array( - 'type' => 'format', + 'type' => 'formatln', 'title' => $lang['qb_ul'], 'icon' => 'ul.png', 'open' => ' * ', - 'close' => '\n', + 'close' => '', 'key' => '.', ), array( diff --git a/lib/scripts/edit.js b/lib/scripts/edit.js index 267f511940ca0cc40ef7f9d6076ccb05691bae8e..51a8e3f8e0a3077b987cdcfa737370ce98c5dd3e 100644 --- a/lib/scripts/edit.js +++ b/lib/scripts/edit.js @@ -186,8 +186,75 @@ function addBtnActionAutohead(btn, props, edid, id) return true; } +/** + * Make intended formattings easier to handle + * + * Listens to all key inputs and handle indentions + * of lists and code blocks + * + * Currently handles space, backspce and enter presses + * + * @author Andreas Gohr <andi@splitbrain.org> + * @fixme handle tabs + * @fixme IE compatibility not tested yet + */ +function keyHandler(e){ + if(e.keyCode != 13 && + e.keyCode != 8 && + e.keyCode != 32) return; //FIXME IE + var field = e.target; + var selection = getSelection(field); + var search = "\n"+field.value.substr(0,selection.start); + var linestart = search.lastIndexOf("\n"); + search = search.substr(linestart); + + if(e.keyCode == 13){ // Enter //FIXME IE + // keep current indention for lists and code + var match = search.match(/(\n +([*-] ?)?)/); + if(match){ + insertAtCarret(field.id,match[1]); + e.preventDefault(); // prevent enter key + } + }else if(e.keyCode == 8){ // Backspace + // unindent lists + var match = search.match(/(\n +)([*-] ?)$/); + if(match){ + var spaces = match[1].length-1; + + if(spaces > 3){ // unindent one level + field.value = field.value.substr(0,linestart)+ + field.value.substr(linestart+2); + selection.start = selection.start - 2; + selection.end = selection.start; + }else{ // delete list point + field.value = field.value.substr(0,linestart)+ + field.value.substr(selection.start); + selection.start = linestart; + selection.end = linestart; + } + setSelection(selection); + e.preventDefault(); // prevent backspace + } + }else if(e.keyCode == 32){ // Space + // intend list item + var match = search.match(/(\n +)([*-] )$/); + if(match){ + field.value = field.value.substr(0,linestart)+' '+ + field.value.substr(linestart); + selection.start = selection.start + 2; + selection.end = selection.start; + setSelection(selection); + e.preventDefault(); // prevent space + } + } +} - +//FIXME consolidate somewhere else +addInitEvent(function(){ + var field = $('wiki__text'); + if(!field) return; + addEvent(field,'keydown',keyHandler); +}); /** * Determine the current section level while editing diff --git a/lib/scripts/textselection.js b/lib/scripts/textselection.js index f005fa3da9d151e69f68437b2a045ba2e0cd8e6b..3ebab35e4826614df49d18ebe9098c14112ad18f 100644 --- a/lib/scripts/textselection.js +++ b/lib/scripts/textselection.js @@ -99,8 +99,8 @@ function setSelection(selection){ * @param string text - the new text to be pasted * @param objct selecttion - selection object returned by getSelection * @param int opts.startofs - number of charcters at the start to skip from new selection - * @param int opts.endofs - number of charcters at the end to skip from new selection - * @param bool opts.ofs - set tru if new text should not be selected + * @param int opts.endofs - number of characters at the end to skip from new selection + * @param bool opts.nosel - set true if new text should not be selected */ function pasteText(selection,text,opts){ if(!opts) opts = {}; diff --git a/lib/scripts/toolbar.js b/lib/scripts/toolbar.js index 1a152b1a7d250b060baf9c273528fa909e09e3db..48a4a4a7e0ec64c61f07f23082efc332be92499d 100644 --- a/lib/scripts/toolbar.js +++ b/lib/scripts/toolbar.js @@ -77,6 +77,40 @@ function tb_format(btn, props, edid) { return false; } +/** + * Button action for format buttons + * + * This works exactly as tb_format() except that, if multiple lines + * are selected, each line will be formatted seperately + * + * @param DOMElement btn Button element to add the action to + * @param array props Associative array of button properties + * @param string edid ID of the editor textarea + * @author Gabriel Birke <birke@d-scribe.de> + * @author Andreas Gohr <andi@splitbrain.org> + */ +function tb_formatln(btn, props, edid) { + var sample = props['title']; + if(props['sample']){ + sample = props['sample']; + } + sample = fixtxt(sample); + + var selection = getSelection($(edid)); + if(selection.getLength()) sample = selection.getText(); + + props['open'] = fixtxt(props['open']); + props['close'] = fixtxt(props['close']); + + sample = sample.split("\n").join(props['close']+"\n"+props['open']); + sample = props['open']+sample+props['close']; + + pasteText(selection,sample,{nosel: true}); + + pickerClose(); + return false; +} + /** * Button action for insert buttons *