Improve display

- CJK wide character support
- Progress reporting for long-running queries (> 0.5sec)
This commit is contained in:
Junegunn Choi 2013-11-02 20:12:39 +09:00
parent 6a9970c98e
commit 1629fe079a

69
fzf
View File

@ -105,7 +105,7 @@ when /darwin/
end end
def self.nfc str, offset def self.nfc str, offset
ret = '' ret = ''
omap = [] omap = []
pend = [] pend = []
str.split(//).each_with_index do |c, idx| str.split(//).each_with_index do |c, idx|
@ -198,7 +198,7 @@ def print_info progress = true, msg = nil
end end
def refresh def refresh
C.setpos cursor_y, 2 + ulen(@query[0, @cursor_x]) C.setpos cursor_y, 2 + width(@query[0, @cursor_x])
C.refresh C.refresh
end end
@ -207,12 +207,12 @@ def ctrl char
end end
if RUBY_VERSION.split('.').map { |e| e.rjust(3, '0') }.join > '001009' if RUBY_VERSION.split('.').map { |e| e.rjust(3, '0') }.join > '001009'
def ulen str def width str
@urx ||= Regexp.new '\p{Han}|\p{Katakana}|\p{Hiragana}|\p{Hangul}' @urx ||= Regexp.new '\p{Han}|\p{Katakana}|\p{Hiragana}|\p{Hangul}'
str.gsub(@urx, ' ').length str.gsub(@urx, ' ').length
end end
else else
def ulen str def width str
str.length str.length
end end
@ -334,6 +334,8 @@ searcher = Thread.new {
new_search = events[:key] || events[:new] new_search = events[:key] || events[:new]
user_input = events[:key] || events[:vcursor] user_input = events[:key] || events[:vcursor]
progress = 0
started_at = Time.now
if new_search && !@lists.empty? if new_search && !@lists.empty?
events.delete :new events.delete :new
@ -346,19 +348,26 @@ searcher = Thread.new {
matches = fcache[q] ||= matches = fcache[q] ||=
begin begin
@smtx.synchronize do
print_info true, ' ..'
print_input
refresh
end unless q.empty?
found = [] found = []
skip = false skip = false
cnt = 0
@lists.each do |pair| @lists.each do |pair|
@mtx.synchronize { skip = @events[:key] } list, cache = pair
cnt += list.length
@mtx.synchronize {
skip = @events[:key]
progress = (100 * cnt / @count)
}
break if skip break if skip
list, cache = pair if !q.empty? && progress < 100 && Time.now - started_at > 0.5
@smtx.synchronize do
print_info true, " (#{progress}%)"
refresh
end
end
found.concat(cache[q] ||= begin found.concat(cache[q] ||= begin
prefix, suffix = @query[0, @cursor_x], @query[@cursor_x..-1] || '' prefix, suffix = @query[0, @cursor_x], @query[@cursor_x..-1] || ''
prefix_cache = suffix_cache = nil prefix_cache = suffix_cache = nil
@ -397,7 +406,7 @@ searcher = Thread.new {
end#new_search end#new_search
# This small delay reduces the number of partial lists # This small delay reduces the number of partial lists
sleep [20, delay += 5].min * 0.01 unless user_input sleep((delay = [20, delay + 5].min) * 0.01) unless user_input
if events.delete(:vcursor) || new_search if events.delete(:vcursor) || new_search
@mtx.synchronize do @mtx.synchronize do
@ -417,7 +426,7 @@ searcher = Thread.new {
end end
end end
maxc = C.cols - 5 maxc = C.cols - 3
matches[0, max_items].each_with_index do |item, idx| matches[0, max_items].each_with_index do |item, idx|
next if !new_search && !((vcursor-1)..(vcursor+1)).include?(idx) next if !new_search && !((vcursor-1)..(vcursor+1)).include?(idx)
@ -426,19 +435,28 @@ searcher = Thread.new {
chosen = idx == vcursor chosen = idx == vcursor
b, e = offset b, e = offset
if line.length > maxc # Overflow
diff = e - (maxc - 2) if width(line) > maxc
if diff > 2 ewidth = width(line[0...e])
line = '..' + line[diff..-1] # Stri..
b -= diff - 2 if ewidth <= maxc
b = [2, b].max line = line[0...-1] while width(line) > maxc - 2
line << '..'
# ..ring
else else
line = line[0, maxc] + '..' # ..ri..
line = line[0...e] + '..' if ewidth < width(line) - 2
while width(line) > maxc - 2
b -= 1
e -= 1
line = line[1..-1]
end
b += 2
e += 2
b = [2, b].max
line = '..' + line
end end
end end
if line.length > maxc
line = line[0, maxc] + '..'
end
C.setpos row, 0 C.setpos row, 0
C.clrtoeol C.clrtoeol
@ -447,8 +465,7 @@ searcher = Thread.new {
C.attron color(:chosen, true) if chosen C.attron color(:chosen, true) if chosen
e = [e, maxc].min if b < e
if b < maxc && b < e
C.addstr line[0, b] C.addstr line[0, b]
cprint line[b...e], color(chosen ? :match! : :match, chosen) cprint line[b...e], color(chosen ? :match! : :match, chosen)
C.attron color(:chosen, true) if chosen C.attron color(:chosen, true) if chosen