draft-js
Version:
A React framework for building text editors.
226 lines (184 loc) • 7.91 kB
Flow
/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @format
* @flow
* @emails oncall+draft_js
*/
;
import type { BlockNodeRecord } from "./BlockNodeRecord";
import type ContentState from "./ContentState";
import type { DraftDecoratorComponentProps } from "./DraftDecorator";
import type { DraftDecoratorType } from "./DraftDecoratorType";
import type { DraftInlineStyle } from "./DraftInlineStyle";
import type SelectionState from "./SelectionState";
import type { BidiDirection } from "fbjs/lib/UnicodeBidiDirection";
import type { List } from "immutable";
const DraftEditorLeaf = require("./DraftEditorLeaf.react");
const DraftOffsetKey = require("./DraftOffsetKey");
const React = require("react");
const Scroll = require("fbjs/lib/Scroll");
const Style = require("fbjs/lib/Style");
const UnicodeBidi = require("fbjs/lib/UnicodeBidi");
const UnicodeBidiDirection = require("fbjs/lib/UnicodeBidiDirection");
const cx = require("fbjs/lib/cx");
const getElementPosition = require("fbjs/lib/getElementPosition");
const getScrollPosition = require("fbjs/lib/getScrollPosition");
const getViewportDimensions = require("fbjs/lib/getViewportDimensions");
const invariant = require("fbjs/lib/invariant");
const isHTMLElement = require("./isHTMLElement");
const nullthrows = require("fbjs/lib/nullthrows");
const SCROLL_BUFFER = 10;
type Props = {
block: BlockNodeRecord,
blockProps?: Object,
blockStyleFn: (block: BlockNodeRecord) => string,
contentState: ContentState,
customStyleFn: (style: DraftInlineStyle, block: BlockNodeRecord) => ?Object,
customStyleMap: Object,
decorator: ?DraftDecoratorType,
direction: BidiDirection,
forceSelection: boolean,
offsetKey: string,
preventScroll?: boolean,
selection: SelectionState,
startIndent?: boolean,
tree: List<any>,
...
};
/**
* Return whether a block overlaps with either edge of the `SelectionState`.
*/
const isBlockOnSelectionEdge = (selection: SelectionState, key: string): boolean => {
return selection.getAnchorKey() === key || selection.getFocusKey() === key;
};
/**
* The default block renderer for a `DraftEditor` component.
*
* A `DraftEditorBlock` is able to render a given `ContentBlock` to its
* appropriate decorator and inline style components.
*/
class DraftEditorBlock extends React.Component<Props> {
_node: ?HTMLDivElement;
shouldComponentUpdate(nextProps: Props): boolean {
return this.props.block !== nextProps.block || this.props.tree !== nextProps.tree || this.props.direction !== nextProps.direction || isBlockOnSelectionEdge(nextProps.selection, nextProps.block.getKey()) && nextProps.forceSelection;
}
/**
* When a block is mounted and overlaps the selection state, we need to make
* sure that the cursor is visible to match native behavior. This may not
* be the case if the user has pressed `RETURN` or pasted some content, since
* programmatically creating these new blocks and setting the DOM selection
* will miss out on the browser natively scrolling to that position.
*
* To replicate native behavior, if the block overlaps the selection state
* on mount, force the scroll position. Check the scroll state of the scroll
* parent, and adjust it to align the entire block to the bottom of the
* scroll parent.
*/
componentDidMount(): void {
if (this.props.preventScroll) {
return;
}
const selection = this.props.selection;
const endKey = selection.getEndKey();
if (!selection.getHasFocus() || endKey !== this.props.block.getKey()) {
return;
}
const blockNode = this._node;
if (blockNode == null) {
return;
}
const scrollParent = Style.getScrollParent(blockNode);
const scrollPosition = getScrollPosition(scrollParent);
let scrollDelta;
if (scrollParent === window) {
const nodePosition = getElementPosition(blockNode);
const nodeBottom = nodePosition.y + nodePosition.height;
const viewportHeight = getViewportDimensions().height;
scrollDelta = nodeBottom - viewportHeight;
if (scrollDelta > 0) {
window.scrollTo(scrollPosition.x, scrollPosition.y + scrollDelta + SCROLL_BUFFER);
}
} else {
invariant(isHTMLElement(blockNode), 'blockNode is not an HTMLElement');
const blockBottom = blockNode.offsetHeight + blockNode.offsetTop;
const pOffset = scrollParent.offsetTop + scrollParent.offsetHeight;
const scrollBottom = pOffset + scrollPosition.y;
scrollDelta = blockBottom - scrollBottom;
if (scrollDelta > 0) {
Scroll.setTop(scrollParent, Scroll.getTop(scrollParent) + scrollDelta + SCROLL_BUFFER);
}
}
}
_renderChildren(): Array<React.Node> {
const block = this.props.block;
const blockKey = block.getKey();
const text = block.getText();
const lastLeafSet = this.props.tree.size - 1;
const hasSelection = isBlockOnSelectionEdge(this.props.selection, blockKey);
return this.props.tree.map((leafSet, ii) => {
const leavesForLeafSet = leafSet.get('leaves'); // T44088704
if (leavesForLeafSet.size === 0) {
return null;
}
const lastLeaf = leavesForLeafSet.size - 1;
const leaves = leavesForLeafSet.map((leaf, jj) => {
const offsetKey = DraftOffsetKey.encode(blockKey, ii, jj);
const start = leaf.get('start');
const end = leaf.get('end');
return <DraftEditorLeaf key={offsetKey} offsetKey={offsetKey} block={block} start={start} selection={hasSelection ? this.props.selection : null} forceSelection={this.props.forceSelection} text={text.slice(start, end)} styleSet={block.getInlineStyleAt(start)} customStyleMap={this.props.customStyleMap} customStyleFn={this.props.customStyleFn} isLast={ii === lastLeafSet && jj === lastLeaf} />;
}).toArray();
const decoratorKey = leafSet.get('decoratorKey');
if (decoratorKey == null) {
return leaves;
}
if (!this.props.decorator) {
return leaves;
}
const decorator = nullthrows(this.props.decorator);
const DecoratorComponent = decorator.getComponentForKey(decoratorKey);
if (!DecoratorComponent) {
return leaves;
}
const decoratorProps = decorator.getPropsForKey(decoratorKey);
const decoratorOffsetKey = DraftOffsetKey.encode(blockKey, ii, 0);
const start = leavesForLeafSet.first().get('start');
const end = leavesForLeafSet.last().get('end');
const decoratedText = text.slice(start, end);
const entityKey = block.getEntityAt(leafSet.get('start')); // Resetting dir to the same value on a child node makes Chrome/Firefox
// confused on cursor movement. See http://jsfiddle.net/d157kLck/3/
const dir = UnicodeBidiDirection.getHTMLDirIfDifferent(UnicodeBidi.getDirection(decoratedText), this.props.direction);
const commonProps: DraftDecoratorComponentProps = {
contentState: this.props.contentState,
decoratedText,
dir: dir,
start,
end,
blockKey,
entityKey,
offsetKey: decoratorOffsetKey
};
return <DecoratorComponent {...decoratorProps} {...commonProps} key={decoratorOffsetKey}>
{leaves}
</DecoratorComponent>;
}).toArray();
}
render(): React.Node {
const {
direction,
offsetKey
} = this.props;
const className = cx({
'public/DraftStyleDefault/block': true,
'public/DraftStyleDefault/ltr': direction === 'LTR',
'public/DraftStyleDefault/rtl': direction === 'RTL'
});
return <div data-offset-key={offsetKey} className={className} ref={ref => this._node = ref}>
{this._renderChildren()}
</div>;
}
}
module.exports = DraftEditorBlock;