diff --git a/src/nvim/api/buffer.c b/src/nvim/api/buffer.c index 9866f1140e..39330690e8 100644 --- a/src/nvim/api/buffer.c +++ b/src/nvim/api/buffer.c @@ -493,11 +493,10 @@ end: /// Return the byte offset for a line. // -/// This includes the end-of-line character, depending on the 'fileformat' -/// option for the current buffer. The first line returns 0. UTF-8 encoding is -/// used, 'fileencoding' is ignored. Sending in the index just below the last -/// line gives the total byte count for the entire file. A final newline is -/// included if it would be written, see 'eol'. +/// The first line returns 0. UTF-8 bytes are counted, and EOL counts as one +/// byte. 'fileformat' and 'fileencoding' are ignored. Sending in the index +/// just below the last line gives the total byte count for the entire buffer. +/// A final EOL is included if it would be written, see 'eol'. /// /// Unlike |line2byte()|, throws error for out-of-bounds indexing. /// Returns -1 for unloaded buffer. @@ -506,7 +505,7 @@ end: /// @param index Line index /// @param[out] err Error details, if any /// @return Integer Byte offset -Integer nvim_buf_get_offset_for_line(Buffer buffer, Integer index, Error *err) +Integer nvim_buf_get_offset(Buffer buffer, Integer index, Error *err) FUNC_API_SINCE(5) { buf_T *buf = find_buffer_by_handle(buffer, err); @@ -524,7 +523,7 @@ Integer nvim_buf_get_offset_for_line(Buffer buffer, Integer index, Error *err) return 0; } - return ml_find_line_or_offset(buf, (int)index+1, NULL); + return ml_find_line_or_offset(buf, (int)index+1, NULL, true); } /// Gets a buffer-scoped (b:) variable. diff --git a/src/nvim/buffer.c b/src/nvim/buffer.c index ce6aa69239..cd77f48320 100644 --- a/src/nvim/buffer.c +++ b/src/nvim/buffer.c @@ -3734,7 +3734,8 @@ int build_stl_str_hl( FALLTHROUGH; case STL_OFFSET: { - long l = ml_find_line_or_offset(wp->w_buffer, wp->w_cursor.lnum, NULL); + long l = ml_find_line_or_offset(wp->w_buffer, wp->w_cursor.lnum, NULL, + false); num = (wp->w_buffer->b_ml.ml_flags & ML_EMPTY) || l < 0 ? 0L : l + 1 + (!(State & INSERT) && empty_line ? 0 : (int)wp->w_cursor.col); diff --git a/src/nvim/eval.c b/src/nvim/eval.c index 052e0e3f35..4e0e3f6f1f 100644 --- a/src/nvim/eval.c +++ b/src/nvim/eval.c @@ -7232,7 +7232,7 @@ static void f_byte2line(typval_T *argvars, typval_T *rettv, FunPtr fptr) rettv->vval.v_number = -1; } else { rettv->vval.v_number = (varnumber_T)ml_find_line_or_offset(curbuf, 0, - &boff); + &boff, false); } } @@ -12084,7 +12084,7 @@ static void f_line2byte(typval_T *argvars, typval_T *rettv, FunPtr fptr) if (lnum < 1 || lnum > curbuf->b_ml.ml_line_count + 1) { rettv->vval.v_number = -1; } else { - rettv->vval.v_number = ml_find_line_or_offset(curbuf, lnum, NULL); + rettv->vval.v_number = ml_find_line_or_offset(curbuf, lnum, NULL, false); } if (rettv->vval.v_number >= 0) { rettv->vval.v_number++; diff --git a/src/nvim/memline.c b/src/nvim/memline.c index aad1750c85..95f3b0c623 100644 --- a/src/nvim/memline.c +++ b/src/nvim/memline.c @@ -3850,13 +3850,17 @@ static void ml_updatechunk(buf_T *buf, linenr_T line, long len, int updtype) ml_upd_lastcurix = curix; } -/* - * Find offset for line or line with offset. - * Find line with offset if "lnum" is 0; return remaining offset in offp - * Find offset of line if "lnum" > 0 - * return -1 if information is not available - */ -long ml_find_line_or_offset(buf_T *buf, linenr_T lnum, long *offp) +/// Find offset for line or line with offset. +/// +/// @param buf buffer to use +/// @param lnum if > 0, find offset of lnum, store offset in offp +/// if == 0, return line with offset *offp +/// @param offp Location where offset of line is stored, or to read offset to +/// use to find line. In the later case, store remaining offset. +/// @param no_ff ignore 'fileformat' option, always use one byte for NL. +/// +/// @return -1 if information is not available +long ml_find_line_or_offset(buf_T *buf, linenr_T lnum, long *offp, bool no_ff) { linenr_T curline; int curix; @@ -3869,7 +3873,7 @@ long ml_find_line_or_offset(buf_T *buf, linenr_T lnum, long *offp) int text_end; long offset; int len; - int ffdos = (get_fileformat(buf) == EOL_DOS); + int ffdos = !no_ff && (get_fileformat(buf) == EOL_DOS); int extra = 0; /* take care of cached line first */ @@ -3983,7 +3987,7 @@ void goto_byte(long cnt) if (boff) { boff--; } - lnum = ml_find_line_or_offset(curbuf, (linenr_T)0, &boff); + lnum = ml_find_line_or_offset(curbuf, (linenr_T)0, &boff, false); if (lnum < 1) { // past the end curwin->w_cursor.lnum = curbuf->b_ml.ml_line_count; curwin->w_curswant = MAXCOL; diff --git a/test/functional/api/buffer_spec.lua b/test/functional/api/buffer_spec.lua index d3739c74d2..d9412f0f13 100644 --- a/test/functional/api/buffer_spec.lua +++ b/test/functional/api/buffer_spec.lua @@ -301,37 +301,40 @@ describe('api/buf', function() end) end) - describe('get_offset_for_line', function() - local get_offset_for_line = curbufmeths.get_offset_for_line + describe('get_offset', function() + local get_offset = curbufmeths.get_offset it('works', function() - curbufmeths.set_lines(0,-1,true,{'Some','exa\000mple', '', 'text'}) - eq(4, curbufmeths.line_count()) - eq(0, get_offset_for_line(0)) - eq(5, get_offset_for_line(1)) - eq(14, get_offset_for_line(2)) - eq(15, get_offset_for_line(3)) - eq(20, get_offset_for_line(4)) - eq({false,'Index out of bounds'}, meth_pcall(get_offset_for_line, 5)) - eq({false,'Index out of bounds'}, meth_pcall(get_offset_for_line, -1)) + curbufmeths.set_lines(0,-1,true,{'Some\r','exa\000mple', '', 'buf\rfer', 'text'}) + eq(5, curbufmeths.line_count()) + eq(0, get_offset(0)) + eq(6, get_offset(1)) + eq(15, get_offset(2)) + eq(16, get_offset(3)) + eq(24, get_offset(4)) + eq(29, get_offset(5)) + eq({false,'Index out of bounds'}, meth_pcall(get_offset, 6)) + eq({false,'Index out of bounds'}, meth_pcall(get_offset, -1)) curbufmeths.set_option('eol', false) curbufmeths.set_option('fixeol', false) - eq(19, get_offset_for_line(4)) + eq(28, get_offset(5)) + -- fileformat is ignored curbufmeths.set_option('fileformat', 'dos') - eq(0, get_offset_for_line(0)) - eq(6, get_offset_for_line(1)) - eq(16, get_offset_for_line(2)) - eq(18, get_offset_for_line(3)) - eq(22, get_offset_for_line(4)) + eq(0, get_offset(0)) + eq(6, get_offset(1)) + eq(15, get_offset(2)) + eq(16, get_offset(3)) + eq(24, get_offset(4)) + eq(28, get_offset(5)) curbufmeths.set_option('eol', true) - eq(24, get_offset_for_line(4)) + eq(29, get_offset(5)) command("set hidden") command("enew") - eq(6, bufmeths.get_offset_for_line(1,1)) + eq(6, bufmeths.get_offset(1,1)) command("bunload! 1") - eq(-1, bufmeths.get_offset_for_line(1,1)) + eq(-1, bufmeths.get_offset(1,1)) end) end)