Skip to content

Instantly share code, notes, and snippets.

@gondar00
Created March 25, 2020 22:11
Show Gist options
  • Save gondar00/0f10d6e9fee26071cf0d8503d596cb3d to your computer and use it in GitHub Desktop.
Save gondar00/0f10d6e9fee26071cf0d8503d596cb3d to your computer and use it in GitHub Desktop.
preserve-cursor-position-react-hooks
import * as React from "react";
import ReactDOM from "react-dom";
import "./styles.css";
const strip = value => value.replace(/[^a-zA-Z\s]/g, "");
function filterOut(text, cursor) {
const beforeCursor = text.slice(0, cursor);
const afterCursor = text.slice(cursor, text.length);
const filterdBeforeCursor = strip(beforeCursor);
const filterAfterCursor = strip(afterCursor);
const newText = filterdBeforeCursor + filterAfterCursor;
const newCursor = filterdBeforeCursor.length;
return [newText, newCursor];
}
function App() {
const [name, setName] = React.useState("");
const runAfterUpdate = useRunAfterUpdate();
const handleNameChange = evt => {
const input = evt.target;
const text = input.value;
const cursor = input.selectionStart;
const [newName, newCursor] = filterOut(text, cursor);
setName(newName);
runAfterUpdate(() => {
input.selectionStart = newCursor;
input.selectionEnd = newCursor;
});
};
return (
<div>
<input
placeholder="Name"
value={name}
onChange={handleNameChange}
/>
</div>
);
}
const rootElement = document.getElementById("root");
ReactDOM.render(<App />, rootElement);
function useRunAfterUpdate() {
const afterPaintRef = React.useRef(null);
React.useLayoutEffect(() => {
if (afterPaintRef.current) {
afterPaintRef.current();
afterPaintRef.current = null;
}
});
const runAfterUpdate = fn => (afterPaintRef.current = fn);
return runAfterUpdate;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment