const Component = require('react').Component const h = require('react-hyperscript') const inherits = require('util').inherits const ethUtil = require('ethereumjs-util') const BN = ethUtil.BN const extend = require('xtend') module.exports = BnAsDecimalInput inherits(BnAsDecimalInput, Component) function BnAsDecimalInput () { this.state = { invalid: null } Component.call(this) } /* Bn as Decimal Input * * A component for allowing easy, decimal editing * of a passed in bn string value. * * On change, calls back its `onChange` function parameter * and passes it an updated bn string. */ BnAsDecimalInput.prototype.render = function () { const props = this.props const state = this.state const { value, precision, onChange, min, max } = props const suffix = props.suffix const style = props.style const scale = Math.pow(10, precision) const newValue = value.toNumber(10) / scale return ( h('.flex-column', [ h('.flex-row', { style: { alignItems: 'flex-end', lineHeight: '13px', fontFamily: 'Montserrat Light', textRendering: 'geometricPrecision', }, }, [ h('input.hex-input', { type: 'number', step: 'any', required: true, min: min, max: max, style: extend({ display: 'block', textAlign: 'right', backgroundColor: 'transparent', border: '1px solid #bdbdbd', }, style), value: newValue, onBlur: (event) => { this.updateValidity(event) }, onChange: (event) => { this.updateValidity(event) const value = (event.target.value === '') ? '' : event.target.value const scaledNumber = Math.floor(scale * value) const precisionBN = new BN(scaledNumber, 10) onChange(precisionBN) }, onInvalid: (event) => { const msg = this.constructWarning() if (msg === state.invalid) { return } this.setState({ invalid: msg }) event.preventDefault() return false }, }), h('div', { style: { color: ' #AEAEAE', fontSize: '12px', marginLeft: '5px', marginRight: '6px', width: '20px', }, }, suffix), ]), state.invalid ? h('span.error', { style: { position: 'absolute', right: '0px', textAlign: 'right', transform: 'translateY(26px)', padding: '3px', background: 'rgba(255,255,255,0.85)', zIndex: '1', textTransform: 'capitalize', border: '2px solid #E20202', }, }, state.invalid) : null, ]) ) } BnAsDecimalInput.prototype.setValid = function (message) { this.setState({ invalid: null }) } BnAsDecimalInput.prototype.updateValidity = function (event) { const target = event.target const value = this.props.value const newValue = target.value if (value === newValue) { return } const valid = target.checkValidity() if (valid) { this.setState({ invalid: null }) } } BnAsDecimalInput.prototype.constructWarning = function () { const { name, min, max } = this.props let message = name ? name + ' ' : '' if (min && max) { message += `must be greater than or equal to ${min} and less than or equal to ${max}.` } else if (min) { message += `must be greater than or equal to ${min}.` } else if (max) { message += `must be less than or equal to ${max}.` } else { message += 'Invalid input.' } return message }