builder.vim 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. " MIT License. Copyright (c) 2013-2021 Bailey Ling et al.
  2. " vim: et ts=2 sts=2 sw=2
  3. scriptencoding utf-8
  4. let s:prototype = {}
  5. " Set the point in the tabline where the builder should insert the titles.
  6. "
  7. " Subsequent calls will overwrite the previous ones, so only the last call
  8. " determines to location to insert titles.
  9. "
  10. " NOTE: The titles are not inserted until |build| is called, so that the
  11. " remaining contents of the tabline can be taken into account.
  12. "
  13. " Callers should define at least |get_title| and |get_group| on the host
  14. " object before calling |build|.
  15. function! s:prototype.insert_titles(current, first, last) dict
  16. let self._first_title = a:first " lowest index
  17. let self._last_title = a:last " highest index
  18. let self._left_title = a:current " next index to add on the left
  19. let self._right_title = a:current + 1 " next index to add on the right
  20. let self._left_position = self.get_position() " left end of titles
  21. let self._right_position = self._left_position " right end of the titles
  22. endfunction
  23. " Insert a title for entry number |index|, of group |group| at position |pos|,
  24. " if there is space for it. Returns 1 if it is inserted, 0 otherwise
  25. "
  26. " |force| inserts the title even if there isn't enough space left for it.
  27. " |sep_size| adjusts the size change that the title is considered to take up,
  28. " to account for changes to the separators
  29. "
  30. " The title is defined by |get_title| on the hosting object, called with
  31. " |index| as its only argument.
  32. " |get_pretitle| and |get_posttitle| may be defined on the host object to
  33. " insert some formatting before or after the title. These should be 0-width.
  34. "
  35. " This method updates |_right_position| and |_remaining_space| on the host
  36. " object, if the title is inserted.
  37. function! s:prototype.try_insert_title(index, group, pos, sep_size, force) dict
  38. let title = self.get_title(a:index)
  39. let title_size = s:tabline_evaluated_length(title) + a:sep_size
  40. if a:force || self._remaining_space >= title_size
  41. let pos = a:pos
  42. if has_key(self, "get_pretitle")
  43. call self.insert_raw(self.get_pretitle(a:index), pos)
  44. let self._right_position += 1
  45. let pos += 1
  46. endif
  47. call self.insert_section(a:group, title, pos)
  48. let self._right_position += 1
  49. let pos += 1
  50. if has_key(self, "get_posttitle")
  51. call self.insert_raw(self.get_posttitle(a:index), pos)
  52. let self._right_position += 1
  53. let pos += 1
  54. endif
  55. let self._remaining_space -= title_size
  56. return 1
  57. endif
  58. return 0
  59. endfunction
  60. function! s:get_separator_change(new_group, old_group, end_group, sep_size, alt_sep_size)
  61. return s:get_separator_change_with_end(a:new_group, a:old_group, a:end_group, a:end_group, a:sep_size, a:alt_sep_size)
  62. endfunction
  63. " Compute the change in size of the tabline caused by separators
  64. "
  65. " This should be kept up-to-date with |s:get_transitioned_separator| and
  66. " |s:get_separator| in autoload/airline/builder.vim
  67. function! s:get_separator_change_with_end(new_group, old_group, new_end_group, old_end_group, sep_size, alt_sep_size)
  68. let sep_change = 0
  69. if !empty(a:new_end_group) " Separator between title and the end
  70. let sep_change += airline#builder#should_change_group(a:new_group, a:new_end_group) ? a:sep_size : a:alt_sep_size
  71. endif
  72. if !empty(a:old_group) " Separator between the title and the one adjacent
  73. let sep_change += airline#builder#should_change_group(a:new_group, a:old_group) ? a:sep_size : a:alt_sep_size
  74. if !empty(a:old_end_group) " Remove mis-predicted separator
  75. let sep_change -= airline#builder#should_change_group(a:old_group, a:old_end_group) ? a:sep_size : a:alt_sep_size
  76. endif
  77. endif
  78. return sep_change
  79. endfunction
  80. " This replaces the build function of the |airline#builder#new| object, to
  81. " insert titles as specified by the last call to |insert_titles| before
  82. " passing to the original build function.
  83. "
  84. " Callers should define at least |get_title| and |get_group| on the host
  85. " object if |insert_titles| has been called on it.
  86. function! s:prototype.build() dict
  87. if has_key(self, '_left_position') && self._first_title <= self._last_title
  88. let self._remaining_space = &columns - s:tabline_evaluated_length(self._build())
  89. let center_active = get(g:, 'airline#extensions#tabline#center_active', 0)
  90. let sep_size = s:tabline_evaluated_length(self._context.left_sep)
  91. let alt_sep_size = s:tabline_evaluated_length(self._context.left_alt_sep)
  92. let outer_left_group = airline#builder#get_prev_group(self._sections, self._left_position)
  93. let outer_right_group = airline#builder#get_next_group(self._sections, self._right_position)
  94. let overflow_marker = get(g:, 'airline#extensions#tabline#overflow_marker', g:airline_symbols.ellipsis)
  95. let overflow_marker_size = s:tabline_evaluated_length(overflow_marker)
  96. " Allow space for the markers before we begin filling in titles.
  97. if self._left_title > self._first_title
  98. let self._remaining_space -= overflow_marker_size +
  99. \ s:get_separator_change(self.overflow_group, "", outer_left_group, sep_size, alt_sep_size)
  100. endif
  101. if self._left_title < self._last_title
  102. let self._remaining_space -= overflow_marker_size +
  103. \ s:get_separator_change(self.overflow_group, "", outer_right_group, sep_size, alt_sep_size)
  104. endif
  105. " Add the current title
  106. let group = self.get_group(self._left_title)
  107. if self._left_title == self._first_title
  108. let sep_change = s:get_separator_change(group, "", outer_left_group, sep_size, alt_sep_size)
  109. else
  110. let sep_change = s:get_separator_change(group, "", self.overflow_group, sep_size, alt_sep_size)
  111. endif
  112. if self._left_title == self._last_title
  113. let sep_change += s:get_separator_change(group, "", outer_right_group, sep_size, alt_sep_size)
  114. else
  115. let sep_change += s:get_separator_change(group, "", self.overflow_group, sep_size, alt_sep_size)
  116. endif
  117. let left_group = group
  118. let right_group = group
  119. let self._left_title -=
  120. \ self.try_insert_title(self._left_title, group, self._left_position, sep_change, 1)
  121. if get(g:, 'airline#extensions#tabline#current_first', 0)
  122. " always have current title first
  123. let self._left_position += 1
  124. endif
  125. if !center_active && self._right_title <= self._last_title
  126. " Add the title to the right
  127. let group = self.get_group(self._right_title)
  128. if self._right_title == self._last_title
  129. let sep_change = s:get_separator_change_with_end(group, right_group, outer_right_group, self.overflow_group, sep_size, alt_sep_size) - overflow_marker_size
  130. else
  131. let sep_change = s:get_separator_change(group, right_group, self.overflow_group, sep_size, alt_sep_size)
  132. endif
  133. let right_group = group
  134. let self._right_title +=
  135. \ self.try_insert_title(self._right_title, group, self._right_position, sep_change, 1)
  136. endif
  137. while self._remaining_space > 0
  138. let done = 0
  139. if self._left_title >= self._first_title
  140. " Insert next title to the left
  141. let group = self.get_group(self._left_title)
  142. if self._left_title == self._first_title
  143. let sep_change = s:get_separator_change_with_end(group, left_group, outer_left_group, self.overflow_group, sep_size, alt_sep_size) - overflow_marker_size
  144. else
  145. let sep_change = s:get_separator_change(group, left_group, self.overflow_group, sep_size, alt_sep_size)
  146. endif
  147. let left_group = group
  148. let done = self.try_insert_title(self._left_title, group, self._left_position, sep_change, 0)
  149. let self._left_title -= done
  150. endif
  151. " If center_active is set, this |if| operates as an independent |if|,
  152. " otherwise as an |elif|.
  153. if self._right_title <= self._last_title && (center_active || !done)
  154. " Insert next title to the right
  155. let group = self.get_group(self._right_title)
  156. if self._right_title == self._last_title
  157. let sep_change = s:get_separator_change_with_end(group, right_group, outer_right_group, self.overflow_group, sep_size, alt_sep_size) - overflow_marker_size
  158. else
  159. let sep_change = s:get_separator_change(group, right_group, self.overflow_group, sep_size, alt_sep_size)
  160. endif
  161. let right_group = group
  162. let done = self.try_insert_title(self._right_title, group, self._right_position, sep_change, 0)
  163. let self._right_title += done
  164. endif
  165. if !done
  166. break
  167. endif
  168. endwhile
  169. if self._left_title >= self._first_title
  170. if get(g:, 'airline#extensions#tabline#current_first', 0)
  171. let self._left_position -= 1
  172. endif
  173. call self.insert_section(self.overflow_group, overflow_marker, self._left_position)
  174. let self._right_position += 1
  175. endif
  176. if self._right_title <= self._last_title
  177. call self.insert_section(self.overflow_group, overflow_marker, self._right_position)
  178. endif
  179. endif
  180. return self._build()
  181. endfunction
  182. let s:prototype.overflow_group = 'airline_tab'
  183. " Extract the text content a tabline will render. (Incomplete).
  184. "
  185. " See :help 'statusline' for the list of fields.
  186. function! s:evaluate_tabline(tabline)
  187. let tabline = a:tabline
  188. let tabline = substitute(tabline, '%{\([^}]\+\)}', '\=eval(submatch(1))', 'g')
  189. let tabline = substitute(tabline, '%#[^#]\+#', '', 'g')
  190. let tabline = substitute(tabline, '%(\([^)]\+\)%)', '\1', 'g')
  191. let tabline = substitute(tabline, '%\d\+[TX]', '', 'g')
  192. let tabline = substitute(tabline, '%=', '', 'g')
  193. let tabline = substitute(tabline, '%\d*\*', '', 'g')
  194. if has('tablineat')
  195. let tabline = substitute(tabline, '%@[^@]\+@', '', 'g')
  196. endif
  197. return tabline
  198. endfunction
  199. function! s:tabline_evaluated_length(tabline)
  200. return airline#util#strchars(s:evaluate_tabline(a:tabline))
  201. endfunction
  202. function! airline#extensions#tabline#builder#new(context)
  203. let builder = airline#builder#new(a:context)
  204. let builder._build = builder.build
  205. call extend(builder, s:prototype, 'force')
  206. return builder
  207. endfunction