in cursor.sml's 'helpTillPrevChr' function, properly handle edge case where graphical chr is followed by multiple line breaks

This commit is contained in:
2024-11-09 06:33:32 +00:00
parent d4d0236dab
commit faff848cb4
3 changed files with 84 additions and 38 deletions

View File

@@ -426,13 +426,13 @@ struct
buildTextAndClear (app, buffer, low) buildTextAndClear (app, buffer, low)
end end
fun helpDeleteToChr (app: app_type, buffer, cursorIdx, count, fMove, chr) = fun helpDeleteToChr (app: app_type, buffer, cursorIdx, count, fMove, fInc, chr) =
if count = 0 then if count = 0 then
buildTextAndClearAfterChr (app, buffer, cursorIdx) buildTextAndClearAfterChr (app, buffer, cursorIdx)
else else
let let
val buffer = LineGap.goToIdx (cursorIdx, buffer) val buffer = LineGap.goToIdx (cursorIdx, buffer)
val otherIdx = fMove (buffer, cursorIdx, chr) + 1 val otherIdx = fInc (fMove (buffer, cursorIdx, chr), 1)
val low = Int.min (cursorIdx, otherIdx) val low = Int.min (cursorIdx, otherIdx)
val high = Int.max (cursorIdx, otherIdx) val high = Int.max (cursorIdx, otherIdx)
@@ -440,11 +440,11 @@ struct
val buffer = LineGap.delete (low, length, buffer) val buffer = LineGap.delete (low, length, buffer)
in in
helpDeleteToChr (app, buffer, low, count - 1, fMove, chr) helpDeleteToChr (app, buffer, low, count - 1, fMove, fInc, chr)
end end
fun deleteToChr (app: app_type, count, fMove, chr) = fun deleteToChr (app: app_type, count, fMove, fInc, chr) =
helpDeleteToChr (app, #buffer app, #cursorIdx app, count, fMove, chr) helpDeleteToChr (app, #buffer app, #cursorIdx app, count, fMove, fInc, chr)
(* command-parsing functions *) (* command-parsing functions *)
(** number of characters which are integers *) (** number of characters which are integers *)
@@ -573,17 +573,23 @@ struct
(* have to continue parsing string *) (* have to continue parsing string *)
case String.sub (str, strPos + 1) of case String.sub (str, strPos + 1) of
#"t" => #"t" =>
(* todo: delete till chr, forwards *) (* delete till chr, forwards *)
(case newCmd of (case newCmd of
CHAR_EVENT chr => CHAR_EVENT chr =>
deleteToChr (app, 1, Cursor.tillNextChr, chr) deleteToChr (app, 1, Cursor.tillNextChr, op+, chr)
| KEY_ESC => | KEY_ESC =>
clearMode app clearMode app
| RESIZE_EVENT (width, height) => | RESIZE_EVENT (width, height) =>
resizeText (app, width, height)) resizeText (app, width, height))
| #"T" => | #"T" =>
(* todo: delete till chr, backwards *) (* delete till chr, backwards *)
clearMode app (case newCmd of
CHAR_EVENT chr =>
deleteToChr (app, 1, Cursor.tillPrevChr, op-, chr)
| KEY_ESC =>
clearMode app
| RESIZE_EVENT (width, height) =>
resizeText (app, width, height))
| #"d" => | #"d" =>
(* todo: delete whole line *) (* todo: delete whole line *)
clearMode app clearMode app

View File

@@ -1384,48 +1384,88 @@ struct
(* tl is empty; return 0 for lineGap start *) (* tl is empty; return 0 for lineGap start *)
0 0
fun helpTillPrevChr (strPos, str, absIdx, stl, ltl, origIdx, findChr, lastNonLine) = fun helpTillPrevChr
if strPos < 0 then ( strPos, str, absIdx, stl, ltl
case (stl, ltl) of , origIdx, findChr
(shd :: stl, lhd :: ltl) => , lastNonLine, lastLine, lastValid
helpTillPrevChr ) =
( String.size shd - 1, shd, absIdx, stl, ltl if strPos < 0 then
, origIdx, findChr, lastNonLine case (stl, ltl) of
) (shd :: stl, lhd :: ltl) =>
| (_, _) => helpTillPrevChr
origIdx ( String.size shd - 1, shd, absIdx, stl, ltl
else , origIdx, findChr
let , lastNonLine, lastLine, lastValid
val chr = String.sub (str, strPos) )
in | (_, _) =>
if chr = findChr then origIdx
lastNonLine else
else let
let val chr = String.sub (str, strPos)
val lastNonLine = in
if chr = #"\n" orelse chr = #"\r" then if chr = findChr then
if lastLine = lastNonLine then
lastNonLine
else if absIdx + 1 = lastLine then
lastValid + 1
else
Int.min (lastLine, lastNonLine)
else
let
val lastLine =
if chr = #"\n" then
absIdx
else
lastLine
val lastNonLine =
if chr = #"\n" then
lastNonLine lastNonLine
else else
absIdx absIdx
in
helpTillPrevChr (* There is a slightly tricky edge case
( strPos - 1, str, absIdx - 1 * which is the reason the lastValid variable.
, stl, ltl, origIdx, findChr, lastNonLine * Say we have a string "a\n\n\nbcd"
) * and we type "Ta" with the cursor at the end.
end * We want the cursor to go to the second line break
end * because (graphical-chr -> \n) should not be selectable.
* However, with only lastLine and lastNonLine variables,
* we only have information about the most recent \n
* and the most recent graphical-chr.
* This means we don't have information about the case
* where a graphical-chr is followed by multiple '\n's.
* The lastValid variable keeps track of this information
* so we can use it to provide the expected behaviour.
* *)
val lastValid =
if lastLine = lastNonLine + 1 then
lastLine + 1
else
Int.min (lastLine, lastNonLine)
in
helpTillPrevChr
( strPos - 1, str, absIdx - 1
, stl, ltl, origIdx, findChr
, lastNonLine, lastLine, lastValid
)
end
end
fun startTillPrevChr (shd, strIdx, absIdx, stl, ltl, findChr) = fun startTillPrevChr (shd, strIdx, absIdx, stl, ltl, findChr) =
(* we want to start iterating from Prev char after strIdx *) (* we want to start iterating from Prev char after strIdx *)
if strIdx > 0 then if strIdx > 0 then
helpTillPrevChr helpTillPrevChr
(strIdx - 1, shd, absIdx - 1, stl, ltl, absIdx, findChr, absIdx) ( strIdx - 1, shd, absIdx - 1
, stl, ltl, absIdx, findChr
, absIdx, absIdx, absIdx
)
else else
case (stl, ltl) of case (stl, ltl) of
(stlhd :: stltl, ltlhd :: ltltl) => (stlhd :: stltl, ltlhd :: ltltl) =>
helpTillPrevChr helpTillPrevChr
( String.size stlhd - 1, stlhd, absIdx - 1 ( String.size stlhd - 1, stlhd, absIdx - 1
, stltl, ltltl, absIdx, findChr, absIdx , stltl, ltltl, absIdx, findChr
, absIdx, absIdx, absIdx
) )
| (_, _) => | (_, _) =>
(* tl is empty; return 0 for lineGap start *) (* tl is empty; return 0 for lineGap start *)

BIN
shf

Binary file not shown.