Auto-list composing

Automatically create lists like "- " or "12. " when press Enter
This commit is contained in:
Lim Chee Aun 2023-09-16 22:57:35 +08:00
parent 1a714d214b
commit 887503e40b

View file

@ -1175,6 +1175,17 @@ function Compose({
); );
} }
function autoResizeTextarea(textarea) {
if (!textarea) return;
const { value, offsetHeight, scrollHeight, clientHeight } = textarea;
if (offsetHeight < window.innerHeight) {
// NOTE: This check is needed because the offsetHeight return 50000 (really large number) on first render
// No idea why it does that, will re-investigate in far future
const offset = offsetHeight - clientHeight;
textarea.style.height = value ? scrollHeight + offset + 'px' : null;
}
}
const Textarea = forwardRef((props, ref) => { const Textarea = forwardRef((props, ref) => {
const { masto } = api(); const { masto } = api();
const [text, setText] = useState(ref.current?.value || ''); const [text, setText] = useState(ref.current?.value || '');
@ -1367,15 +1378,46 @@ const Textarea = forwardRef((props, ref) => {
ref={ref} ref={ref}
name="status" name="status"
value={text} value={text}
onInput={(e) => { onKeyDown={(e) => {
const { scrollHeight, offsetHeight, clientHeight, value } = e.target; // Get line before cursor position after pressing 'Enter'
setText(value); const { key, target } = e;
if (offsetHeight < window.innerHeight) { if (key === 'Enter') {
// NOTE: This check is needed because the offsetHeight return 50000 (really large number) on first render try {
// No idea why it does that, will re-investigate in far future const { value, selectionStart } = target;
const offset = offsetHeight - clientHeight; const textBeforeCursor = value.slice(0, selectionStart);
e.target.style.height = value ? scrollHeight + offset + 'px' : null; const lastLine = textBeforeCursor.split('\n').slice(-1)[0];
if (lastLine) {
// If line starts with "- " or "12. "
if (/^\s*(-|\d+\.)\s/.test(lastLine)) {
// insert "- " at cursor position
const [_, preSpaces, bullet, postSpaces, anything] =
lastLine.match(/^(\s*)(-|\d+\.)(\s+)(.+)?/) || [];
if (anything) {
e.preventDefault();
const [number] = bullet.match(/\d+/) || [];
const newBullet = number ? `${+number + 1}.` : '-';
const text = `\n${preSpaces}${newBullet}${postSpaces}`;
target.setRangeText(text, selectionStart, selectionStart);
const pos = selectionStart + text.length;
target.setSelectionRange(pos, pos);
} else {
// trim the line before the cursor, then insert new line
const pos = selectionStart - lastLine.length;
target.setRangeText('', pos, selectionStart);
}
autoResizeTextarea(target);
}
}
} catch (e) {
// silent fail
console.error(e);
}
} }
}}
onInput={(e) => {
const { target } = e;
setText(target.value);
autoResizeTextarea(target);
props.onInput?.(e); props.onInput?.(e);
}} }}
style={{ style={{