Show result
This commit is contained in:
parent
9d1a6dad3d
commit
1a35735698
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,8 @@ class ContractFunctionsContainer extends Component {
|
||||||
render={({contractProfile, contractFunctions, postContractFunction}) => (
|
render={({contractProfile, contractFunctions, postContractFunction}) => (
|
||||||
<ContractFunctions contractProfile={contractProfile}
|
<ContractFunctions contractProfile={contractProfile}
|
||||||
contractFunctions={contractFunctions}
|
contractFunctions={contractFunctions}
|
||||||
postContractFunction={postContractFunction} />
|
constructor={true}
|
||||||
|
postContractFunction={postContractFunction}/>
|
||||||
)} />
|
)} />
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -101,21 +101,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});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
|
@ -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"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in New Issue