Skip to content

Commit

Permalink
format all the things (facebook#1513)
Browse files Browse the repository at this point in the history
  • Loading branch information
thegreatercurve authored and acywatson committed Apr 9, 2022
1 parent c2d6695 commit 622a063
Show file tree
Hide file tree
Showing 50 changed files with 6,269 additions and 6,169 deletions.
6 changes: 1 addition & 5 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -149,7 +149,6 @@ jobs:
path: test-results/
retention-days: 7


e2e-windows:
runs-on: windows-latest
strategy:
Expand Down Expand Up @@ -190,7 +189,6 @@ jobs:
path: test-results/
retention-days: 7


e2e-collab-mac:
runs-on: macos-latest
strategy:
Expand Down Expand Up @@ -227,7 +225,6 @@ jobs:
path: test-results/
retention-days: 7


e2e-collab-linux:
runs-on: ubuntu-latest
strategy:
Expand Down Expand Up @@ -268,7 +265,6 @@ jobs:
path: test-results/
retention-days: 7


e2e-collab-windows:
runs-on: windows-latest
strategy:
Expand All @@ -292,7 +288,7 @@ jobs:
C:\Users\runneradmin\AppData\Local\ms-playwright
key: ${{ runner.os }}-v4-${{ hashFiles('package-lock.json') }}
- name: Install dependencies
# if: steps.cache.outputs.cache-hit != 'true'
# if: steps.cache.outputs.cache-hit != 'true'
run: npm ci
- name: Download browsers
run: npx playwright install
Expand Down
3 changes: 2 additions & 1 deletion .lintstagedrc.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
module.exports = {
'*.(js|mjs|jsx)': ['flow focus-check', 'prettier --write', 'eslint --fix'],
'*': 'prettier --write',
'*.(js|mjs|jsx)': ['flow focus-check', 'eslint --fix'],
};
14 changes: 9 additions & 5 deletions docs/useHistory.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
# Lexical's useHistory adds support for Undo and Redo

Part of undo and redo requires supporting coalescing of certain dynamic operations such as continuous typing.
The below provides background information regarding the continuous typing undo coalescing feature.
The below provides background information regarding the continuous typing undo coalescing feature.

**Undo Coalescing of Typed Text**
**TL;DR**
Expand All @@ -11,39 +12,42 @@ As mentioned above, continuous typing should be coalesced into a single undo act
There are other types of edits that require undo coalescing such as rapidly resizing an image or rapidly changing the font size of selected text. These latter two examples fall under what might be termed “dynamic operations” and their implementation might vary quite a bit from coalescing of typing.
**What Constitutes Continuous Typing?**
For the purposes of undo coalescing, continuous typing events are safely limited to these three actions:

1. Forward typing characters (heads up, composed character typing discussed later).
2. Deleting backwards, e.g. backspace gestures including backspace key presses.
3. Forward deletion, e.g. forward delete gestures including Delete key presses.

Any interruption to the above breaks the notion of continuous typing. Note that switching between either of these three actions may itself be considered an interruption of continuous typing or it may not. This ends up being a usability decision. In my opinion, when examining various word processors, switching between these three actions should maintain continuity, because users often make typos or corrections in the flow of typing and therefore would lean towards the whole sequence of typing, backspacing and deleting to be within a single undoable session.
Any interruption to the above breaks the notion of continuous typing. Note that switching between either of these three actions may itself be considered an interruption of continuous typing or it may not. This ends up being a usability decision. In my opinion, when examining various word processors, switching between these three actions should maintain continuity, because users often make typos or corrections in the flow of typing and therefore would lean towards the whole sequence of typing, backspacing and deleting to be within a single undoable session.
Note that Google Docs and Quip treat a switch between typing, backspacing and deleting as disruptions to the continuous typing sessions.
**Start with Selected Text and Followup with Typing**
Let’s start with some selected text. The user may have selected the text directly, as a result of pasting text or through some other sequence of steps. When followed by continuous typing, the initial selection should be included along with the following typing events as part of a single coalesced undo session.
**Timeouts**
Text editing is generally a modeless user experience and so just about any event beyond the above typing actions will disrupt continuous typing. The first and foremost is a timeout.
Quip seems to require continuous typing to occur within 1 second intervals, whereas Google Docs requires about 2 seconds. In my opinion, the timeout may stretch to as long as 3 to 5 seconds. The more time between typing events, the more chance the user has to intentionally input a block of text, a thought or concept.
The use of a timeout is not necessarily required, and the amount of time is certainly up for debate. Collaborative co-editing also makes use of timeouts when grouping text that is merged across clients. Longer timeouts tend to make collaboration a smoother, less disruptive experience, because receiving other client input while expressing one’s thoughts tends to be unhelpful.
The use of a timeout is not necessarily required, and the amount of time is certainly up for debate. Collaborative co-editing also makes use of timeouts when grouping text that is merged across clients. Longer timeouts tend to make collaboration a smoother, less disruptive experience, because receiving other client input while expressing one’s thoughts tends to be unhelpful.
Implementers should consider the correct timeout based on their customer expectations and related collaborative features that might share this timeout.
**Hard and Soft Returns**
Hard returns as generated by pressing the Return key tend to disrupt and thereby reset continuous typing.
For Google Docs and Quip, soft returns as generated by pressing shift+Return key do not terminate continuous typing. This makes sense in that the soft return ends text wrap but does not end the paragraph.
**Changing Selections and Key Focus**
Any change to the selection by the user should terminate coalescing. This includes:

- Arrow gesture navigation.
- Gestures to select a word, a sentence or a paragraph.
- Any other selection change that does not move the blinking caret to the next or previous character by way of the above mentioned typing events.

Additionally, ending keyboard focus on the editing window, including moving focus to another editing window or a different view within the same window should also terminate coalescing. This also includes setting focus to modal popovers, such as, for example, a hyperlink property editor.
**Composed Characters**
Character composition occurs when typing diacritics, CJK and even emojis. The process usually involves at least these three phases:

- Start composition.
- Replace composed text with other text and remain composing. This step may occur many times within a single composition session.
- Stop composition either through accepting the text or canceling the latest input.

Coalescing should terminate at start composition. Replacing composed text should generally be coalesced. Stopping composition may or may not reset coalescing depending upon the implementation. Quip appears to try to coalesce multiple composed characters, but disrupts coalescing when switching between composed and non-composed characters. Google Docs appears to disrupt coalescing more frequently. Both coalesce the text replacements.
Coalescing should terminate at start composition. Replacing composed text should generally be coalesced. Stopping composition may or may not reset coalescing depending upon the implementation. Quip appears to try to coalesce multiple composed characters, but disrupts coalescing when switching between composed and non-composed characters. Google Docs appears to disrupt coalescing more frequently. Both coalesce the text replacements.
**Auto Correct, Spelling Corrections and Text Substitutions**
Text substitution behaves much like composed characters. When the text is transformed, the coalescing should terminate and the transformed text becomes the first coalesced entry in the next undo session.
**Copy & Pasting**
Any actions like copy and paste typically trigger events that are not part of the 3 continuous typing actions described above. So, copy and paste also terminates coalescing.
**In Conclusion**
Text undo coalescing greatly improves the usability of undo in that it allows the user to bypass all the intermediate events that went into forming or re-forming text within a paragraph. Given the modeless nature of text editing, limiting continuous typing to strictly three events allows for a robust and clean implementation. Implementers will need tight bottlenecks to trap selection changes, key focus changes as well as changes from collaborative clients. While the above is not complete by any means, I hope readers will find it helpful.
Text undo coalescing greatly improves the usability of undo in that it allows the user to bypass all the intermediate events that went into forming or re-forming text within a paragraph. Given the modeless nature of text editing, limiting continuous typing to strictly three events allows for a robust and clean implementation. Implementers will need tight bottlenecks to trap selection changes, key focus changes as well as changes from collaborative clients. While the above is not complete by any means, I hope readers will find it helpful.
28 changes: 14 additions & 14 deletions packages/lexical-clipboard/LexicalClipboard.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,26 +6,26 @@
*
*/

import type {LexicalEditor, RangeSelection} from 'lexical';
import type {LexicalEditor, RangeSelection} from 'lexical';

/*
* Rich Text
*/
/*
* Rich Text
*/

export function $insertDataTransferForRichText(
dataTransfer: DataTransfer,
selection: RangeSelection,
editor: LexicalEditor,
): void;
dataTransfer: DataTransfer,
selection: RangeSelection,
editor: LexicalEditor,
): void;

