// Copyright 2015, 2016 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 { pick, omitBy } from 'lodash'; import { observer } from 'mobx-react'; import React, { Component, PropTypes } from 'react'; import { FormattedMessage } from 'react-intl'; import { connect } from 'react-redux'; import { BusyStep, Button, CompletedStep, CopyToClipboard, GasPriceEditor, IdentityIcon, Modal, TxHash, Warning } from '~/ui'; import { CancelIcon, DoneIcon } from '~/ui/Icons'; import { ERRORS, validateAbi, validateCode, validateName } from '~/util/validation'; import DetailsStep from './DetailsStep'; import ParametersStep from './ParametersStep'; import ErrorStep from './ErrorStep'; import styles from './deployContract.css'; import { ERROR_CODES } from '~/api/transport/error'; const STEPS = { CONTRACT_DETAILS: { title: ( ) }, CONTRACT_PARAMETERS: { title: ( ) }, DEPLOYMENT: { waiting: true, title: ( ) }, COMPLETED: { 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, balances: PropTypes.object, code: PropTypes.string, gasLimit: PropTypes.object.isRequired, onClose: 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, code: '', codeError: ERRORS.invalidCode, deployState: '', deployError: null, description: '', descriptionError: null, fromAddress: Object.keys(this.props.accounts)[0], fromAddressError: null, name: '', nameError: ERRORS.invalidName, params: [], paramsError: [], inputs: [], rejected: false, 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, deployError, rejected, inputs } = this.state; const realStep = Object.keys(STEPS).findIndex((k) => k === step); const realSteps = deployError || rejected ? null : Object.keys(STEPS) .filter((k) => k !== 'CONTRACT_PARAMETERS' || inputs.length > 0) .map((k) => STEPS[k]); const title = realSteps ? null : (deployError ? : ); const waiting = realSteps ? realSteps.map((s, i) => s.waiting ? i : false).filter((v) => v !== false) : null; return ( s.title) : null } title={ title } visible waiting={ waiting }> { 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, codeError, nameError, descriptionError, fromAddressError, fromAddress, step } = this.state; const isValid = !nameError && !fromAddressError && !descriptionError && !abiError && !codeError; const cancelBtn = (