Created
January 10, 2023 18:38
-
-
Save segunadebayo/2c195421ffe599b416c700be7e785977 to your computer and use it in GitHub Desktop.
Restore Cursor
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import { useRef } from "react"; | |
import warning from "rc-util/lib/warning"; | |
/** | |
* Keep input cursor in the correct position if possible. | |
* Is this necessary since we have `formatter` which may mass the content? | |
*/ | |
export default function useCursor( | |
input: HTMLInputElement, | |
focused: boolean | |
): [() => void, () => void] { | |
const selectionRef = useRef<{ | |
start?: number; | |
end?: number; | |
value?: string; | |
beforeTxt?: string; | |
afterTxt?: string; | |
}>(null); | |
function recordCursor() { | |
// Record position | |
try { | |
const { selectionStart: start, selectionEnd: end, value } = input; | |
const beforeTxt = value.substring(0, start); | |
const afterTxt = value.substring(end); | |
selectionRef.current = { | |
start, | |
end, | |
value, | |
beforeTxt, | |
afterTxt, | |
}; | |
} catch (e) { | |
// Fix error in Chrome: | |
// Failed to read the 'selectionStart' property from 'HTMLInputElement' | |
// http://stackoverflow.com/q/21177489/3040605 | |
} | |
} | |
/** | |
* Restore logic: | |
* 1. back string same | |
* 2. start string same | |
*/ | |
function restoreCursor() { | |
if (input && selectionRef.current && focused) { | |
try { | |
const { value } = input; | |
const { beforeTxt, afterTxt, start } = selectionRef.current; | |
let startPos = value.length; | |
if (value.endsWith(afterTxt)) { | |
startPos = value.length - selectionRef.current.afterTxt.length; | |
} else if (value.startsWith(beforeTxt)) { | |
startPos = beforeTxt.length; | |
} else { | |
const beforeLastChar = beforeTxt[start - 1]; | |
const newIndex = value.indexOf(beforeLastChar, start - 1); | |
if (newIndex !== -1) { | |
startPos = newIndex + 1; | |
} | |
} | |
input.setSelectionRange(startPos, startPos); | |
} catch (e) { | |
warning( | |
false, | |
`Something warning of cursor restore. Please fire issue about this: ${e.message}` | |
); | |
} | |
} | |
} | |
return [recordCursor, restoreCursor]; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment