1*2946d023SBram Moolenaar" Function to left and right align text. 2*2946d023SBram Moolenaar" 3*2946d023SBram Moolenaar" Written by: Preben "Peppe" Guldberg <[email protected]> 4*2946d023SBram Moolenaar" Created: 980806 14:13 (or around that time anyway) 5*2946d023SBram Moolenaar" Revised: 001103 00:36 (See "Revisions" below) 6*2946d023SBram Moolenaar 7*2946d023SBram Moolenaar 8*2946d023SBram Moolenaar" function Justify( [ textwidth [, maxspaces [, indent] ] ] ) 9*2946d023SBram Moolenaar" 10*2946d023SBram Moolenaar" Justify() will left and right align a line by filling in an 11*2946d023SBram Moolenaar" appropriate amount of spaces. Extra spaces are added to existing 12*2946d023SBram Moolenaar" spaces starting from the right side of the line. As an example, the 13*2946d023SBram Moolenaar" following documentation has been justified. 14*2946d023SBram Moolenaar" 15*2946d023SBram Moolenaar" The function takes the following arguments: 16*2946d023SBram Moolenaar 17*2946d023SBram Moolenaar" textwidth argument 18*2946d023SBram Moolenaar" ------------------ 19*2946d023SBram Moolenaar" If not specified, the value of the 'textwidth' option is used. If 20*2946d023SBram Moolenaar" 'textwidth' is zero a value of 80 is used. 21*2946d023SBram Moolenaar" 22*2946d023SBram Moolenaar" Additionally the arguments 'tw' and '' are accepted. The value of 23*2946d023SBram Moolenaar" 'textwidth' will be used. These are handy, if you just want to specify 24*2946d023SBram Moolenaar" the maxspaces argument. 25*2946d023SBram Moolenaar 26*2946d023SBram Moolenaar" maxspaces argument 27*2946d023SBram Moolenaar" ------------------ 28*2946d023SBram Moolenaar" If specified, alignment will only be done, if the longest space run 29*2946d023SBram Moolenaar" after alignment is no longer than maxspaces. 30*2946d023SBram Moolenaar" 31*2946d023SBram Moolenaar" An argument of '' is accepted, should the user like to specify all 32*2946d023SBram Moolenaar" arguments. 33*2946d023SBram Moolenaar" 34*2946d023SBram Moolenaar" To aid user defined commands, negative values are accepted aswell. 35*2946d023SBram Moolenaar" Using a negative value specifies the default behaviour: any length of 36*2946d023SBram Moolenaar" space runs will be used to justify the text. 37*2946d023SBram Moolenaar 38*2946d023SBram Moolenaar" indent argument 39*2946d023SBram Moolenaar" --------------- 40*2946d023SBram Moolenaar" This argument specifies how a line should be indented. The default is 41*2946d023SBram Moolenaar" to keep the current indentation. 42*2946d023SBram Moolenaar" 43*2946d023SBram Moolenaar" Negative values: Keep current amount of leading whitespace. 44*2946d023SBram Moolenaar" Positive values: Indent all lines with leading whitespace using this 45*2946d023SBram Moolenaar" amount of whitespace. 46*2946d023SBram Moolenaar" 47*2946d023SBram Moolenaar" Note that the value 0, needs to be quoted as a string. This value 48*2946d023SBram Moolenaar" leads to a left flushed text. 49*2946d023SBram Moolenaar" 50*2946d023SBram Moolenaar" Additionally units of 'shiftwidth'/'sw' and 'tabstop'/'ts' may be 51*2946d023SBram Moolenaar" added. In this case, if the value of indent is positive, the amount of 52*2946d023SBram Moolenaar" whitespace to be added will be multiplied by the value of the 53*2946d023SBram Moolenaar" 'shiftwidth' and 'tabstop' settings. If these units are used, the 54*2946d023SBram Moolenaar" argument must be given as a string, eg. Justify('','','2sw'). 55*2946d023SBram Moolenaar" 56*2946d023SBram Moolenaar" If the values of 'sw' or 'tw' are negative, they are treated as if 57*2946d023SBram Moolenaar" they were 0, which means that the text is flushed left. There is no 58*2946d023SBram Moolenaar" check if a negative number prefix is used to change the sign of a 59*2946d023SBram Moolenaar" negative 'sw' or 'ts' value. 60*2946d023SBram Moolenaar" 61*2946d023SBram Moolenaar" As with the other arguments, '' may be used to get the default 62*2946d023SBram Moolenaar" behaviour. 63*2946d023SBram Moolenaar 64*2946d023SBram Moolenaar 65*2946d023SBram Moolenaar" Notes: 66*2946d023SBram Moolenaar" 67*2946d023SBram Moolenaar" If the line, adjusted for space runs and leading/trailing whitespace, 68*2946d023SBram Moolenaar" is wider than the used textwidth, the line will be left untouched (no 69*2946d023SBram Moolenaar" whitespace removed). This should be equivalent to the behaviour of 70*2946d023SBram Moolenaar" :left, :right and :center. 71*2946d023SBram Moolenaar" 72*2946d023SBram Moolenaar" If the resulting line is shorter than the used textwidth it is left 73*2946d023SBram Moolenaar" untouched. 74*2946d023SBram Moolenaar" 75*2946d023SBram Moolenaar" All space runs in the line are truncated before the alignment is 76*2946d023SBram Moolenaar" carried out. 77*2946d023SBram Moolenaar" 78*2946d023SBram Moolenaar" If you have set 'noexpandtab', :retab! is used to replace space runs 79*2946d023SBram Moolenaar" with whitespace using the value of 'tabstop'. This should be 80*2946d023SBram Moolenaar" conformant with :left, :right and :center. 81*2946d023SBram Moolenaar" 82*2946d023SBram Moolenaar" If joinspaces is set, an extra space is added after '.', '?' and '!'. 83*2946d023SBram Moolenaar" If 'cpooptions' include 'j', extra space is only added after '.'. 84*2946d023SBram Moolenaar" (This may on occasion conflict with maxspaces.) 85*2946d023SBram Moolenaar 86*2946d023SBram Moolenaar 87*2946d023SBram Moolenaar" Related mappings: 88*2946d023SBram Moolenaar" 89*2946d023SBram Moolenaar" Mappings that will align text using the current text width, using at 90*2946d023SBram Moolenaar" most four spaces in a space run and keeping current indentation. 91*2946d023SBram Moolenaarnmap _j :%call Justify('tw',4)<CR> 92*2946d023SBram Moolenaarvmap _j :call Justify('tw',4)<CR> 93*2946d023SBram Moolenaar" 94*2946d023SBram Moolenaar" Mappings that will remove space runs and format lines (might be useful 95*2946d023SBram Moolenaar" prior to aligning the text). 96*2946d023SBram Moolenaarnmap ,gq :%s/\s\+/ /g<CR>gq1G 97*2946d023SBram Moolenaarvmap ,gq :s/\s\+/ /g<CR>gvgq 98*2946d023SBram Moolenaar 99*2946d023SBram Moolenaar 100*2946d023SBram Moolenaar" User defined command: 101*2946d023SBram Moolenaar" 102*2946d023SBram Moolenaar" The following is an ex command that works as a shortcut to the Justify 103*2946d023SBram Moolenaar" function. Arguments to Justify() can be added after the command. 104*2946d023SBram Moolenaarcom! -range -nargs=* Justify <line1>,<line2>call Justify(<f-args>) 105*2946d023SBram Moolenaar" 106*2946d023SBram Moolenaar" The following commands are all equivalent: 107*2946d023SBram Moolenaar" 108*2946d023SBram Moolenaar" 1. Simplest use of Justify(): 109*2946d023SBram Moolenaar" :call Justify() 110*2946d023SBram Moolenaar" :Justify 111*2946d023SBram Moolenaar" 112*2946d023SBram Moolenaar" 2. The _j mapping above via the ex command: 113*2946d023SBram Moolenaar" :%Justify tw 4 114*2946d023SBram Moolenaar" 115*2946d023SBram Moolenaar" 3. Justify visualised text at 72nd column while indenting all 116*2946d023SBram Moolenaar" previously indented text two shiftwidths 117*2946d023SBram Moolenaar" :'<,'>call Justify(72,'','2sw') 118*2946d023SBram Moolenaar" :'<,'>Justify 72 -1 2sw 119*2946d023SBram Moolenaar" 120*2946d023SBram Moolenaar" This documentation has been justified using the following command: 121*2946d023SBram Moolenaar":se et|kz|1;/^" function Justify(/+,'z-g/^" /s/^" //|call Justify(70,3)|s/^/" / 122*2946d023SBram Moolenaar 123*2946d023SBram Moolenaar" Revisions: 124*2946d023SBram Moolenaar" 001103: If 'joinspaces' was set, calculations could be wrong. 125*2946d023SBram Moolenaar" Tabs at start of line could also lead to errors. 126*2946d023SBram Moolenaar" Use setline() instead of "exec 's/foo/bar/' - safer. 127*2946d023SBram Moolenaar" Cleaned up the code a bit. 128*2946d023SBram Moolenaar" 129*2946d023SBram Moolenaar" Todo: Convert maps to the new script specific form 130*2946d023SBram Moolenaar 131*2946d023SBram Moolenaar" Error function 132*2946d023SBram Moolenaarfunction! Justify_error(message) 133*2946d023SBram Moolenaar echohl Error 134*2946d023SBram Moolenaar echo "Justify([tw, [maxspaces [, indent]]]): " . a:message 135*2946d023SBram Moolenaar echohl None 136*2946d023SBram Moolenaarendfunction 137*2946d023SBram Moolenaar 138*2946d023SBram Moolenaar 139*2946d023SBram Moolenaar" Now for the real thing 140*2946d023SBram Moolenaarfunction! Justify(...) range 141*2946d023SBram Moolenaar 142*2946d023SBram Moolenaar if a:0 > 3 143*2946d023SBram Moolenaar call Justify_error("Too many arguments (max 3)") 144*2946d023SBram Moolenaar return 1 145*2946d023SBram Moolenaar endif 146*2946d023SBram Moolenaar 147*2946d023SBram Moolenaar " Set textwidth (accept 'tw' and '' as arguments) 148*2946d023SBram Moolenaar if a:0 >= 1 149*2946d023SBram Moolenaar if a:1 =~ '^\(tw\)\=$' 150*2946d023SBram Moolenaar let tw = &tw 151*2946d023SBram Moolenaar elseif a:1 =~ '^\d\+$' 152*2946d023SBram Moolenaar let tw = a:1 153*2946d023SBram Moolenaar else 154*2946d023SBram Moolenaar call Justify_error("tw must be a number (>0), '' or 'tw'") 155*2946d023SBram Moolenaar return 2 156*2946d023SBram Moolenaar endif 157*2946d023SBram Moolenaar else 158*2946d023SBram Moolenaar let tw = &tw 159*2946d023SBram Moolenaar endif 160*2946d023SBram Moolenaar if tw == 0 161*2946d023SBram Moolenaar let tw = 80 162*2946d023SBram Moolenaar endif 163*2946d023SBram Moolenaar 164*2946d023SBram Moolenaar " Set maximum number of spaces between WORDs 165*2946d023SBram Moolenaar if a:0 >= 2 166*2946d023SBram Moolenaar if a:2 == '' 167*2946d023SBram Moolenaar let maxspaces = tw 168*2946d023SBram Moolenaar elseif a:2 =~ '^-\d\+$' 169*2946d023SBram Moolenaar let maxspaces = tw 170*2946d023SBram Moolenaar elseif a:2 =~ '^\d\+$' 171*2946d023SBram Moolenaar let maxspaces = a:2 172*2946d023SBram Moolenaar else 173*2946d023SBram Moolenaar call Justify_error("maxspaces must be a number or ''") 174*2946d023SBram Moolenaar return 3 175*2946d023SBram Moolenaar endif 176*2946d023SBram Moolenaar else 177*2946d023SBram Moolenaar let maxspaces = tw 178*2946d023SBram Moolenaar endif 179*2946d023SBram Moolenaar if maxspaces <= 1 180*2946d023SBram Moolenaar call Justify_error("maxspaces should be larger than 1") 181*2946d023SBram Moolenaar return 4 182*2946d023SBram Moolenaar endif 183*2946d023SBram Moolenaar 184*2946d023SBram Moolenaar " Set the indentation style (accept sw and ts units) 185*2946d023SBram Moolenaar let indent_fix = '' 186*2946d023SBram Moolenaar if a:0 >= 3 187*2946d023SBram Moolenaar if (a:3 == '') || a:3 =~ '^-[1-9]\d*\(shiftwidth\|sw\|tabstop\|ts\)\=$' 188*2946d023SBram Moolenaar let indent = -1 189*2946d023SBram Moolenaar elseif a:3 =~ '^-\=0\(shiftwidth\|sw\|tabstop\|ts\)\=$' 190*2946d023SBram Moolenaar let indent = 0 191*2946d023SBram Moolenaar elseif a:3 =~ '^\d\+\(shiftwidth\|sw\|tabstop\|ts\)\=$' 192*2946d023SBram Moolenaar let indent = substitute(a:3, '\D', '', 'g') 193*2946d023SBram Moolenaar elseif a:3 =~ '^\(shiftwidth\|sw\|tabstop\|ts\)$' 194*2946d023SBram Moolenaar let indent = 1 195*2946d023SBram Moolenaar else 196*2946d023SBram Moolenaar call Justify_error("indent: a number with 'sw'/'ts' unit") 197*2946d023SBram Moolenaar return 5 198*2946d023SBram Moolenaar endif 199*2946d023SBram Moolenaar if indent >= 0 200*2946d023SBram Moolenaar while indent > 0 201*2946d023SBram Moolenaar let indent_fix = indent_fix . ' ' 202*2946d023SBram Moolenaar let indent = indent - 1 203*2946d023SBram Moolenaar endwhile 204*2946d023SBram Moolenaar let indent_sw = 0 205*2946d023SBram Moolenaar if a:3 =~ '\(shiftwidth\|sw\)' 206*2946d023SBram Moolenaar let indent_sw = &sw 207*2946d023SBram Moolenaar elseif a:3 =~ '\(tabstop\|ts\)' 208*2946d023SBram Moolenaar let indent_sw = &ts 209*2946d023SBram Moolenaar endif 210*2946d023SBram Moolenaar let indent_fix2 = '' 211*2946d023SBram Moolenaar while indent_sw > 0 212*2946d023SBram Moolenaar let indent_fix2 = indent_fix2 . indent_fix 213*2946d023SBram Moolenaar let indent_sw = indent_sw - 1 214*2946d023SBram Moolenaar endwhile 215*2946d023SBram Moolenaar let indent_fix = indent_fix2 216*2946d023SBram Moolenaar endif 217*2946d023SBram Moolenaar else 218*2946d023SBram Moolenaar let indent = -1 219*2946d023SBram Moolenaar endif 220*2946d023SBram Moolenaar 221*2946d023SBram Moolenaar " Avoid substitution reports 222*2946d023SBram Moolenaar let save_report = &report 223*2946d023SBram Moolenaar set report=1000000 224*2946d023SBram Moolenaar 225*2946d023SBram Moolenaar " Check 'joinspaces' and 'cpo' 226*2946d023SBram Moolenaar if &js == 1 227*2946d023SBram Moolenaar if &cpo =~ 'j' 228*2946d023SBram Moolenaar let join_str = '\(\. \)' 229*2946d023SBram Moolenaar else 230*2946d023SBram Moolenaar let join_str = '\([.!?!] \)' 231*2946d023SBram Moolenaar endif 232*2946d023SBram Moolenaar endif 233*2946d023SBram Moolenaar 234*2946d023SBram Moolenaar let cur = a:firstline 235*2946d023SBram Moolenaar while cur <= a:lastline 236*2946d023SBram Moolenaar 237*2946d023SBram Moolenaar let str_orig = getline(cur) 238*2946d023SBram Moolenaar let save_et = &et 239*2946d023SBram Moolenaar set et 240*2946d023SBram Moolenaar exec cur . "retab" 241*2946d023SBram Moolenaar let &et = save_et 242*2946d023SBram Moolenaar let str = getline(cur) 243*2946d023SBram Moolenaar 244*2946d023SBram Moolenaar let indent_str = indent_fix 245*2946d023SBram Moolenaar let indent_n = strlen(indent_str) 246*2946d023SBram Moolenaar " Shall we remember the current indentation 247*2946d023SBram Moolenaar if indent < 0 248*2946d023SBram Moolenaar let indent_orig = matchstr(str_orig, '^\s*') 249*2946d023SBram Moolenaar if strlen(indent_orig) > 0 250*2946d023SBram Moolenaar let indent_str = indent_orig 251*2946d023SBram Moolenaar let indent_n = strlen(matchstr(str, '^\s*')) 252*2946d023SBram Moolenaar endif 253*2946d023SBram Moolenaar endif 254*2946d023SBram Moolenaar 255*2946d023SBram Moolenaar " Trim trailing, leading and running whitespace 256*2946d023SBram Moolenaar let str = substitute(str, '\s\+$', '', '') 257*2946d023SBram Moolenaar let str = substitute(str, '^\s\+', '', '') 258*2946d023SBram Moolenaar let str = substitute(str, '\s\+', ' ', 'g') 259*2946d023SBram Moolenaar let str_n = strdisplaywidth(str) 260*2946d023SBram Moolenaar 261*2946d023SBram Moolenaar " Possible addition of space after punctuation 262*2946d023SBram Moolenaar if exists("join_str") 263*2946d023SBram Moolenaar let str = substitute(str, join_str, '\1 ', 'g') 264*2946d023SBram Moolenaar endif 265*2946d023SBram Moolenaar let join_n = strdisplaywidth(str) - str_n 266*2946d023SBram Moolenaar 267*2946d023SBram Moolenaar " Can extraspaces be added? 268*2946d023SBram Moolenaar " Note that str_n may be less than strlen(str) [joinspaces above] 269*2946d023SBram Moolenaar if strdisplaywidth(str) <= tw - indent_n && str_n > 0 270*2946d023SBram Moolenaar " How many spaces should be added 271*2946d023SBram Moolenaar let s_add = tw - str_n - indent_n - join_n 272*2946d023SBram Moolenaar let s_nr = strlen(substitute(str, '\S', '', 'g') ) - join_n 273*2946d023SBram Moolenaar let s_dup = s_add / s_nr 274*2946d023SBram Moolenaar let s_mod = s_add % s_nr 275*2946d023SBram Moolenaar 276*2946d023SBram Moolenaar " Test if the changed line fits with tw 277*2946d023SBram Moolenaar if 0 <= (str_n + (maxspaces - 1)*s_nr + indent_n) - tw 278*2946d023SBram Moolenaar 279*2946d023SBram Moolenaar " Duplicate spaces 280*2946d023SBram Moolenaar while s_dup > 0 281*2946d023SBram Moolenaar let str = substitute(str, '\( \+\)', ' \1', 'g') 282*2946d023SBram Moolenaar let s_dup = s_dup - 1 283*2946d023SBram Moolenaar endwhile 284*2946d023SBram Moolenaar 285*2946d023SBram Moolenaar " Add extra spaces from the end 286*2946d023SBram Moolenaar while s_mod > 0 287*2946d023SBram Moolenaar let str = substitute(str, '\(\(\s\+\S\+\)\{' . s_mod . '}\)$', ' \1', '') 288*2946d023SBram Moolenaar let s_mod = s_mod - 1 289*2946d023SBram Moolenaar endwhile 290*2946d023SBram Moolenaar 291*2946d023SBram Moolenaar " Indent the line 292*2946d023SBram Moolenaar if indent_n > 0 293*2946d023SBram Moolenaar let str = substitute(str, '^', indent_str, '' ) 294*2946d023SBram Moolenaar endif 295*2946d023SBram Moolenaar 296*2946d023SBram Moolenaar " Replace the line 297*2946d023SBram Moolenaar call setline(cur, str) 298*2946d023SBram Moolenaar 299*2946d023SBram Moolenaar " Convert to whitespace 300*2946d023SBram Moolenaar if &et == 0 301*2946d023SBram Moolenaar exec cur . 'retab!' 302*2946d023SBram Moolenaar endif 303*2946d023SBram Moolenaar 304*2946d023SBram Moolenaar endif " Change of line 305*2946d023SBram Moolenaar endif " Possible change 306*2946d023SBram Moolenaar 307*2946d023SBram Moolenaar let cur = cur + 1 308*2946d023SBram Moolenaar endwhile 309*2946d023SBram Moolenaar 310*2946d023SBram Moolenaar norm ^ 311*2946d023SBram Moolenaar 312*2946d023SBram Moolenaar let &report = save_report 313*2946d023SBram Moolenaar 314*2946d023SBram Moolenaarendfunction 315*2946d023SBram Moolenaar 316*2946d023SBram Moolenaar" EOF vim: tw=78 ts=8 sw=4 sts=4 noet ai 317