diff --git a/ui/app/components/gas-customization/gas-modal-page-container/advanced-tab-content/advanced-tab-content.component.js b/ui/app/components/gas-customization/gas-modal-page-container/advanced-tab-content/advanced-tab-content.component.js index 44aba358c..4dd18ce2b 100644 --- a/ui/app/components/gas-customization/gas-modal-page-container/advanced-tab-content/advanced-tab-content.component.js +++ b/ui/app/components/gas-customization/gas-modal-page-container/advanced-tab-content/advanced-tab-content.component.js @@ -97,7 +97,7 @@ export default class AdvancedTabContent extends Component { updateCustomGasLimit ) }
Live Gas Price Predictions
- +
Slower Faster diff --git a/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.component.js b/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.component.js index de14e1b38..8f23b22e0 100644 --- a/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.component.js +++ b/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.component.js @@ -47,6 +47,7 @@ export default class GasModalPageContainer extends Component { customGasLimit, newTotalFiat, gasChartProps, + currentTimeEstimate, }) { const { transactionFee } = this.props return ( @@ -55,7 +56,7 @@ export default class GasModalPageContainer extends Component { updateCustomGasLimit={convertThenUpdateCustomGasLimit} customGasPrice={customGasPrice} customGasLimit={customGasLimit} - timeRemaining="1 min 31 sec" + timeRemaining={currentTimeEstimate} transactionFee={transactionFee} totalFee={newTotalFiat} gasChartProps={gasChartProps} diff --git a/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.container.js b/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.container.js index 84eae1880..3a62d21cc 100644 --- a/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.container.js +++ b/ui/app/components/gas-customization/gas-modal-page-container/gas-modal-page-container.container.js @@ -10,6 +10,7 @@ import { setCustomGasPrice, setCustomGasLimit, resetCustomData, + setCustomTimeEstimate, } from '../../../ducks/gas.duck' import { hideGasButtonGroup, @@ -29,6 +30,7 @@ import { getBasicGasEstimateLoadingStatus, getAveragePriceEstimateInHexWEI, getDefaultActiveButtonIndex, + formatTimeEstimate, } from '../../../selectors/custom-gas' import { formatCurrency, @@ -65,20 +67,24 @@ const mapStateToProps = state => { const hideBasic = state.appState.modal.modalState.props.hideBasic + const customGasPrice = calcCustomGasPrice(customModalGasPriceInHex) + return { hideBasic, isConfirm: isConfirm(state), customModalGasPriceInHex, customModalGasLimitInHex, - customGasPrice: calcCustomGasPrice(customModalGasPriceInHex), + customGasPrice, customGasLimit: calcCustomGasLimit(customModalGasLimitInHex), newTotalFiat, + currentTimeEstimate: getRenderableTimeEstimate(customGasPrice, state.gas.priceAndTimeEstimates), gasPriceButtonGroupProps: { buttonDataLoading, defaultActiveButtonIndex: getDefaultActiveButtonIndex(gasButtonInfo, customModalGasPriceInHex), gasButtonInfo, }, gasChartProps: { + currentPrice: customGasPrice, priceAndTimeEstimates: state.gas.priceAndTimeEstimates, }, infoRowProps: { @@ -111,6 +117,7 @@ const mapDispatchToProps = dispatch => { return dispatch(updateGasAndCalculate({ gasLimit, gasPrice })) }, hideGasButtonGroup: () => dispatch(hideGasButtonGroup()), + setCustomTimeEstimate: (timeEstimateInSeconds) => dispatch(setCustomTimeEstimate(timeEstimateInSeconds)), } } @@ -181,3 +188,25 @@ function addHexWEIsToRenderableFiat (aHexWEI, bHexWEI, convertedCurrency, conver partialRight(formatCurrency, [convertedCurrency]), )(aHexWEI, bHexWEI) } + +function getRenderableTimeEstimate (currentGasPrice, priceAndTimeEstimates) { + const gasPrices = priceAndTimeEstimates.map(({ gasprice }) => gasprice) + const estimatedTimes = priceAndTimeEstimates.map(({ expectedTime }) => expectedTime) + + const closestLowerValueIndex = gasPrices.findIndex((e, i, a) => { + return e <= currentGasPrice && a[i + 1] >= currentGasPrice + }) + const closestHigherValueIndex = gasPrices.findIndex((e, i, a) => { + return e > currentGasPrice + }) + + const closestLowerValue = gasPrices[closestLowerValueIndex] + const closestHigherValue = gasPrices[closestHigherValueIndex] + const estimatedClosestLowerTimeEstimate = estimatedTimes[closestLowerValueIndex] + const estimatedClosestHigherTimeEstimate = estimatedTimes[closestHigherValueIndex] + + const slope = (estimatedClosestHigherTimeEstimate - estimatedClosestLowerTimeEstimate) / (closestHigherValue - closestLowerValue) + const newTimeEstimate = -1 * (slope * (closestHigherValue - currentGasPrice) - estimatedClosestHigherTimeEstimate) + + return formatTimeEstimate(newTimeEstimate) +} diff --git a/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-component.test.js b/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-component.test.js index bb1a28136..61871f5f3 100644 --- a/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-component.test.js +++ b/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-component.test.js @@ -65,6 +65,7 @@ describe('GasModalPageContainer Component', function () { customGasLimit={54321} gasPriceButtonGroupProps={mockGasPriceButtonGroupProps} infoRowProps={mockInfoRowProps} + currentTimeEstimate={'1 min 31 sec'} customGasPriceInHex={'mockCustomGasPriceInHex'} customGasLimitInHex={'mockCustomGasLimitInHex'} />, { context: { t: (str1, str2) => str2 ? str1 + str2 : str1 } }) @@ -199,6 +200,7 @@ describe('GasModalPageContainer Component', function () { customGasPrice: 123, customGasLimit: 456, newTotalFiat: '$0.30', + currentTimeEstimate: '1 min 31 sec', }) const advancedTabContentProps = renderAdvancedTabContentResult.props assert.equal(advancedTabContentProps.updateCustomGasPrice(), 'mockConvertThenUpdateCustomGasPrice') diff --git a/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-container.test.js b/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-container.test.js index c16a07b76..3f3be8d0c 100644 --- a/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-container.test.js +++ b/ui/app/components/gas-customization/gas-modal-page-container/tests/gas-modal-page-container-container.test.js @@ -75,7 +75,12 @@ describe('gas-modal-page-container container', () => { limit: 'aaaaaaaa', price: 'ffffffff', }, - priceAndTimeEstimates: 'mockPriceAndTimeEstimates', + priceAndTimeEstimates: [ + { gasprice: 3, expectedTime: '31' }, + { gasprice: 4, expectedTime: '62' }, + { gasprice: 5, expectedTime: '93' }, + { gasprice: 6, expectedTime: '124' }, + ], }, confirmTransaction: { txData: { @@ -93,11 +98,18 @@ describe('gas-modal-page-container container', () => { isConfirm: true, customGasPrice: 4.294967295, customGasLimit: 2863311530, + currentTimeEstimate: '~1 min 11 sec', newTotalFiat: '637.41', customModalGasLimitInHex: 'aaaaaaaa', customModalGasPriceInHex: 'ffffffff', gasChartProps: { - priceAndTimeEstimates: 'mockPriceAndTimeEstimates', + 'currentPrice': 4.294967295, + priceAndTimeEstimates: [ + { gasprice: 3, expectedTime: '31' }, + { gasprice: 4, expectedTime: '62' }, + { gasprice: 5, expectedTime: '93' }, + { gasprice: 6, expectedTime: '124' }, + ], }, gasPriceButtonGroupProps: { buttonDataLoading: 'mockBasicGasEstimateLoadingStatus:4', diff --git a/ui/app/components/gas-customization/gas-price-chart/gas-price-chart.component.js b/ui/app/components/gas-customization/gas-price-chart/gas-price-chart.component.js index 85893f771..69bbd12f6 100644 --- a/ui/app/components/gas-customization/gas-price-chart/gas-price-chart.component.js +++ b/ui/app/components/gas-customization/gas-price-chart/gas-price-chart.component.js @@ -35,6 +35,43 @@ function appendOrUpdateCircle ({ circle, data, itemIndex, cx, cy, cssId, appendO } } +function setSelectedCircle ({ chart, gasPrices, currentPrice, chartXStart, chartWidth }) { + const numberOfValues = chart.internal.data.xs.data1.length + const closestLowerValueIndex = gasPrices.findIndex((e, i, a) => { + return e <= currentPrice && a[i + 1] >= currentPrice + }) + const closestHigherValueIndex = gasPrices.findIndex((e, i, a) => { + return e > currentPrice + }) + const closestHigherValue = gasPrices[closestHigherValueIndex] + const closestLowerValue = gasPrices[closestLowerValueIndex] + + if (closestHigherValue && closestLowerValue) { + const closestLowerCircle = d3.select(`.c3-circle-${closestLowerValueIndex}`) + const closestHigherCircle = d3.select(`.c3-circle-${closestHigherValueIndex}`) + const { x: lowerX, y: lowerY } = closestLowerCircle.node().getBoundingClientRect() + const { x: higherX, y: higherY } = closestHigherCircle.node().getBoundingClientRect() + const currentX = lowerX + (higherX - lowerX) * (currentPrice - closestLowerValue) / (closestHigherValue - closestLowerValue) + const slope = (higherY - lowerY) / (higherX - lowerX) + const newTimeEstimate = -1 * (slope * (higherX - currentX) - higherY) + chart.internal.selectPointB({ + x: currentX, + value: newTimeEstimate, + id: 'data1', + index: numberOfValues, + name: 'data1', + }, numberOfValues) + } else { + const setCircle = d3.select('#set-circle') + if (!setCircle.empty()) { + setCircle.remove() + } + d3.select('.c3-tooltip-container').style('display', 'none !important') + chart.internal.hideXGridFocus() + return + } +} + export default class GasPriceChart extends Component { static contextTypes = { t: PropTypes.func, @@ -42,12 +79,15 @@ export default class GasPriceChart extends Component { static propTypes = { priceAndTimeEstimates: PropTypes.array, + currentPrice: PropTypes.number, + updateCustomGasPrice: PropTypes.func, } - renderChart (priceAndTimeEstimates) { + renderChart (currentPrice, priceAndTimeEstimates, updateCustomGasPrice) { const gasPrices = priceAndTimeEstimates.map(({ gasprice }) => gasprice) const gasPricesMax = gasPrices[gasPrices.length - 1] + 1 const estimatedTimes = priceAndTimeEstimates.map(({ expectedTime }) => expectedTime) + const estimatedTimesMax = estimatedTimes[0] const chart = c3.generate({ size: { @@ -187,6 +227,28 @@ export default class GasPriceChart extends Component { }) } + chart.internal.selectPointB = function (data, itemIndex = (data.index || 0)) { + const { x: chartXStart, y: chartYStart } = d3.select('.c3-areas-data1') + .node() + .getBoundingClientRect() + + d3.select('#set-circle').remove() + + const circle = this.main + .select('.' + 'c3-selected-circles' + this.getTargetSelectorSuffix(data.id)) + .selectAll('.' + 'c3-selected-circle' + '-' + itemIndex) + + appendOrUpdateCircle.bind(this)({ + circle, + data, + itemIndex, + cx: () => data.x - chartXStart + 11, + cy: () => data.value - chartYStart + 10, + cssId: 'set-circle', + appendOnly: true, + }) + } + chart.internal.overlayPoint = function (data, itemIndex) { const circle = this.main .select('.' + 'c3-selected-circles' + this.getTargetSelectorSuffix(data.id)) @@ -238,13 +300,13 @@ export default class GasPriceChart extends Component { setTimeout(function () { setTickPosition('y', 0, -5, 8) - setTickPosition('y', 1, -3) - setTickPosition('x', 0, 3, 20) - setTickPosition('x', 1, 3, -10) + setTickPosition('y', 1, -3, -5) + setTickPosition('x', 0, 3, 15) + setTickPosition('x', 1, 3, -8) // TODO: Confirm the below constants work with all data sets and screen sizes d3.select('.c3-axis-x-label').attr('transform', 'translate(0,-15)') - d3.select('.c3-axis-y-label').attr('transform', 'translate(32, 2) rotate(-90)') + d3.select('.c3-axis-y-label').attr('transform', 'translate(52, 2) rotate(-90)') d3.select('.c3-xgrid-focus line').attr('y2', 98) d3.select('.c3-chart').on('mouseout', () => { @@ -262,6 +324,7 @@ export default class GasPriceChart extends Component { const overlayedCircle = d3.select('#overlayed-circle') const numberOfValues = chart.internal.data.xs.data1.length const { x: circleX, y: circleY } = overlayedCircle.node().getBoundingClientRect() + const { x: xData } = overlayedCircle.datum() chart.internal.selectPoint({ x: circleX - chartXStart, value: circleY - 1.5, @@ -269,13 +332,15 @@ export default class GasPriceChart extends Component { index: numberOfValues, name: 'data1', }, numberOfValues) + updateCustomGasPrice(xData) }) + setSelectedCircle({ chart, gasPrices, currentPrice, chartXStart, chartWidth }) + d3.select('.c3-chart').on('mousemove', function () { const chartMouseXPos = d3.event.clientX - chartXStart const posPercentile = chartMouseXPos / chartWidth - const currentPosValue = (gasPrices[gasPrices.length - 1] - gasPrices[0]) * posPercentile + gasPrices[0] const closestLowerValueIndex = gasPrices.findIndex((e, i, a) => { return e <= currentPosValue && a[i + 1] >= currentPosValue @@ -326,11 +391,26 @@ export default class GasPriceChart extends Component { }) }, 0) + this.chart = chart + } + componentDidUpdate (prevProps) { + if (prevProps.currentPrice !== this.props.currentPrice) { + const chartRect = d3.select('.c3-areas-data1') + const { x: chartXStart, width: chartWidth } = chartRect.node().getBoundingClientRect() + setSelectedCircle({ + chart: this.chart, + currentPrice: this.props.currentPrice, + gasPrices: this.props.priceAndTimeEstimates.map(({ gasprice }) => gasprice), + chartXStart, + chartWidth, + }) + } } componentDidMount () { - this.renderChart(this.props.priceAndTimeEstimates) + const { currentPrice, priceAndTimeEstimates, updateCustomGasPrice } = this.props + this.renderChart(currentPrice, priceAndTimeEstimates, updateCustomGasPrice) } render () { diff --git a/ui/app/components/gas-customization/gas-price-chart/tests/gas-price-chart.component.test.js b/ui/app/components/gas-customization/gas-price-chart/tests/gas-price-chart.component.test.js index ae98659cc..48b8a6525 100644 --- a/ui/app/components/gas-customization/gas-price-chart/tests/gas-price-chart.component.test.js +++ b/ui/app/components/gas-customization/gas-price-chart/tests/gas-price-chart.component.test.js @@ -10,6 +10,9 @@ const mockSelectReturn = { node: () => ({ getBoundingClientRect: () => ({ x: 123, y: 321, width: 400 }), }), + empty: sinon.spy(), + remove: sinon.spy(), + style: sinon.spy(), select: d3.select, attr: sinon.spy(), on: sinon.spy(), @@ -17,11 +20,17 @@ const mockSelectReturn = { const GasPriceChart = proxyquire('../gas-price-chart.component.js', { 'c3': { - generate: function () { + generate: function ({ data: { columns } }) { return { internal: { showTooltip: () => {}, showXGridFocus: () => {}, + hideXGridFocus: () => {}, + data: { + xs: { + [columns[1][0]]: columns[1].slice(1), + }, + }, }, } }, diff --git a/ui/app/ducks/gas.duck.js b/ui/app/ducks/gas.duck.js index 35e26544a..2dcec91de 100644 --- a/ui/app/ducks/gas.duck.js +++ b/ui/app/ducks/gas.duck.js @@ -212,7 +212,7 @@ export function fetchGasEstimates (blockTime) { const estimatedPricesAndTimes = r.map(({ expectedTime, expectedWait, gasprice }) => ({ expectedTime, expectedWait, gasprice })) const estimatedTimeWithUniquePrices = uniqBy(({ expectedTime }) => expectedTime, estimatedPricesAndTimes) const timeMappedToSeconds = estimatedTimeWithUniquePrices.map(({ expectedWait, gasprice }) => { - const expectedTime = (new BigNumber(expectedWait)).times(Number(blockTime), 10).div(60, 10).toString(10) + const expectedTime = (new BigNumber(expectedWait)).times(Number(blockTime), 10).toString(10) return { expectedTime, expectedWait, diff --git a/ui/app/selectors/custom-gas.js b/ui/app/selectors/custom-gas.js index d5a3972df..abc8ba191 100644 --- a/ui/app/selectors/custom-gas.js +++ b/ui/app/selectors/custom-gas.js @@ -31,6 +31,7 @@ const selectors = { getAveragePriceEstimateInHexWEI, getDefaultActiveButtonIndex, priceEstimateToWei, + formatTimeEstimate, } module.exports = selectors