Show result

This commit is contained in:
Anthony Laibe 2018-08-16 14:13:36 +01:00 committed by Pascal Precht
parent 3c9cea594c
commit 870efbcf20
No known key found for this signature in database
GPG Key ID: 0EE28D8D6FD85D7D
4 changed files with 70 additions and 42 deletions

View File

@ -5,15 +5,31 @@ import {
Grid, Grid,
Form, Form,
Button, Button,
List,
Card Card
} from "tabler-react"; } from "tabler-react";
class ContractFunction extends Component { class ContractFunction extends Component {
constructor(props) { constructor(props) {
super(props) super(props);
this.state = { inputs: {} }; this.state = { inputs: {} };
} }
buttonTitle() {
const { method } =this.props;
if (method.name === 'constructor') {
return 'Deploy';
}
return (method.mutability === 'view' || method.mutability === 'pure') ? 'Call' : 'Send';
}
inputsAsArray(){
return this.props.method.inputs
.map(input => this.state.inputs[input.name])
.filter(value => value);
}
handleChange(e, name) { handleChange(e, name) {
let newInputs = this.state.inputs; let newInputs = this.state.inputs;
newInputs[name] = e.target.value; newInputs[name] = e.target.value;
@ -22,8 +38,11 @@ class ContractFunction extends Component {
handleCall(e) { handleCall(e) {
e.preventDefault(); e.preventDefault();
const inputs = this.props.method.inputs.map(input => this.state.inputs[input.name]); this.props.postContractFunction(this.props.contractProfile.name, this.props.method.name, this.inputsAsArray());
this.props.postContractFunction(this.props.contractProfile.name, this.props.method.name, inputs); }
callDisabled() {
return this.inputsAsArray().length !== this.props.method.inputs.length;
} }
render() { render() {
@ -34,17 +53,25 @@ class ContractFunction extends Component {
<Card.Header> <Card.Header>
<Card.Title>{this.props.method.name}</Card.Title> <Card.Title>{this.props.method.name}</Card.Title>
</Card.Header> </Card.Header>
{this.props.method.inputs.length > 0 &&
<Card.Body> <Card.Body>
{this.props.method.inputs.map(input => ( {this.props.method.inputs.map(input => (
<Form.Group key={input.name} label={input.name}> <Form.Group key={input.name} label={input.name}>
<Form.Input placeholder={input.type} onChange={(e) => this.handleChange(e, input.name)}/> <Form.Input placeholder={input.type} onChange={(e) => this.handleChange(e, input.name)}/>
</Form.Group> </Form.Group>
))} ))}
<Button color="primary" disabled={this.callDisabled()} onClick={(e) => this.handleCall(e)}>
{this.buttonTitle()}
</Button>
</Card.Body> </Card.Body>
}
<Card.Footer> <Card.Footer>
<Button color="primary" onClick={(e) => this.handleCall(e)}>Call</Button> <List>
{this.props.contractFunctions.map(contractFunction => (
<List.Item key={contractFunction.result}>
{contractFunction.inputs.length > 0 && <p>Inputs: {contractFunction.inputs.join(', ')}</p>}
<strong>Result: {contractFunction.result}</strong>
</List.Item>
))}
</List>
</Card.Footer> </Card.Footer>
</Card> </Card>
</Grid.Col> </Grid.Col>
@ -53,7 +80,6 @@ class ContractFunction extends Component {
} }
} }
ContractFunction.propTypes = { ContractFunction.propTypes = {
contractProfile: PropTypes.object, contractProfile: PropTypes.object,
method: PropTypes.object, method: PropTypes.object,
@ -61,22 +87,40 @@ ContractFunction.propTypes = {
postContractFunction: PropTypes.func postContractFunction: PropTypes.func
}; };
const filterContractFunctions = (contractFunctions, contractName, method) => {
return contractFunctions.filter((contractFunction) => (
contractFunction.contractName === contractName && contractFunction.method === method
));
};
const ContractFunctions = (props) => { const ContractFunctions = (props) => {
const {contractProfile} = props; const {contractProfile} = props;
return ( return (
<Page.Content title={contractProfile.name + ' Functions'}> <Page.Content title={contractProfile.name + ' Functions'}>
{contractProfile.methods {contractProfile.methods
.filter(method => method.name !== 'constructor') .filter((method) => {
.map(method => <ContractFunction key={method.name} method={method} {...props} />)} return props.onlyConstructor ? method.name === 'constructor' : method.name !== 'constructor';
})
.map(method => <ContractFunction key={method.name}
method={method}
contractFunctions={filterContractFunctions(props.contractFunctions, contractProfile.name, method.name)}
contractProfile={contractProfile}
postContractFunction={props.postContractFunction} />)}
</Page.Content> </Page.Content>
) );
}; };
ContractFunctions.propTypes = { ContractFunctions.propTypes = {
onlyConstructor: PropTypes.bool,
contractProfile: PropTypes.object, contractProfile: PropTypes.object,
contractFunctions: PropTypes.arrayOf(PropTypes.object), contractFunctions: PropTypes.arrayOf(PropTypes.object),
postContractFunction: PropTypes.func postContractFunction: PropTypes.func
}; };
ContractFunctions.defaultProps = {
onlyConstructor: false
};
export default ContractFunctions; export default ContractFunctions;

View File

@ -20,6 +20,7 @@ class ContractFunctionsContainer extends Component {
render={({contractProfile, contractFunctions, postContractFunction}) => ( render={({contractProfile, contractFunctions, postContractFunction}) => (
<ContractFunctions contractProfile={contractProfile} <ContractFunctions contractProfile={contractProfile}
contractFunctions={contractFunctions} contractFunctions={contractFunctions}
constructor={true}
postContractFunction={postContractFunction}/> postContractFunction={postContractFunction}/>
)} /> )} />
); );

View File

@ -103,21 +103,25 @@ class ContractsManager {
account: (callback) => { account: (callback) => {
self.events.request("blockchain:defaultAccount:get", (account) => callback(null, account)); self.events.request("blockchain:defaultAccount:get", (account) => callback(null, account));
} }
}, function (err, result) { }, function (error, result) {
if (err) { if (error) {
return res.send({error: err.message}); return res.send({error: error.message});
} }
const {account, contract} = result; const {account, contract} = result;
const contractObj = new web3.eth.Contract(contract.abiDefinition, contract.deployedAddress); const contractObj = new web3.eth.Contract(contract.abiDefinition, contract.deployedAddress);
const abi = contract.abiDefinition.find(definition => definition.name === req.body.method); const abi = contract.abiDefinition.find(definition => definition.name === req.body.method);
const funcCall = (abi.constant === true || abi.stateMutability === 'view' || abi.stateMutability === 'pure') ? 'call' : 'send'; const funcCall = (abi.constant === true || abi.stateMutability === 'view' || abi.stateMutability === 'pure') ? 'call' : 'send';
contractObj.methods[req.body.method].apply(this, req.body.inputs)[funcCall]({from: account}, (err, result) => { try {
if (err) { contractObj.methods[req.body.method].apply(this, req.body.inputs)[funcCall]({from: account}, (error, result) => {
return res.send({error: err.message}); if (error) {
return res.send({result: error.message});
} }
res.send({result: result}); return res.send({result});
}); });
} catch (e) {
return res.send({result: e.message});
}
}); });
} }
); );

View File

@ -1,22 +1 @@
{ {}
"0xf957b38706f783689e96d5b17c2c24a2a1b2093b8e7c5f43e748878a73456984": {
"contracts": {
"0xa72e15b4fc637528e678fdcab66f178b753f609a6d2a66564f1676fde4d51ed3": {
"name": "SimpleStorage",
"address": "0x595dE951E7458Cf6e49606C91294A764098248AD"
},
"0x05a320b7f23891fc8c8c40312e0ae5a104d4734b3c0692eb24f9e99fca0ada3f": {
"name": "ENSRegistry",
"address": "0x07dF106F12b516a36e29e23b2acF1e47b26Cb81B"
},
"0xa971bc944c276c3a2f98bdad24d102d00615680517267a9e48cf7091ce512c88": {
"name": "Resolver",
"address": "0x8e43bD6a0357385D5a35980baCa7A2644459038b"
},
"0x618f78549c37a154c4931bb2f10e4b92fbf8004fc1dfa6c11b3276257d3c84f0": {
"name": "FIFSRegistrar",
"address": "0xf2086ADff73E5B6412c1ca089250b77f9B28984B"
}
}
}
}