in cursor.sml's 'helpTillPrevChr' function, properly handle edge case where graphical chr is followed by multiple line breaks
This commit is contained in:
@@ -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
|
||||||
|
|||||||
@@ -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 *)
|
||||||
|
|||||||
Reference in New Issue
Block a user