// Copyright 2015-2017 Parity Technologies (UK) Ltd. // This file is part of Parity. // Parity is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // Parity is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License // along with Parity. If not, see . import BigNumber from 'bignumber.js'; import { observer } from 'mobx-react'; import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { FormattedMessage } from 'react-intl'; import { connect } from 'react-redux'; import { bindActionCreators } from 'redux'; import { setRequest } from '@parity/shared/redux/providers/requestsActions'; import { ERRORS, validateAbi, validateCode, validateName, validatePositiveNumber } from '@parity/shared/util/validation'; import { deploy, deployEstimateGas } from '@parity/shared/util/tx'; import { Button, GasPriceEditor, IdentityIcon, Portal, Warning } from '@parity/ui'; import { CancelIcon } from '@parity/ui/Icons'; import DetailsStep from './DetailsStep'; import ParametersStep from './ParametersStep'; import Extras from '@parity/dapp-account/Transfer/Extras'; const STEPS = { CONTRACT_DETAILS: { title: ( ) }, CONTRACT_PARAMETERS: { title: ( ) }, EXTRAS: { title: ( ) } }; @observer class DeployContract extends Component { static contextTypes = { api: PropTypes.object.isRequired, store: PropTypes.object.isRequired }; static propTypes = { accounts: PropTypes.object.isRequired, abi: PropTypes.string, code: PropTypes.string, gasLimit: PropTypes.object.isRequired, onClose: PropTypes.func.isRequired, onSetRequest: PropTypes.func.isRequired, readOnly: PropTypes.bool, source: PropTypes.string }; static defaultProps = { readOnly: false, source: '' }; gasStore = new GasPriceEditor.Store(this.context.api, { gasLimit: this.props.gasLimit }); state = { abi: '', abiError: ERRORS.invalidAbi, amount: '0', amountValue: new BigNumber(0), amountError: '', code: '', codeError: ERRORS.invalidCode, description: '', descriptionError: null, extras: false, fromAddress: Object.keys(this.props.accounts)[0], fromAddressError: null, name: '', nameError: ERRORS.invalidName, params: [], paramsError: [], inputs: [], step: 'CONTRACT_DETAILS' }; componentWillMount () { const { abi, code } = this.props; if (abi && code) { this.setState({ abi, code }); } } componentWillReceiveProps (nextProps) { const { abi, code } = nextProps; const newState = {}; if (abi !== this.props.abi) { newState.abi = abi; } if (code !== this.props.code) { newState.code = code; } if (Object.keys(newState).length) { this.setState(newState); } } render () { const { step, inputs } = this.state; const realStepKeys = Object.keys(STEPS) .filter((k) => { if (k === 'CONTRACT_PARAMETERS') { return inputs.length > 0; } if (k === 'EXTRAS') { return this.state.extras; } return true; }); const realStep = realStepKeys.findIndex((k) => k === step); const realSteps = realStepKeys.map((k) => STEPS[k]); return ( s.title) } > { this.renderExceptionWarning() } { this.renderStep() } ); } renderExceptionWarning () { const { step } = this.state; const { errorEstimated } = this.gasStore; const realStep = Object.keys(STEPS).findIndex((k) => k === step); if (!errorEstimated || realStep >= 2) { return null; } return ( ); } renderDialogActions () { const { deployError, abiError, amountError, codeError, nameError, descriptionError, fromAddressError, fromAddress, step } = this.state; const isValid = !nameError && !fromAddressError && !descriptionError && !abiError && !codeError && !amountError; const cancelBtn = (