Skip to content

Commit e364416

Browse files
authored
nonprinting_start and nonprinting_end should be removed (#771)
1 parent 01872fd commit e364416

File tree

3 files changed

+26
-14
lines changed

3 files changed

+26
-14
lines changed

lib/reline/line_editor.rb

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -464,7 +464,7 @@ def update_dialogs(key = nil)
464464
def render_finished
465465
render_differential([], 0, 0)
466466
lines = @buffer_of_lines.size.times.map do |i|
467-
line = prompt_list[i] + modified_lines[i]
467+
line = Reline::Unicode.strip_non_printing_start_end(prompt_list[i]) + modified_lines[i]
468468
wrapped_lines, = split_by_width(line, screen_width)
469469
wrapped_lines.last.empty? ? "#{line} " : line
470470
end
@@ -473,7 +473,7 @@ def render_finished
473473

474474
def print_nomultiline_prompt
475475
# Readline's test `TestRelineAsReadline#test_readline` requires first output to be prompt, not cursor reset escape sequence.
476-
@output.write @prompt if @prompt && !@is_multiline
476+
@output.write Reline::Unicode.strip_non_printing_start_end(@prompt) if @prompt && !@is_multiline
477477
end
478478

479479
def render

lib/reline/unicode.rb

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -132,10 +132,8 @@ def self.split_by_width(str, max_width, encoding = str.encoding, offset: 0)
132132
case
133133
when non_printing_start
134134
in_zero_width = true
135-
lines.last << NON_PRINTING_START
136135
when non_printing_end
137136
in_zero_width = false
138-
lines.last << NON_PRINTING_END
139137
when csi
140138
lines.last << csi
141139
unless in_zero_width
@@ -147,7 +145,7 @@ def self.split_by_width(str, max_width, encoding = str.encoding, offset: 0)
147145
end
148146
when osc
149147
lines.last << osc
150-
seq << osc
148+
seq << osc unless in_zero_width
151149
when gc
152150
unless in_zero_width
153151
mbchar_width = get_mbchar_width(gc)
@@ -170,6 +168,10 @@ def self.split_by_width(str, max_width, encoding = str.encoding, offset: 0)
170168
[lines, height]
171169
end
172170

171+
def self.strip_non_printing_start_end(prompt)
172+
prompt.gsub(/\x01([^\x02]*)(?:\x02|\z)/) { $1 }
173+
end
174+
173175
# Take a chunk of a String cut by width with escape sequences.
174176
def self.take_range(str, start_col, max_width)
175177
take_mbchar_range(str, start_col, max_width).first
@@ -189,10 +191,8 @@ def self.take_mbchar_range(str, start_col, width, cover_begin: false, cover_end:
189191
case
190192
when non_printing_start
191193
in_zero_width = true
192-
chunk << NON_PRINTING_START
193194
when non_printing_end
194195
in_zero_width = false
195-
chunk << NON_PRINTING_END
196196
when csi
197197
has_csi = true
198198
chunk << csi

test/reline/test_unicode.rb

Lines changed: 19 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -33,33 +33,45 @@ def test_split_by_width
3333
assert_equal [['abc', nil, 'de'], 2], Reline::Unicode.split_by_width('abcde', 3)
3434
assert_equal [['abc', nil, 'def', nil, ''], 3], Reline::Unicode.split_by_width('abcdef', 3)
3535
assert_equal [['ab', nil, 'あd', nil, 'ef'], 3], Reline::Unicode.split_by_width('abあdef', 3)
36-
assert_equal [["ab\1zero\2c", nil, 'def', nil, ''], 3], Reline::Unicode.split_by_width("ab\1zero\2cdef", 3)
36+
assert_equal [['ab[zero]c', nil, 'def', nil, ''], 3], Reline::Unicode.split_by_width("ab\1[zero]\2cdef", 3)
3737
assert_equal [["\e[31mabc", nil, "\e[31md\e[42mef", nil, "\e[31m\e[42mg"], 3], Reline::Unicode.split_by_width("\e[31mabcd\e[42mefg", 3)
3838
assert_equal [["ab\e]0;1\ac", nil, "\e]0;1\ad"], 2], Reline::Unicode.split_by_width("ab\e]0;1\acd", 3)
3939
end
4040

4141
def test_split_by_width_csi_reset_sgr_optimization
4242
assert_equal [["\e[1ma\e[mb\e[2mc", nil, "\e[2md\e[0me\e[3mf", nil, "\e[3mg"], 3], Reline::Unicode.split_by_width("\e[1ma\e[mb\e[2mcd\e[0me\e[3mfg", 3)
43-
assert_equal [["\e[1ma\1\e[mzero\e[0m\2\e[2mb", nil, "\e[1m\e[2mc"], 2], Reline::Unicode.split_by_width("\e[1ma\1\e[mzero\e[0m\2\e[2mbc", 2)
43+
assert_equal [["\e[1ma\e[mzero\e[0m\e[2mb", nil, "\e[1m\e[2mc"], 2], Reline::Unicode.split_by_width("\e[1ma\1\e[mzero\e[0m\2\e[2mbc", 2)
4444
end
4545

4646
def test_take_range
4747
assert_equal 'cdef', Reline::Unicode.take_range('abcdefghi', 2, 4)
4848
assert_equal 'あde', Reline::Unicode.take_range('abあdef', 2, 4)
49-
assert_equal "\1zero\2cdef", Reline::Unicode.take_range("ab\1zero\2cdef", 2, 4)
50-
assert_equal "b\1zero\2cde", Reline::Unicode.take_range("ab\1zero\2cdef", 1, 4)
49+
assert_equal '[zero]cdef', Reline::Unicode.take_range("ab\1[zero]\2cdef", 2, 4)
50+
assert_equal 'b[zero]cde', Reline::Unicode.take_range("ab\1[zero]\2cdef", 1, 4)
5151
assert_equal "\e[31mcd\e[42mef", Reline::Unicode.take_range("\e[31mabcd\e[42mefg", 2, 4)
5252
assert_equal "\e]0;1\acd", Reline::Unicode.take_range("ab\e]0;1\acd", 2, 3)
5353
assert_equal 'いう', Reline::Unicode.take_range('あいうえお', 2, 4)
5454
end
5555

56+
def test_nonprinting_start_end
57+
# \1 and \2 should be removed
58+
assert_equal 'ab[zero]cd', Reline::Unicode.take_range("ab\1[zero]\2cdef", 0, 4)
59+
assert_equal [['ab[zero]cd', nil, 'ef'], 2], Reline::Unicode.split_by_width("ab\1[zero]\2cdef", 4)
60+
# CSI between \1 and \2 does not need to be applied to the sebsequent line
61+
assert_equal [["\e[31mab\e[32mcd", nil, "\e[31mef"], 2], Reline::Unicode.split_by_width("\e[31mab\1\e[32m\2cdef", 4)
62+
end
63+
64+
def test_strip_non_printing_start_end
65+
assert_equal "ab[zero]cd[ze\1ro]ef[zero]", Reline::Unicode.strip_non_printing_start_end("ab\1[zero]\2cd\1[ze\1ro]\2ef\1[zero]")
66+
end
67+
5668
def test_calculate_width
5769
assert_equal 9, Reline::Unicode.calculate_width('abcdefghi')
5870
assert_equal 9, Reline::Unicode.calculate_width('abcdefghi', true)
5971
assert_equal 7, Reline::Unicode.calculate_width('abあdef')
6072
assert_equal 7, Reline::Unicode.calculate_width('abあdef', true)
61-
assert_equal 14, Reline::Unicode.calculate_width("ab\1zero\2cdef")
62-
assert_equal 6, Reline::Unicode.calculate_width("ab\1zero\2cdef", true)
73+
assert_equal 16, Reline::Unicode.calculate_width("ab\1[zero]\2cdef")
74+
assert_equal 6, Reline::Unicode.calculate_width("ab\1[zero]\2cdef", true)
6375
assert_equal 19, Reline::Unicode.calculate_width("\e[31mabcd\e[42mefg")
6476
assert_equal 7, Reline::Unicode.calculate_width("\e[31mabcd\e[42mefg", true)
6577
assert_equal 12, Reline::Unicode.calculate_width("ab\e]0;1\acd")
@@ -86,7 +98,7 @@ def test_take_mbchar_range
8698
assert_equal [' うえお ', 3, 10], Reline::Unicode.take_mbchar_range('あいうえお', 3, 10, padding: true)
8799
assert_equal [" \e[41mうえお\e[0m ", 3, 10], Reline::Unicode.take_mbchar_range("あい\e[41mうえお", 3, 10, padding: true)
88100
assert_equal ["\e[41m \e[42mい\e[43m ", 1, 4], Reline::Unicode.take_mbchar_range("\e[41mあ\e[42mい\e[43mう", 1, 4, padding: true)
89-
assert_equal ["\e[31mc\1ABC\2d\e[0mef", 2, 4], Reline::Unicode.take_mbchar_range("\e[31mabc\1ABC\2d\e[0mefghi", 2, 4)
101+
assert_equal ["\e[31mc[ABC]d\e[0mef", 2, 4], Reline::Unicode.take_mbchar_range("\e[31mabc\1[ABC]\2d\e[0mefghi", 2, 4)
90102
assert_equal ["\e[41m \e[42mい\e[43m ", 1, 4], Reline::Unicode.take_mbchar_range("\e[41mあ\e[42mい\e[43mう", 1, 4, padding: true)
91103
end
92104

0 commit comments

Comments
 (0)