export function getHtmlContent(editor: LexicalEditor): string;
export function $getLexicalContent(editor: LexicalEditor): string;

/*
* Plain Text
*/
/*
* Plain Text
*/

export function $insertDataTransferForPlainText(
dataTransfer: DataTransfer,
selection: RangeSelection,
): void;
dataTransfer: DataTransfer,
selection: RangeSelection,
): void;
8 changes: 4 additions & 4 deletions packages/lexical-file/LexicalFile.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,10 @@
*
*/

import { LexicalEditor } from "lexical";
import {LexicalEditor} from 'lexical';

export function importFile(editor: LexicalEditor): void;
export function exportFile(
editor: LexicalEditor,
config?: Readonly<{fileName?: string, source?: string}>
): void;
editor: LexicalEditor,
config?: Readonly<{fileName?: string; source?: string}>,
): void;
2 changes: 1 addition & 1 deletion packages/lexical-file/flow/LexicalFile.js.flow
Original file line number Diff line number Diff line change
Expand Up @@ -13,5 +13,5 @@ import type {LexicalEditor} from 'lexical';
declare export function importFile(editor: LexicalEditor): void;
declare export function exportFile(
editor: LexicalEditor,
config?: $ReadOnly<{fileName?: string, source?: string}>
config?: $ReadOnly<{fileName?: string, source?: string}>,
): void;
40 changes: 23 additions & 17 deletions packages/lexical-list/LexicalList.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,14 @@
*
*/

