@blueprintjs/core
Version:
Core styles & components
245 lines (243 loc) • 38.4 kB
JavaScript
/*
* Copyright 2016 Palantir Technologies, Inc. All rights reserved.
* Licensed under the BSD-3 License as modified (the “License”); you may obtain a copy
* of the license at https://github.com/palantir/blueprint/blob/master/LICENSE
* and https://github.com/palantir/blueprint/blob/master/PATENTS
*/
"use strict";
var __extends = (this && this.__extends) || function (d, b) {
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
function __() { this.constructor = d; }
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
};
var __assign = (this && this.__assign) || Object.assign || function(t) {
for (var s, i = 1, n = arguments.length; i < n; i++) {
s = arguments[i];
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
t[p] = s[p];
}
return t;
};
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var classNames = require("classnames");
var PureRender = require("pure-render-decorator");
var React = require("react");
var Classes = require("../../common/classes");
var Keys = require("../../common/keys");
var utils_1 = require("../../common/utils");
var BUFFER_WIDTH = 30;
var EditableText = (function (_super) {
__extends(EditableText, _super);
function EditableText(props, context) {
var _this = this;
_super.call(this, props, context);
this.refHandlers = {
content: function (spanElement) {
_this.valueElement = spanElement;
},
input: function (input) {
if (input != null) {
input.focus();
var length_1 = input.value.length;
input.setSelectionRange(_this.props.selectAllOnFocus ? 0 : length_1, length_1);
}
},
};
this.cancelEditing = function () {
var lastValue = _this.state.lastValue;
_this.setState({ isEditing: false, value: lastValue });
// invoke onCancel after onChange so consumers' onCancel can override their onChange
utils_1.safeInvoke(_this.props.onChange, lastValue);
utils_1.safeInvoke(_this.props.onCancel, lastValue);
};
this.toggleEditing = function () {
if (_this.state.isEditing) {
var value = _this.state.value;
_this.setState({
isEditing: false,
lastValue: value,
});
utils_1.safeInvoke(_this.props.onChange, value);
utils_1.safeInvoke(_this.props.onConfirm, value);
}
else if (!_this.props.disabled) {
_this.setState({ isEditing: true });
}
};
this.handleFocus = function () {
if (!_this.props.disabled) {
_this.setState({ isEditing: true });
}
};
this.handleTextChange = function (event) {
var value = event.target.value;
// state value should be updated only when uncontrolled
if (_this.props.value == null) {
_this.setState({ value: value });
}
utils_1.safeInvoke(_this.props.onChange, value);
};
this.handleKeyEvent = function (_a) {
var ctrlKey = _a.ctrlKey, metaKey = _a.metaKey, which = _a.which;
if (which === Keys.ENTER && (!_this.props.multiline || ctrlKey || metaKey)) {
_this.toggleEditing();
}
else if (which === Keys.ESCAPE) {
_this.cancelEditing();
}
};
this.state = {
inputHeight: 0,
inputWidth: 0,
isEditing: props.isEditing === true && props.disabled === false,
lastValue: getValue(props),
value: getValue(props),
};
}
EditableText.prototype.render = function () {
var _a = this.props, disabled = _a.disabled, multiline = _a.multiline;
var value = (this.props.value == null ? this.state.value : this.props.value);
var hasValue = (value != null && value !== "");
var classes = classNames(Classes.EDITABLE_TEXT, Classes.intentClass(this.props.intent), (_b = {},
_b[Classes.DISABLED] = disabled,
_b["pt-editable-editing"] = this.state.isEditing,
_b["pt-editable-placeholder"] = !hasValue,
_b["pt-multiline"] = multiline,
_b
), this.props.className);
var contentStyle;
if (multiline) {
// set height only in multiline mode when not editing
// otherwise we're measuring this element to determine appropriate height of text
contentStyle = { height: !this.state.isEditing ? this.state.inputHeight : null };
}
else {
// minWidth only applies in single line mode (multiline == width 100%)
contentStyle = {
height: this.state.inputHeight,
lineHeight: this.state.inputHeight != null ? this.state.inputHeight + "px" : null,
minWidth: this.props.minWidth,
};
}
// make enclosing div focusable when not editing, so it can still be tabbed to focus
// (when editing, input itself is focusable so div doesn't need to be)
var tabIndex = this.state.isEditing || disabled ? null : 0;
return (React.createElement("div", {className: classes, onFocus: this.handleFocus, tabIndex: tabIndex},
this.maybeRenderInput(value),
React.createElement("span", {className: "pt-editable-content", ref: this.refHandlers.content, style: contentStyle}, hasValue ? value : this.props.placeholder)));
var _b;
};
EditableText.prototype.componentDidMount = function () {
this.updateInputDimensions();
};
EditableText.prototype.componentDidUpdate = function (_, prevState) {
if (this.state.isEditing && !prevState.isEditing) {
utils_1.safeInvoke(this.props.onEdit);
}
this.updateInputDimensions();
};
EditableText.prototype.componentWillReceiveProps = function (nextProps) {
var state = { value: getValue(nextProps) };
if (nextProps.isEditing != null) {
state.isEditing = nextProps.isEditing;
}
if (nextProps.disabled || (nextProps.disabled == null && this.props.disabled)) {
state.isEditing = false;
}
this.setState(state);
};
EditableText.prototype.maybeRenderInput = function (value) {
var multiline = this.props.multiline;
if (!this.state.isEditing) {
return undefined;
}
var props = {
className: "pt-editable-input",
onBlur: this.toggleEditing,
onChange: this.handleTextChange,
onKeyDown: this.handleKeyEvent,
ref: this.refHandlers.input,
style: {
height: this.state.inputHeight,
lineHeight: !multiline && this.state.inputHeight != null ? this.state.inputHeight + "px" : null,
width: multiline ? "100%" : this.state.inputWidth,
},
value: value,
};
return multiline ? React.createElement("textarea", __assign({}, props)) : React.createElement("input", __assign({type: "text"}, props));
};
EditableText.prototype.updateInputDimensions = function () {
if (this.valueElement != null) {
var _a = this.props, maxLines = _a.maxLines, minLines = _a.minLines, minWidth = _a.minWidth, multiline = _a.multiline;
var _b = this.valueElement, parentElement_1 = _b.parentElement, scrollHeight_1 = _b.scrollHeight, scrollWidth = _b.scrollWidth, textContent = _b.textContent;
var lineHeight = getLineHeight(this.valueElement);
// add one line to computed <span> height if text ends in newline
// because <span> collapses that trailing whitespace but <textarea> shows it
if (multiline && this.state.isEditing && /\n$/.test(textContent)) {
scrollHeight_1 += lineHeight;
}
if (lineHeight > 0) {
// line height could be 0 if the isNaN block from getLineHeight kicks in
scrollHeight_1 = utils_1.clamp(scrollHeight_1, minLines * lineHeight, maxLines * lineHeight);
}
// Chrome's input caret height misaligns text so the line-height must be larger than font-size.
// The computed scrollHeight must also account for a larger inherited line-height from the parent.
scrollHeight_1 = Math.max(scrollHeight_1, getFontSize(this.valueElement) + 1, getLineHeight(parentElement_1));
// IE11 needs a small buffer so text does not shift prior to resizing
this.setState({
inputHeight: scrollHeight_1,
inputWidth: Math.max(scrollWidth + BUFFER_WIDTH, minWidth),
});
// synchronizes the ::before pseudo-element's height while editing for Chrome 53
if (multiline && this.state.isEditing) {
setTimeout(function () { return parentElement_1.style.height = scrollHeight_1 + "px"; });
}
}
};
EditableText.defaultProps = {
defaultValue: "",
disabled: false,
maxLines: Infinity,
minLines: 1,
minWidth: 80,
multiline: false,
placeholder: "Click to Edit",
};
EditableText = __decorate([
PureRender
], EditableText);
return EditableText;
}(React.Component));
exports.EditableText = EditableText;
function getValue(props) {
return props.value == null ? props.defaultValue : props.value;
}
function getFontSize(element) {
var fontSize = getComputedStyle(element).fontSize;
return fontSize === "" ? 0 : parseInt(fontSize.slice(0, -2), 10);
}
function getLineHeight(element) {
// getComputedStyle() => 18.0001px => 18
var lineHeight = parseInt(getComputedStyle(element).lineHeight.slice(0, -2), 10);
// this check will be true if line-height is a keyword like "normal"
if (isNaN(lineHeight)) {
// @see http://stackoverflow.com/a/18430767/6342931
var line = document.createElement("span");
line.innerHTML = "<br>";
element.appendChild(line);
var singleLineHeight = element.offsetHeight;
line.innerHTML = "<br><br>";
var doubleLineHeight = element.offsetHeight;
element.removeChild(line);
// this can return 0 in edge cases
lineHeight = doubleLineHeight - singleLineHeight;
}
return lineHeight;
}
exports.EditableTextFactory = React.createFactory(EditableText);
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/components/editable-text/editableText.tsx"],"names":[],"mappings":"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","file":"components/editable-text/editableText.js","sourcesContent":["/*\n * Copyright 2016 Palantir Technologies, Inc. All rights reserved.\n * Licensed under the BSD-3 License as modified (the “License”); you may obtain a copy\n * of the license at https://github.com/palantir/blueprint/blob/master/LICENSE\n * and https://github.com/palantir/blueprint/blob/master/PATENTS\n */\n\nimport * as classNames from \"classnames\";\nimport * as PureRender from \"pure-render-decorator\";\nimport * as React from \"react\";\n\nimport * as Classes from \"../../common/classes\";\nimport * as Keys from \"../../common/keys\";\nimport { IIntentProps, IProps } from \"../../common/props\";\nimport { clamp, safeInvoke } from \"../../common/utils\";\n\nexport interface IEditableTextProps extends IIntentProps, IProps {\n    /** Default text value of uncontrolled input. */\n    defaultValue?: string;\n\n    /**\n     * Whether the text can be edited.\n     * @default false\n     */\n    disabled?: boolean;\n\n    /** Whether the component is currently being edited. */\n    isEditing?: boolean;\n\n    /** Minimum width in pixels of the input, when not `multiline`. */\n    minWidth?: number;\n\n    /**\n     * Whether the component supports multiple lines of text.\n     * This prop should not be changed during the component's lifetime.\n     * @default false\n     */\n    multiline?: boolean;\n\n    /**\n     * Maximum number of lines before scrolling begins, when `multiline`.\n     */\n    maxLines?: number;\n\n    /**\n     * Minimum number of lines (essentially minimum height), when `multiline`.\n     * @default 1\n     */\n    minLines?: number;\n\n    /**\n     * Placeholder text when there is no value.\n     * @default \"Click to Edit\"\n     */\n    placeholder?: string;\n\n    /**\n     * Whether the entire text field should be selected on focus.\n     * If false, the cursor is placed at the end of the text.\n     * @default false\n     */\n    selectAllOnFocus?: boolean;\n\n    /** Text value of controlled input. */\n    value?: string;\n\n    /** Callback invoked when user cancels input with the `esc` key. Receives last confirmed value. */\n    onCancel?(value: string): void;\n\n    /** Callback invoked when user changes input in any way. */\n    onChange?(value: string): void;\n\n    /** Callback invoked when user confirms value with `enter` key or by blurring input. */\n    onConfirm?(value: string): void;\n\n    /** Callback invoked after the user enters edit mode. */\n    onEdit?(): void;\n}\n\nexport interface IEditableTextState {\n    /** Pixel height of the input, measured from span size */\n    inputHeight?: number;\n    /** Pixel width of the input, measured from span size */\n    inputWidth?: number;\n    /** Whether the value is currently being edited */\n    isEditing?: boolean;\n    /** The last confirmed value */\n    lastValue?: string;\n    /** The controlled input value, may be different from prop during editing */\n    value?: string;\n}\n\nconst BUFFER_WIDTH = 30;\n\n@PureRender\nexport class EditableText extends React.Component<IEditableTextProps, IEditableTextState> {\n    public static defaultProps: IEditableTextProps = {\n        defaultValue: \"\",\n        disabled: false,\n        maxLines: Infinity,\n        minLines: 1,\n        minWidth: 80,\n        multiline: false,\n        placeholder: \"Click to Edit\",\n    };\n\n    private valueElement: HTMLSpanElement;\n    private refHandlers = {\n        content: (spanElement: HTMLSpanElement) => {\n            this.valueElement = spanElement;\n        },\n        input: (input: HTMLInputElement | HTMLTextAreaElement) => {\n            if (input != null) {\n                input.focus();\n                const { length } = input.value;\n                input.setSelectionRange(this.props.selectAllOnFocus ? 0 : length, length);\n            }\n        },\n    };\n\n    public constructor(props?: IEditableTextProps, context?: any) {\n        super(props, context);\n\n        this.state = {\n            inputHeight: 0,\n            inputWidth: 0,\n            isEditing: props.isEditing === true && props.disabled === false,\n            lastValue: getValue(props),\n            value: getValue(props),\n        };\n    }\n\n    public render() {\n        const { disabled, multiline } = this.props;\n        const value = (this.props.value == null ? this.state.value : this.props.value);\n        const hasValue = (value != null && value !== \"\");\n\n        const classes = classNames(\n            Classes.EDITABLE_TEXT,\n            Classes.intentClass(this.props.intent),\n            {\n                [Classes.DISABLED]: disabled,\n                \"pt-editable-editing\": this.state.isEditing,\n                \"pt-editable-placeholder\": !hasValue,\n                \"pt-multiline\": multiline,\n            },\n            this.props.className,\n        );\n\n        let contentStyle: React.CSSProperties;\n        if (multiline) {\n            // set height only in multiline mode when not editing\n            // otherwise we're measuring this element to determine appropriate height of text\n            contentStyle = { height: !this.state.isEditing ? this.state.inputHeight : null };\n        } else {\n            // minWidth only applies in single line mode (multiline == width 100%)\n            contentStyle = {\n                height: this.state.inputHeight,\n                lineHeight: this.state.inputHeight != null ? `${this.state.inputHeight}px` : null,\n                minWidth: this.props.minWidth,\n            };\n        }\n\n        // make enclosing div focusable when not editing, so it can still be tabbed to focus\n        // (when editing, input itself is focusable so div doesn't need to be)\n        const tabIndex = this.state.isEditing || disabled ? null : 0;\n        return (\n            <div className={classes} onFocus={this.handleFocus} tabIndex={tabIndex}>\n                {this.maybeRenderInput(value)}\n                <span className=\"pt-editable-content\" ref={this.refHandlers.content} style={contentStyle}>\n                    {hasValue ? value : this.props.placeholder}\n                </span>\n            </div>\n        );\n    }\n\n    public componentDidMount() {\n        this.updateInputDimensions();\n    }\n\n    public componentDidUpdate(_: IEditableTextProps, prevState: IEditableTextState) {\n        if (this.state.isEditing && !prevState.isEditing) {\n            safeInvoke(this.props.onEdit);\n        }\n        this.updateInputDimensions();\n    }\n\n    public componentWillReceiveProps(nextProps: IEditableTextProps) {\n        const state: IEditableTextState = { value: getValue(nextProps) };\n        if (nextProps.isEditing != null)  {\n            state.isEditing = nextProps.isEditing;\n        }\n        if (nextProps.disabled || (nextProps.disabled == null && this.props.disabled)) {\n            state.isEditing = false;\n        }\n        this.setState(state);\n    }\n\n    public cancelEditing = () => {\n        const { lastValue } = this.state;\n        this.setState({ isEditing: false, value: lastValue });\n        // invoke onCancel after onChange so consumers' onCancel can override their onChange\n        safeInvoke(this.props.onChange, lastValue);\n        safeInvoke(this.props.onCancel, lastValue);\n    }\n\n    public toggleEditing = () => {\n        if (this.state.isEditing) {\n            const { value } = this.state;\n            this.setState({\n                isEditing: false,\n                lastValue: value,\n            });\n            safeInvoke(this.props.onChange, value);\n            safeInvoke(this.props.onConfirm, value);\n        } else if (!this.props.disabled) {\n            this.setState({ isEditing: true });\n        }\n    }\n\n    private handleFocus = () => {\n        if (!this.props.disabled) {\n            this.setState({ isEditing: true });\n        }\n    }\n\n    private handleTextChange = (event: React.FormEvent<HTMLElement>) => {\n        const value = (event.target as HTMLInputElement).value;\n        // state value should be updated only when uncontrolled\n        if (this.props.value == null) { this.setState({ value }); }\n        safeInvoke(this.props.onChange, value);\n    }\n\n    private handleKeyEvent = ({ ctrlKey, metaKey, which }: React.KeyboardEvent<HTMLElement>) => {\n        if (which === Keys.ENTER && (!this.props.multiline || ctrlKey || metaKey)) {\n            this.toggleEditing();\n        } else if (which === Keys.ESCAPE) {\n            this.cancelEditing();\n        }\n    }\n\n    private maybeRenderInput(value: string) {\n        const { multiline } = this.props;\n        if (!this.state.isEditing) {\n            return undefined;\n        }\n        const props: React.HTMLProps<HTMLInputElement | HTMLTextAreaElement> = {\n            className: \"pt-editable-input\",\n            onBlur: this.toggleEditing,\n            onChange: this.handleTextChange,\n            onKeyDown: this.handleKeyEvent,\n            ref: this.refHandlers.input,\n            style: {\n                height: this.state.inputHeight,\n                lineHeight: !multiline && this.state.inputHeight != null ? `${this.state.inputHeight}px` : null,\n                width: multiline ? \"100%\" : this.state.inputWidth,\n            },\n            value,\n        };\n        return multiline ? <textarea {...props} /> : <input type=\"text\" {...props} />;\n    }\n\n    private updateInputDimensions() {\n        if (this.valueElement != null) {\n            const { maxLines, minLines, minWidth, multiline } = this.props;\n            let { parentElement, scrollHeight, scrollWidth, textContent } = this.valueElement;\n            const lineHeight = getLineHeight(this.valueElement);\n            // add one line to computed <span> height if text ends in newline\n            // because <span> collapses that trailing whitespace but <textarea> shows it\n            if (multiline && this.state.isEditing && /\\n$/.test(textContent)) {\n                scrollHeight += lineHeight;\n            }\n            if (lineHeight > 0) {\n                // line height could be 0 if the isNaN block from getLineHeight kicks in\n                scrollHeight = clamp(scrollHeight, minLines * lineHeight, maxLines * lineHeight);\n            }\n            // Chrome's input caret height misaligns text so the line-height must be larger than font-size.\n            // The computed scrollHeight must also account for a larger inherited line-height from the parent.\n            scrollHeight = Math.max(scrollHeight, getFontSize(this.valueElement) + 1, getLineHeight(parentElement));\n            // IE11 needs a small buffer so text does not shift prior to resizing\n            this.setState({\n                inputHeight: scrollHeight,\n                inputWidth: Math.max(scrollWidth + BUFFER_WIDTH, minWidth),\n            });\n            // synchronizes the ::before pseudo-element's height while editing for Chrome 53\n            if (multiline && this.state.isEditing) {\n                setTimeout(() => parentElement.style.height = `${scrollHeight}px`);\n            }\n        }\n    }\n}\n\nfunction getValue(props: IEditableTextProps) {\n    return props.value == null ? props.defaultValue : props.value;\n}\n\nfunction getFontSize(element: HTMLElement) {\n    const fontSize = getComputedStyle(element).fontSize;\n    return fontSize === \"\" ? 0 : parseInt(fontSize.slice(0, -2), 10);\n}\n\nfunction getLineHeight(element: HTMLElement) {\n    // getComputedStyle() => 18.0001px => 18\n    let lineHeight = parseInt(getComputedStyle(element).lineHeight.slice(0, -2), 10);\n    // this check will be true if line-height is a keyword like \"normal\"\n    if (isNaN(lineHeight)) {\n        // @see http://stackoverflow.com/a/18430767/6342931\n        const line = document.createElement(\"span\");\n        line.innerHTML = \"<br>\";\n        element.appendChild(line);\n        const singleLineHeight = element.offsetHeight;\n        line.innerHTML = \"<br><br>\";\n        const doubleLineHeight = element.offsetHeight;\n        element.removeChild(line);\n        // this can return 0 in edge cases\n        lineHeight = doubleLineHeight - singleLineHeight;\n    }\n    return lineHeight;\n}\n\nexport const EditableTextFactory = React.createFactory(EditableText);\n"],"sourceRoot":"/source/"}