// Copyright 2015, 2016 Ethcore (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 React, { Component, PropTypes } from 'react'; import { connect } from 'react-redux'; import { bindActionCreators } from 'redux'; import { observer } from 'mobx-react'; import ActionDoneAll from 'material-ui/svg-icons/action/done-all'; import ContentClear from 'material-ui/svg-icons/content/clear'; import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back'; import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward'; import { BusyStep, Button, CompletedStep, GasPriceEditor, IdentityIcon, Modal, TxHash } from '~/ui'; import { MAX_GAS_ESTIMATION } from '~/util/constants'; import { validateAddress, validateUint } from '~/util/validation'; import { parseAbiType } from '~/util/abi'; import DetailsStep from './DetailsStep'; import { ERROR_CODES } from '~/api/transport/error'; const STEP_DETAILS = 0; const STEP_BUSY_OR_GAS = 1; const STEP_BUSY = 2; const TITLES = { transfer: 'function details', sending: 'sending', complete: 'complete', gas: 'gas selection', rejected: 'rejected' }; const STAGES_BASIC = [TITLES.transfer, TITLES.sending, TITLES.complete]; const STAGES_GAS = [TITLES.transfer, TITLES.gas, TITLES.sending, TITLES.complete]; @observer class ExecuteContract extends Component { static contextTypes = { api: PropTypes.object.isRequired, store: PropTypes.object.isRequired } static propTypes = { isTest: PropTypes.bool, fromAddress: PropTypes.string, accounts: PropTypes.object, contract: PropTypes.object, gasLimit: PropTypes.object.isRequired, onClose: PropTypes.func.isRequired, onFromAddressChange: PropTypes.func.isRequired } gasStore = new GasPriceEditor.Store(this.context.api, this.props.gasLimit); state = { amount: '0', amountError: null, busyState: null, fromAddressError: null, func: null, funcError: null, gasEdit: false, rejected: false, step: STEP_DETAILS, sending: false, values: [], valuesError: [], txhash: null } componentDidMount () { const { contract } = this.props; const functions = contract.functions .filter((func) => !func.constant) .sort((a, b) => (a.name || '').localeCompare(b.name || '')); this.onFuncChange(null, functions[0]); } componentWillReceiveProps (newProps) { if (newProps.fromAddress !== this.props.fromAddress) { this.estimateGas(newProps.fromAddress); } } render () { const { sending, step, gasEdit, rejected } = this.state; const steps = gasEdit ? STAGES_GAS : STAGES_BASIC; if (rejected) { steps[steps.length - 1] = TITLES.rejected; } return ( { this.renderStep() } ); } renderDialogActions () { const { onClose, fromAddress } = this.props; const { gasEdit, sending, step, fromAddressError, valuesError } = this.state; const hasError = fromAddressError || valuesError.find((error) => error); const cancelBtn = (