import { ListNodeTagType } from "./src/LexicalListNode";
import { ElementNode, LexicalNode, LexicalEditor, ParagraphNode, RangeSelection } from 'lexical';
import {ListNodeTagType} from './src/LexicalListNode';
import {
ElementNode,
LexicalNode,
LexicalEditor,
ParagraphNode,
RangeSelection,
} from 'lexical';

export function $createListItemNode(): ListItemNode;
export function $createListNode(tag: ListNodeTagType, start?: number): ListNode;
Expand All @@ -18,22 +24,22 @@ export function $isListNode(node?: LexicalNode): boolean;
export function indentList(): boolean;
export function insertList(editor: LexicalEditor, listType: 'ul' | 'ol'): void;
export declare class ListItemNode extends ElementNode {
append(...nodes: LexicalNode[]): ListItemNode;
replace<N extends LexicalNode>(replaceWithNode: N): N;
insertAfter(node: LexicalNode): LexicalNode;
insertNewAfter(): ListItemNode | ParagraphNode;
collapseAtStart(selection: RangeSelection): true;
getIndent(): number;
setIndent(indent: number): this;
insertBefore(nodeToInsert: LexicalNode): LexicalNode;
canInsertAfter(node: LexicalNode): boolean;
canReplaceWith(replacement: LexicalNode): boolean;
canMergeWith(node: LexicalNode): boolean;
}
append(...nodes: LexicalNode[]): ListItemNode;
replace<N extends LexicalNode>(replaceWithNode: N): N;
insertAfter(node: LexicalNode): LexicalNode;
insertNewAfter(): ListItemNode | ParagraphNode;
collapseAtStart(selection: RangeSelection): true;
getIndent(): number;
setIndent(indent: number): this;
insertBefore(nodeToInsert: LexicalNode): LexicalNode;
canInsertAfter(node: LexicalNode): boolean;
canReplaceWith(replacement: LexicalNode): boolean;
canMergeWith(node: LexicalNode): boolean;
}
export declare class ListNode extends ElementNode {
canBeEmpty(): false;
append(...nodesToAppend: LexicalNode[]): ListNode;
getTag(): ListNodeTagType;
canBeEmpty(): false;
append(...nodesToAppend: LexicalNode[]): ListNode;
getTag(): ListNodeTagType;
}
export function outdentList(): boolean;
export function removeList(editor: LexicalEditor): boolean;
59 changes: 37 additions & 22 deletions packages/lexical-list/flow/LexicalList.js.flow
Original file line number Diff line number Diff line change
Expand Up @@ -7,36 +7,51 @@
* @flow strict
*/

