// 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 { api } from '../parity'; import { callRegister, postRegister } from '../services'; import Button from '../Button'; import IdentityIcon from '../IdentityIcon'; import styles from './import.css'; export default class Import extends Component { static propTypes = { accounts: PropTypes.object.isRequired, fromAddress: PropTypes.string.isRequired, instance: PropTypes.object.isRequired, visible: PropTypes.bool.isRequired, onClose: PropTypes.func.isRequired, onSetFromAddress: PropTypes.func.isRequired } state = { abi: null, abiParsed: null, abiError: 'Please add a valid ABI definition', functions: null, fnstate: {} } render () { const { visible, onClose } = this.props; const { abiError } = this.state; if (!visible) { return null; } return (
abi import
{ abiError ? this.renderCapture() : this.renderRegister() }
); } renderCapture () { const { abiError } = this.state; return (
Provide the ABI (Contract Interface) in the space provided below. Only non-constant functions (names & types) will be imported, while constant functions and existing signatures will be ignored.
{ abiError }
); } renderRegister () { const { accounts, fromAddress } = this.props; const account = accounts[fromAddress]; const count = this.countFunctions(); let buttons = null; if (count) { buttons = (
); } return (
The following functions have been extracted from the ABI provided and the state has been determined from interacting with the signature contract.
{ this.renderFunctions() }
{ count || 'no' } functions available for registration
{ buttons }
); } renderFunctions () { const { functions, fnstate } = this.state; if (!functions) { return null; } return functions.map((fn) => { if (fn.constant) { fnstate[fn.signature] = 'fnconstant'; } else if (!fnstate[fn.signature]) { this.testFunction(fn); } return (
{ fn.id }
); }); } sortFunctions = (a, b) => { return a.name.localeCompare(b.name); } countFunctions () { const { functions, fnstate } = this.state; if (!functions) { return 0; } return functions.filter((fn) => fnstate[fn.signature] === 'fntodo').length; } testFunction (fn) { const { instance } = this.props; const { fnstate } = this.state; callRegister(instance, fn.id) .then((result) => { fnstate[fn.signature] = result ? 'fntodo' : 'fnexists'; this.setState(fnstate); }) .catch((error) => { console.error(error); }); } onAbiEdit = (event) => { let functions = null; let abiError = null; let abiParsed = null; let abi = null; try { abiParsed = JSON.parse(event.target.value); functions = api.newContract(abiParsed).functions.sort(this.sortFunctions); abi = JSON.stringify(abiParsed); } catch (error) { console.error('onAbiEdit', error); abiError = error.message; } this.setState({ functions, abiError, abiParsed, abi }); } onRegister = () => { const { instance, fromAddress, onClose } = this.props; const { functions, fnstate } = this.state; Promise .all( functions .filter((fn) => !fn.constant) .filter((fn) => fnstate[fn.signature] === 'fntodo') .map((fn) => postRegister(instance, fn.id, { from: fromAddress })) ) .then(() => { onClose(); }) .catch((error) => { console.error('onRegister', error); }); } onSelectFromAddress = () => { const { accounts, fromAddress, onSetFromAddress } = this.props; const addresses = Object.keys(accounts); let index = 0; addresses.forEach((address, _index) => { if (address === fromAddress) { index = _index; } }); index++; if (index >= addresses.length) { index = 0; } onSetFromAddress(addresses[index]); } }