Skip to content

Commit 60ca800

Browse files
tompngmatzbot
authored andcommitted
[ruby/reline] Fix split_by_width to retain color sequences
(ruby/reline#490) * Fix split_by_width to retain color sequences * Add OSC escape sequence testcase of Reline::Unicode.split_by_width
1 parent 9bc2dbd commit 60ca800

File tree

2 files changed

+15
-1
lines changed

2 files changed

+15
-1
lines changed

lib/reline/unicode.rb

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -160,24 +160,29 @@ def self.split_by_width(str, max_width, encoding = str.encoding)
160160
width = 0
161161
rest = str.encode(Encoding::UTF_8)
162162
in_zero_width = false
163+
seq = String.new(encoding: encoding)
163164
rest.scan(WIDTH_SCANNER) do |gc|
164165
case
165166
when gc[NON_PRINTING_START_INDEX]
166167
in_zero_width = true
168+
lines.last << NON_PRINTING_START
167169
when gc[NON_PRINTING_END_INDEX]
168170
in_zero_width = false
171+
lines.last << NON_PRINTING_END
169172
when gc[CSI_REGEXP_INDEX]
170173
lines.last << gc[CSI_REGEXP_INDEX]
174+
seq << gc[CSI_REGEXP_INDEX]
171175
when gc[OSC_REGEXP_INDEX]
172176
lines.last << gc[OSC_REGEXP_INDEX]
177+
seq << gc[OSC_REGEXP_INDEX]
173178
when gc[GRAPHEME_CLUSTER_INDEX]
174179
gc = gc[GRAPHEME_CLUSTER_INDEX]
175180
unless in_zero_width
176181
mbchar_width = get_mbchar_width(gc)
177182
if (width += mbchar_width) > max_width
178183
width = mbchar_width
179184
lines << nil
180-
lines << String.new(encoding: encoding)
185+
lines << seq.dup
181186
height += 1
182187
end
183188
end

test/reline/test_unicode.rb

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,15 @@ def test_ambiguous_width
1818
assert_equal 2, Reline::Unicode.calculate_width('√', true)
1919
end
2020

21+
def test_split_by_width
22+
assert_equal [['abc', nil, 'de'], 2], Reline::Unicode.split_by_width('abcde', 3)
23+
assert_equal [['abc', nil, 'def', nil, ''], 3], Reline::Unicode.split_by_width('abcdef', 3)
24+
assert_equal [['ab', nil, 'あd', nil, 'ef'], 3], Reline::Unicode.split_by_width('abあdef', 3)
25+
assert_equal [["ab\1zero\2c", nil, 'def', nil, ''], 3], Reline::Unicode.split_by_width("ab\1zero\2cdef", 3)
26+
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)
27+
assert_equal [["ab\e]0;1\ac", nil, "\e]0;1\ad"], 2], Reline::Unicode.split_by_width("ab\e]0;1\acd", 3)
28+
end
29+
2130
def test_take_range
2231
assert_equal 'cdef', Reline::Unicode.take_range('abcdefghi', 2, 4)
2332
assert_equal 'いう', Reline::Unicode.take_range('あいうえお', 2, 4)

0 commit comments

Comments
 (0)