import type { ListNodeTagType } from "../src/LexicalListNode";
import type { LexicalNode, LexicalEditor, ParagraphNode, RangeSelection } from 'lexical';
import { ElementNode } from 'lexical';
import type {ListNodeTagType} from '../src/LexicalListNode';
import type {
LexicalNode,
LexicalEditor,
ParagraphNode,
RangeSelection,
} from 'lexical';
import {ElementNode} from 'lexical';

declare export function $createListItemNode(): ListItemNode;
declare export function $createListNode(tag: ListNodeTagType, start?: number): ListNode;
declare export function $createListNode(
tag: ListNodeTagType,
start?: number,
): ListNode;
declare export function $getListDepth(listNode: ListNode): number;
declare export function $handleListInsertParagraph(): boolean;
declare export function $isListItemNode(node: ?LexicalNode): boolean %checks(node instanceof ListItemNode);
declare export function $isListNode(node: ?LexicalNode): boolean %checks(node instanceof ListNode);
declare export function $isListItemNode(
node: ?LexicalNode,
): boolean %checks(node instanceof ListItemNode);
declare export function $isListNode(
node: ?LexicalNode,
): boolean %checks(node instanceof ListNode);
declare export function indentList(): boolean;
declare export function insertList(editor: LexicalEditor, listType: 'ul' | 'ol'): void;
declare export function insertList(
editor: LexicalEditor,
listType: 'ul' | 'ol',
): void;
declare export class ListItemNode extends ElementNode {
append(...nodes: LexicalNode[]): ListItemNode;
replace<N: LexicalNode>(replaceWithNode: N): N;
insertAfter(node: LexicalNode): LexicalNode;
insertNewAfter(): ListItemNode | ParagraphNode;
collapseAtStart(selection: RangeSelection): true;
getIndent(): number;
setIndent(indent: number): this;
insertBefore(nodeToInsert: LexicalNode): LexicalNode;
canInsertAfter(node: LexicalNode): boolean;
canReplaceWith(replacement: LexicalNode): boolean;
canMergeWith(node: LexicalNode): boolean;
append(...nodes: LexicalNode[]): ListItemNode;
replace<N: LexicalNode>(replaceWithNode: N): N;
insertAfter(node: LexicalNode): LexicalNode;
insertNewAfter(): ListItemNode | ParagraphNode;
collapseAtStart(selection: RangeSelection): true;
getIndent(): number;
setIndent(indent: number): this;
insertBefore(nodeToInsert: LexicalNode): LexicalNode;
canInsertAfter(node: LexicalNode): boolean;
canReplaceWith(replacement: LexicalNode): boolean;
canMergeWith(node: LexicalNode): boolean;
}
declare export class ListNode extends ElementNode {
canBeEmpty(): false;
append(...nodesToAppend: LexicalNode[]): ListNode;
getTag(): ListNodeTagType;
getStart(): number;
canBeEmpty(): false;
append(...nodesToAppend: LexicalNode[]): ListNode;
getTag(): ListNodeTagType;
getStart(): number;
}
declare export function outdentList(): boolean;
declare export function removeList(editor: LexicalEditor): boolean;
Loading

0 comments on commit 622a063

Please sign in to comment.