UNPKG

matrix-react-sdk

Version:
225 lines (193 loc) 25.7 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _Keyboard = require("../../../Keyboard"); var _replaceableComponent = require("../../../utils/replaceableComponent"); var _dec, _class, _class2, _temp; let EditableText = (_dec = (0, _replaceableComponent.replaceableComponent)("views.elements.EditableText"), _dec(_class = (_temp = _class2 = class EditableText extends _react.default.Component { constructor(props) { super(props); // we track value as an JS object field rather than in React state // as React doesn't play nice with contentEditable. (0, _defineProperty2.default)(this, "state", { phase: EditableText.Phases.Display }); (0, _defineProperty2.default)(this, "showPlaceholder", show => { if (show) { this._editable_div.current.textContent = this.props.placeholder; this._editable_div.current.setAttribute("class", this.props.className + " " + this.props.placeholderClassName); this.placeholder = true; this.value = ''; } else { this._editable_div.current.textContent = this.value; this._editable_div.current.setAttribute("class", this.props.className); this.placeholder = false; } }); (0, _defineProperty2.default)(this, "getValue", () => this.value); (0, _defineProperty2.default)(this, "setValue", value => { this.value = value; this.showPlaceholder(!this.value); }); (0, _defineProperty2.default)(this, "edit", () => { this.setState({ phase: EditableText.Phases.Edit }); }); (0, _defineProperty2.default)(this, "cancelEdit", () => { this.setState({ phase: EditableText.Phases.Display }); this.value = this.props.initialValue; this.showPlaceholder(!this.value); this.onValueChanged(false); this._editable_div.current.blur(); }); (0, _defineProperty2.default)(this, "onValueChanged", shouldSubmit => { this.props.onValueChanged(this.value, shouldSubmit); }); (0, _defineProperty2.default)(this, "onKeyDown", ev => { // console.log("keyDown: textContent=" + ev.target.textContent + ", value=" + this.value + ", placeholder=" + this.placeholder); if (this.placeholder) { this.showPlaceholder(false); } if (ev.key === _Keyboard.Key.ENTER) { ev.stopPropagation(); ev.preventDefault(); } // console.log("keyDown: textContent=" + ev.target.textContent + ", value=" + this.value + ", placeholder=" + this.placeholder); }); (0, _defineProperty2.default)(this, "onKeyUp", ev => { // console.log("keyUp: textContent=" + ev.target.textContent + ", value=" + this.value + ", placeholder=" + this.placeholder); if (!ev.target.textContent) { this.showPlaceholder(true); } else if (!this.placeholder) { this.value = ev.target.textContent; } if (ev.key === _Keyboard.Key.ENTER) { this.onFinish(ev); } else if (ev.key === _Keyboard.Key.ESCAPE) { this.cancelEdit(); } // console.log("keyUp: textContent=" + ev.target.textContent + ", value=" + this.value + ", placeholder=" + this.placeholder); }); (0, _defineProperty2.default)(this, "onClickDiv", ev => { if (!this.props.editable) return; this.setState({ phase: EditableText.Phases.Edit }); }); (0, _defineProperty2.default)(this, "onFocus", ev => { //ev.target.setSelectionRange(0, ev.target.textContent.length); const node = ev.target.childNodes[0]; if (node) { const range = document.createRange(); range.setStart(node, 0); range.setEnd(node, node.length); const sel = window.getSelection(); sel.removeAllRanges(); sel.addRange(range); } }); (0, _defineProperty2.default)(this, "onFinish", (ev, shouldSubmit) => { const self = this; const submit = ev.key === _Keyboard.Key.ENTER || shouldSubmit; this.setState({ phase: EditableText.Phases.Display }, () => { if (this.value !== this.props.initialValue) { self.onValueChanged(submit); } }); }); (0, _defineProperty2.default)(this, "onBlur", ev => { const sel = window.getSelection(); sel.removeAllRanges(); if (this.props.blurToCancel) { this.cancelEdit(); } else { this.onFinish(ev, this.props.blurToSubmit); } this.showPlaceholder(!this.value); }); this.value = ''; this.placeholder = false; this._editable_div = /*#__PURE__*/(0, _react.createRef)(); } // TODO: [REACT-WARNING] Replace with appropriate lifecycle event // eslint-disable-next-line camelcase UNSAFE_componentWillReceiveProps(nextProps) { if (nextProps.initialValue !== this.props.initialValue) { this.value = nextProps.initialValue; if (this._editable_div.current) { this.showPlaceholder(!this.value); } } } componentDidMount() { this.value = this.props.initialValue; if (this._editable_div.current) { this.showPlaceholder(!this.value); } } render() { const { className, editable, initialValue, label, labelClassName } = this.props; let editableEl; if (!editable || this.state.phase === EditableText.Phases.Display && (label || labelClassName) && !this.value) { // show the label editableEl = /*#__PURE__*/_react.default.createElement("div", { className: className + " " + labelClassName, onClick: this.onClickDiv }, label || initialValue); } else { // show the content editable div, but manually manage its contents as react and contentEditable don't play nice together editableEl = /*#__PURE__*/_react.default.createElement("div", { ref: this._editable_div, contentEditable: true, className: className, onKeyDown: this.onKeyDown, onKeyUp: this.onKeyUp, onFocus: this.onFocus, onBlur: this.onBlur }); } return editableEl; } }, (0, _defineProperty2.default)(_class2, "propTypes", { onValueChanged: _propTypes.default.func, initialValue: _propTypes.default.string, label: _propTypes.default.string, placeholder: _propTypes.default.string, className: _propTypes.default.string, labelClassName: _propTypes.default.string, placeholderClassName: _propTypes.default.string, // Overrides blurToSubmit if true blurToCancel: _propTypes.default.bool, // Will cause onValueChanged(value, true) to fire on blur blurToSubmit: _propTypes.default.bool, editable: _propTypes.default.bool }), (0, _defineProperty2.default)(_class2, "Phases", { Display: "display", Edit: "edit" }), (0, _defineProperty2.default)(_class2, "defaultProps", { onValueChanged() {}, initialValue: '', label: '', placeholder: '', editable: true, className: "mx_EditableText", placeholderClassName: "mx_EditableText_placeholder", blurToSubmit: false }), _temp)) || _class); exports.default = EditableText; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/elements/EditableText.js"],"names":["EditableText","React","Component","constructor","props","phase","Phases","Display","show","_editable_div","current","textContent","placeholder","setAttribute","className","placeholderClassName","value","showPlaceholder","setState","Edit","initialValue","onValueChanged","blur","shouldSubmit","ev","key","Key","ENTER","stopPropagation","preventDefault","target","onFinish","ESCAPE","cancelEdit","editable","node","childNodes","range","document","createRange","setStart","setEnd","length","sel","window","getSelection","removeAllRanges","addRange","self","submit","blurToCancel","blurToSubmit","UNSAFE_componentWillReceiveProps","nextProps","componentDidMount","render","label","labelClassName","editableEl","state","onClickDiv","onKeyDown","onKeyUp","onFocus","onBlur","PropTypes","func","string","bool"],"mappings":";;;;;;;;;;;;;AAiBA;;AACA;;AACA;;AACA;;;;IAGqBA,Y,WADpB,gDAAqB,6BAArB,C,mCAAD,MACqBA,YADrB,SAC0CC,eAAMC,SADhD,CAC0D;AAgCtDC,EAAAA,WAAW,CAACC,KAAD,EAAQ;AACf,UAAMA,KAAN,EADe,CAGf;AACA;;AAJe,iDAWX;AACJC,MAAAA,KAAK,EAAEL,YAAY,CAACM,MAAb,CAAoBC;AADvB,KAXW;AAAA,2DAiCDC,IAAI,IAAI;AACtB,UAAIA,IAAJ,EAAU;AACN,aAAKC,aAAL,CAAmBC,OAAnB,CAA2BC,WAA3B,GAAyC,KAAKP,KAAL,CAAWQ,WAApD;;AACA,aAAKH,aAAL,CAAmBC,OAAnB,CAA2BG,YAA3B,CAAwC,OAAxC,EAAiD,KAAKT,KAAL,CAAWU,SAAX,GAC3C,GAD2C,GACrC,KAAKV,KAAL,CAAWW,oBADvB;;AAEA,aAAKH,WAAL,GAAmB,IAAnB;AACA,aAAKI,KAAL,GAAa,EAAb;AACH,OAND,MAMO;AACH,aAAKP,aAAL,CAAmBC,OAAnB,CAA2BC,WAA3B,GAAyC,KAAKK,KAA9C;;AACA,aAAKP,aAAL,CAAmBC,OAAnB,CAA2BG,YAA3B,CAAwC,OAAxC,EAAiD,KAAKT,KAAL,CAAWU,SAA5D;;AACA,aAAKF,WAAL,GAAmB,KAAnB;AACH;AACJ,KA7CkB;AAAA,oDA+CR,MAAM,KAAKI,KA/CH;AAAA,oDAiDRA,KAAK,IAAI;AAChB,WAAKA,KAAL,GAAaA,KAAb;AACA,WAAKC,eAAL,CAAqB,CAAC,KAAKD,KAA3B;AACH,KApDkB;AAAA,gDAsDZ,MAAM;AACT,WAAKE,QAAL,CAAc;AACVb,QAAAA,KAAK,EAAEL,YAAY,CAACM,MAAb,CAAoBa;AADjB,OAAd;AAGH,KA1DkB;AAAA,sDA4DN,MAAM;AACf,WAAKD,QAAL,CAAc;AACVb,QAAAA,KAAK,EAAEL,YAAY,CAACM,MAAb,CAAoBC;AADjB,OAAd;AAGA,WAAKS,KAAL,GAAa,KAAKZ,KAAL,CAAWgB,YAAxB;AACA,WAAKH,eAAL,CAAqB,CAAC,KAAKD,KAA3B;AACA,WAAKK,cAAL,CAAoB,KAApB;;AACA,WAAKZ,aAAL,CAAmBC,OAAnB,CAA2BY,IAA3B;AACH,KApEkB;AAAA,0DAsEFC,YAAY,IAAI;AAC7B,WAAKnB,KAAL,CAAWiB,cAAX,CAA0B,KAAKL,KAA/B,EAAsCO,YAAtC;AACH,KAxEkB;AAAA,qDA0EPC,EAAE,IAAI;AACd;AAEA,UAAI,KAAKZ,WAAT,EAAsB;AAClB,aAAKK,eAAL,CAAqB,KAArB;AACH;;AAED,UAAIO,EAAE,CAACC,GAAH,KAAWC,cAAIC,KAAnB,EAA0B;AACtBH,QAAAA,EAAE,CAACI,eAAH;AACAJ,QAAAA,EAAE,CAACK,cAAH;AACH,OAVa,CAYd;;AACH,KAvFkB;AAAA,mDAyFTL,EAAE,IAAI;AACZ;AAEA,UAAI,CAACA,EAAE,CAACM,MAAH,CAAUnB,WAAf,EAA4B;AACxB,aAAKM,eAAL,CAAqB,IAArB;AACH,OAFD,MAEO,IAAI,CAAC,KAAKL,WAAV,EAAuB;AAC1B,aAAKI,KAAL,GAAaQ,EAAE,CAACM,MAAH,CAAUnB,WAAvB;AACH;;AAED,UAAIa,EAAE,CAACC,GAAH,KAAWC,cAAIC,KAAnB,EAA0B;AACtB,aAAKI,QAAL,CAAcP,EAAd;AACH,OAFD,MAEO,IAAIA,EAAE,CAACC,GAAH,KAAWC,cAAIM,MAAnB,EAA2B;AAC9B,aAAKC,UAAL;AACH,OAbW,CAeZ;;AACH,KAzGkB;AAAA,sDA2GNT,EAAE,IAAI;AACf,UAAI,CAAC,KAAKpB,KAAL,CAAW8B,QAAhB,EAA0B;AAE1B,WAAKhB,QAAL,CAAc;AACVb,QAAAA,KAAK,EAAEL,YAAY,CAACM,MAAb,CAAoBa;AADjB,OAAd;AAGH,KAjHkB;AAAA,mDAmHTK,EAAE,IAAI;AACZ;AAEA,YAAMW,IAAI,GAAGX,EAAE,CAACM,MAAH,CAAUM,UAAV,CAAqB,CAArB,CAAb;;AACA,UAAID,IAAJ,EAAU;AACN,cAAME,KAAK,GAAGC,QAAQ,CAACC,WAAT,EAAd;AACAF,QAAAA,KAAK,CAACG,QAAN,CAAeL,IAAf,EAAqB,CAArB;AACAE,QAAAA,KAAK,CAACI,MAAN,CAAaN,IAAb,EAAmBA,IAAI,CAACO,MAAxB;AAEA,cAAMC,GAAG,GAAGC,MAAM,CAACC,YAAP,EAAZ;AACAF,QAAAA,GAAG,CAACG,eAAJ;AACAH,QAAAA,GAAG,CAACI,QAAJ,CAAaV,KAAb;AACH;AACJ,KAhIkB;AAAA,oDAkIR,CAACb,EAAD,EAAKD,YAAL,KAAsB;AAC7B,YAAMyB,IAAI,GAAG,IAAb;AACA,YAAMC,MAAM,GAAIzB,EAAE,CAACC,GAAH,KAAWC,cAAIC,KAAhB,IAA0BJ,YAAzC;AACA,WAAKL,QAAL,CAAc;AACVb,QAAAA,KAAK,EAAEL,YAAY,CAACM,MAAb,CAAoBC;AADjB,OAAd,EAEG,MAAM;AACL,YAAI,KAAKS,KAAL,KAAe,KAAKZ,KAAL,CAAWgB,YAA9B,EAA4C;AACxC4B,UAAAA,IAAI,CAAC3B,cAAL,CAAoB4B,MAApB;AACH;AACJ,OAND;AAOH,KA5IkB;AAAA,kDA8IVzB,EAAE,IAAI;AACX,YAAMmB,GAAG,GAAGC,MAAM,CAACC,YAAP,EAAZ;AACAF,MAAAA,GAAG,CAACG,eAAJ;;AAEA,UAAI,KAAK1C,KAAL,CAAW8C,YAAf,EAA6B;AACzB,aAAKjB,UAAL;AACH,OAFD,MAEO;AACH,aAAKF,QAAL,CAAcP,EAAd,EAAkB,KAAKpB,KAAL,CAAW+C,YAA7B;AACH;;AAED,WAAKlC,eAAL,CAAqB,CAAC,KAAKD,KAA3B;AACH,KAzJkB;AAKf,SAAKA,KAAL,GAAa,EAAb;AACA,SAAKJ,WAAL,GAAmB,KAAnB;AAEA,SAAKH,aAAL,gBAAqB,uBAArB;AACH;;AAMD;AACA;AACA2C,EAAAA,gCAAgC,CAACC,SAAD,EAAY;AACxC,QAAIA,SAAS,CAACjC,YAAV,KAA2B,KAAKhB,KAAL,CAAWgB,YAA1C,EAAwD;AACpD,WAAKJ,KAAL,GAAaqC,SAAS,CAACjC,YAAvB;;AACA,UAAI,KAAKX,aAAL,CAAmBC,OAAvB,EAAgC;AAC5B,aAAKO,eAAL,CAAqB,CAAC,KAAKD,KAA3B;AACH;AACJ;AACJ;;AAEDsC,EAAAA,iBAAiB,GAAG;AAChB,SAAKtC,KAAL,GAAa,KAAKZ,KAAL,CAAWgB,YAAxB;;AACA,QAAI,KAAKX,aAAL,CAAmBC,OAAvB,EAAgC;AAC5B,WAAKO,eAAL,CAAqB,CAAC,KAAKD,KAA3B;AACH;AACJ;;AA4HDuC,EAAAA,MAAM,GAAG;AACL,UAAM;AAACzC,MAAAA,SAAD;AAAYoB,MAAAA,QAAZ;AAAsBd,MAAAA,YAAtB;AAAoCoC,MAAAA,KAApC;AAA2CC,MAAAA;AAA3C,QAA6D,KAAKrD,KAAxE;AACA,QAAIsD,UAAJ;;AAEA,QAAI,CAACxB,QAAD,IAAc,KAAKyB,KAAL,CAAWtD,KAAX,KAAqBL,YAAY,CAACM,MAAb,CAAoBC,OAAzC,KACbiD,KAAK,IAAIC,cADI,KACe,CAAC,KAAKzC,KADvC,EAEE;AACE;AACA0C,MAAAA,UAAU,gBAAG;AAAK,QAAA,SAAS,EAAE5C,SAAS,GAAG,GAAZ,GAAkB2C,cAAlC;AAAkD,QAAA,OAAO,EAAE,KAAKG;AAAhE,SACPJ,KAAK,IAAIpC,YADF,CAAb;AAGH,KAPD,MAOO;AACH;AACAsC,MAAAA,UAAU,gBAAG;AACT,QAAA,GAAG,EAAE,KAAKjD,aADD;AAET,QAAA,eAAe,EAAE,IAFR;AAGT,QAAA,SAAS,EAAEK,SAHF;AAIT,QAAA,SAAS,EAAE,KAAK+C,SAJP;AAKT,QAAA,OAAO,EAAE,KAAKC,OALL;AAMT,QAAA,OAAO,EAAE,KAAKC,OANL;AAOT,QAAA,MAAM,EAAE,KAAKC;AAPJ,QAAb;AASH;;AAED,WAAON,UAAP;AACH;;AApNqD,C,sDACnC;AACfrC,EAAAA,cAAc,EAAE4C,mBAAUC,IADX;AAEf9C,EAAAA,YAAY,EAAE6C,mBAAUE,MAFT;AAGfX,EAAAA,KAAK,EAAES,mBAAUE,MAHF;AAIfvD,EAAAA,WAAW,EAAEqD,mBAAUE,MAJR;AAKfrD,EAAAA,SAAS,EAAEmD,mBAAUE,MALN;AAMfV,EAAAA,cAAc,EAAEQ,mBAAUE,MANX;AAOfpD,EAAAA,oBAAoB,EAAEkD,mBAAUE,MAPjB;AAQf;AACAjB,EAAAA,YAAY,EAAEe,mBAAUG,IATT;AAUf;AACAjB,EAAAA,YAAY,EAAEc,mBAAUG,IAXT;AAYflC,EAAAA,QAAQ,EAAE+B,mBAAUG;AAZL,C,oDAeH;AACZ7D,EAAAA,OAAO,EAAE,SADG;AAEZY,EAAAA,IAAI,EAAE;AAFM,C,0DAKM;AAClBE,EAAAA,cAAc,GAAG,CAAE,CADD;;AAElBD,EAAAA,YAAY,EAAE,EAFI;AAGlBoC,EAAAA,KAAK,EAAE,EAHW;AAIlB5C,EAAAA,WAAW,EAAE,EAJK;AAKlBsB,EAAAA,QAAQ,EAAE,IALQ;AAMlBpB,EAAAA,SAAS,EAAE,iBANO;AAOlBC,EAAAA,oBAAoB,EAAE,6BAPJ;AAQlBoC,EAAAA,YAAY,EAAE;AARI,C","sourcesContent":["/*\nCopyright 2015, 2016 OpenMarket Ltd\nCopyright 2018 New Vector Ltd\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React, {createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport {Key} from \"../../../Keyboard\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\n@replaceableComponent(\"views.elements.EditableText\")\nexport default class EditableText extends React.Component {\n    static propTypes = {\n        onValueChanged: PropTypes.func,\n        initialValue: PropTypes.string,\n        label: PropTypes.string,\n        placeholder: PropTypes.string,\n        className: PropTypes.string,\n        labelClassName: PropTypes.string,\n        placeholderClassName: PropTypes.string,\n        // Overrides blurToSubmit if true\n        blurToCancel: PropTypes.bool,\n        // Will cause onValueChanged(value, true) to fire on blur\n        blurToSubmit: PropTypes.bool,\n        editable: PropTypes.bool,\n    };\n\n    static Phases = {\n        Display: \"display\",\n        Edit: \"edit\",\n    };\n\n    static defaultProps = {\n        onValueChanged() {},\n        initialValue: '',\n        label: '',\n        placeholder: '',\n        editable: true,\n        className: \"mx_EditableText\",\n        placeholderClassName: \"mx_EditableText_placeholder\",\n        blurToSubmit: false,\n    };\n\n    constructor(props) {\n        super(props);\n\n        // we track value as an JS object field rather than in React state\n        // as React doesn't play nice with contentEditable.\n        this.value = '';\n        this.placeholder = false;\n\n        this._editable_div = createRef();\n    }\n\n    state = {\n        phase: EditableText.Phases.Display,\n    };\n\n    // TODO: [REACT-WARNING] Replace with appropriate lifecycle event\n    // eslint-disable-next-line camelcase\n    UNSAFE_componentWillReceiveProps(nextProps) {\n        if (nextProps.initialValue !== this.props.initialValue) {\n            this.value = nextProps.initialValue;\n            if (this._editable_div.current) {\n                this.showPlaceholder(!this.value);\n            }\n        }\n    }\n\n    componentDidMount() {\n        this.value = this.props.initialValue;\n        if (this._editable_div.current) {\n            this.showPlaceholder(!this.value);\n        }\n    }\n\n    showPlaceholder = show => {\n        if (show) {\n            this._editable_div.current.textContent = this.props.placeholder;\n            this._editable_div.current.setAttribute(\"class\", this.props.className\n                + \" \" + this.props.placeholderClassName);\n            this.placeholder = true;\n            this.value = '';\n        } else {\n            this._editable_div.current.textContent = this.value;\n            this._editable_div.current.setAttribute(\"class\", this.props.className);\n            this.placeholder = false;\n        }\n    };\n\n    getValue = () => this.value;\n\n    setValue = value => {\n        this.value = value;\n        this.showPlaceholder(!this.value);\n    };\n\n    edit = () => {\n        this.setState({\n            phase: EditableText.Phases.Edit,\n        });\n    };\n\n    cancelEdit = () => {\n        this.setState({\n            phase: EditableText.Phases.Display,\n        });\n        this.value = this.props.initialValue;\n        this.showPlaceholder(!this.value);\n        this.onValueChanged(false);\n        this._editable_div.current.blur();\n    };\n\n    onValueChanged = shouldSubmit => {\n        this.props.onValueChanged(this.value, shouldSubmit);\n    };\n\n    onKeyDown = ev => {\n        // console.log(\"keyDown: textContent=\" + ev.target.textContent + \", value=\" + this.value + \", placeholder=\" + this.placeholder);\n\n        if (this.placeholder) {\n            this.showPlaceholder(false);\n        }\n\n        if (ev.key === Key.ENTER) {\n            ev.stopPropagation();\n            ev.preventDefault();\n        }\n\n        // console.log(\"keyDown: textContent=\" + ev.target.textContent + \", value=\" + this.value + \", placeholder=\" + this.placeholder);\n    };\n\n    onKeyUp = ev => {\n        // console.log(\"keyUp: textContent=\" + ev.target.textContent + \", value=\" + this.value + \", placeholder=\" + this.placeholder);\n\n        if (!ev.target.textContent) {\n            this.showPlaceholder(true);\n        } else if (!this.placeholder) {\n            this.value = ev.target.textContent;\n        }\n\n        if (ev.key === Key.ENTER) {\n            this.onFinish(ev);\n        } else if (ev.key === Key.ESCAPE) {\n            this.cancelEdit();\n        }\n\n        // console.log(\"keyUp: textContent=\" + ev.target.textContent + \", value=\" + this.value + \", placeholder=\" + this.placeholder);\n    };\n\n    onClickDiv = ev => {\n        if (!this.props.editable) return;\n\n        this.setState({\n            phase: EditableText.Phases.Edit,\n        });\n    };\n\n    onFocus = ev => {\n        //ev.target.setSelectionRange(0, ev.target.textContent.length);\n\n        const node = ev.target.childNodes[0];\n        if (node) {\n            const range = document.createRange();\n            range.setStart(node, 0);\n            range.setEnd(node, node.length);\n\n            const sel = window.getSelection();\n            sel.removeAllRanges();\n            sel.addRange(range);\n        }\n    };\n\n    onFinish = (ev, shouldSubmit) => {\n        const self = this;\n        const submit = (ev.key === Key.ENTER) || shouldSubmit;\n        this.setState({\n            phase: EditableText.Phases.Display,\n        }, () => {\n            if (this.value !== this.props.initialValue) {\n                self.onValueChanged(submit);\n            }\n        });\n    };\n\n    onBlur = ev => {\n        const sel = window.getSelection();\n        sel.removeAllRanges();\n\n        if (this.props.blurToCancel) {\n            this.cancelEdit();\n        } else {\n            this.onFinish(ev, this.props.blurToSubmit);\n        }\n\n        this.showPlaceholder(!this.value);\n    };\n\n    render() {\n        const {className, editable, initialValue, label, labelClassName} = this.props;\n        let editableEl;\n\n        if (!editable || (this.state.phase === EditableText.Phases.Display &&\n            (label || labelClassName) && !this.value)\n        ) {\n            // show the label\n            editableEl = <div className={className + \" \" + labelClassName} onClick={this.onClickDiv}>\n                { label || initialValue }\n            </div>;\n        } else {\n            // show the content editable div, but manually manage its contents as react and contentEditable don't play nice together\n            editableEl = <div\n                ref={this._editable_div}\n                contentEditable={true}\n                className={className}\n                onKeyDown={this.onKeyDown}\n                onKeyUp={this.onKeyUp}\n                onFocus={this.onFocus}\n                onBlur={this.onBlur}\n            />;\n        }\n\n        return editableEl;\n    }\n}\n"]}