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