parent
fa95303e1e
commit
22d9e3a7e6
@ -1,57 +1,88 @@ |
|||||||
const Component = require('react').Component |
const { Component } = require('react') |
||||||
|
const PropTypes = require('prop-types') |
||||||
const h = require('react-hyperscript') |
const h = require('react-hyperscript') |
||||||
const inherits = require('util').inherits |
const classnames = require('classnames') |
||||||
const findDOMNode = require('react-dom').findDOMNode |
|
||||||
|
|
||||||
module.exports = EditableLabel |
class EditableLabel extends Component { |
||||||
|
constructor (props) { |
||||||
|
super(props) |
||||||
|
|
||||||
|
this.state = { |
||||||
|
isEditing: false, |
||||||
|
value: props.defaultValue || '', |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
handleSubmit () { |
||||||
|
const { value } = this.state |
||||||
|
|
||||||
|
if (value === '') { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
Promise.resolve(this.props.onSubmit(value)) |
||||||
|
.then(() => this.setState({ isEditing: false })) |
||||||
|
} |
||||||
|
|
||||||
inherits(EditableLabel, Component) |
saveIfEnter (event) { |
||||||
function EditableLabel () { |
if (event.key === 'Enter') { |
||||||
Component.call(this) |
this.handleSubmit() |
||||||
|
} |
||||||
} |
} |
||||||
|
|
||||||
EditableLabel.prototype.render = function () { |
renderEditing () { |
||||||
const props = this.props |
const { value } = this.state |
||||||
const state = this.state |
|
||||||
|
|
||||||
if (state && state.isEditingLabel) { |
return ([ |
||||||
return h('div.editable-label', [ |
h('input.large-input.editable-label__input', { |
||||||
h('input.sizing-input', { |
type: 'text', |
||||||
defaultValue: props.textValue, |
required: true, |
||||||
maxLength: '20', |
value: this.state.value, |
||||||
onKeyPress: (event) => { |
onKeyPress: (event) => { |
||||||
this.saveIfEnter(event) |
if (event.key === 'Enter') { |
||||||
|
this.handleSubmit() |
||||||
|
} |
||||||
}, |
}, |
||||||
|
onChange: event => this.setState({ value: event.target.value }), |
||||||
|
className: classnames({ 'editable-label__input--error': value === '' }), |
||||||
|
}), |
||||||
|
h('div.editable-label__icon-wrapper', [ |
||||||
|
h('i.fa.fa-check.editable-label__icon', { |
||||||
|
onClick: () => this.handleSubmit(), |
||||||
}), |
}), |
||||||
h('button.editable-button', { |
]), |
||||||
onClick: () => this.saveText(), |
|
||||||
}, 'Save'), |
|
||||||
]) |
]) |
||||||
} else { |
|
||||||
return h('div.name-label', { |
|
||||||
onClick: (event) => { |
|
||||||
const nameAttribute = event.target.getAttribute('name') |
|
||||||
// checks for class to handle smaller CTA above the account name
|
|
||||||
const classAttribute = event.target.getAttribute('class') |
|
||||||
if (nameAttribute === 'edit' || classAttribute === 'edit-text') { |
|
||||||
this.setState({ isEditingLabel: true }) |
|
||||||
} |
|
||||||
}, |
|
||||||
}, this.props.children) |
|
||||||
} |
} |
||||||
|
|
||||||
|
renderReadonly () { |
||||||
|
return ([ |
||||||
|
h('div.editable-label__value', this.state.value), |
||||||
|
h('div.editable-label__icon-wrapper', [ |
||||||
|
h('i.fa.fa-pencil.editable-label__icon', { |
||||||
|
onClick: () => this.setState({ isEditing: true }), |
||||||
|
}), |
||||||
|
]), |
||||||
|
]) |
||||||
} |
} |
||||||
|
|
||||||
EditableLabel.prototype.saveIfEnter = function (event) { |
render () { |
||||||
if (event.key === 'Enter') { |
const { isEditing } = this.state |
||||||
this.saveText() |
const { className } = this.props |
||||||
|
|
||||||
|
return ( |
||||||
|
h('div.editable-label', { className: classnames(className) }, |
||||||
|
isEditing |
||||||
|
? this.renderEditing() |
||||||
|
: this.renderReadonly() |
||||||
|
) |
||||||
|
) |
||||||
} |
} |
||||||
} |
} |
||||||
|
|
||||||
EditableLabel.prototype.saveText = function () { |
EditableLabel.propTypes = { |
||||||
// eslint-disable-next-line react/no-find-dom-node
|
onSubmit: PropTypes.func.isRequired, |
||||||
var container = findDOMNode(this) |
defaultValue: PropTypes.string, |
||||||
var text = container.querySelector('.editable-label input').value |
className: PropTypes.string, |
||||||
var truncatedText = text.substring(0, 20) |
|
||||||
this.props.saveText(truncatedText) |
|
||||||
this.setState({ isEditingLabel: false, textLabel: truncatedText }) |
|
||||||
} |
} |
||||||
|
|
||||||
|
module.exports = EditableLabel |
||||||
|
@ -0,0 +1,34 @@ |
|||||||
|
.editable-label { |
||||||
|
display: flex; |
||||||
|
align-items: center; |
||||||
|
justify-content: center; |
||||||
|
position: relative; |
||||||
|
|
||||||
|
&__value { |
||||||
|
max-width: 250px; |
||||||
|
overflow: hidden; |
||||||
|
white-space: nowrap; |
||||||
|
text-overflow: ellipsis; |
||||||
|
} |
||||||
|
|
||||||
|
&__input { |
||||||
|
width: 250px; |
||||||
|
font-size: 14px; |
||||||
|
text-align: center; |
||||||
|
|
||||||
|
&--error { |
||||||
|
border: 1px solid $monzo; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
&__icon-wrapper { |
||||||
|
position: absolute; |
||||||
|
margin-left: 10px; |
||||||
|
left: 100%; |
||||||
|
} |
||||||
|
|
||||||
|
&__icon { |
||||||
|
cursor: pointer; |
||||||
|
color: $dusty-gray; |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue