-
Notifications
You must be signed in to change notification settings - Fork 7
/
ui.js
233 lines (217 loc) · 8.24 KB
/
ui.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
let _currentNetwork = 'ethereum';
// inits the UI, should be called in window.onload
function uiInit() {
_initUiComponents();
$('#main-menu > a').click(_mainMenuOnClick);
$('#load-contract').click(_loadContractOnClick);
const params = $.deparam();
if (params.network) {
_currentNetwork = params.network;
}
if (params.contract) {
$('#contract-address').val(params.contract);
_loadContract(params.contract);
}
$(`#main-menu > a[href='${window.location.hash}']`).trigger('click');
}
// helper to init special SemanticUI components that have JS behavior
function _initUiComponents() {
$('.ui.dropdown').dropdown();
$('.ui.accordion').accordion();
// custom dropdown component for changing decimals of numbers
$('.ui.dropdown[data-tpl=decimals]').dropdown({
onChange: (decimals, _, $choice) => {
const inputEl = $choice.closest('.ui.input').find('[data-tpl=input]');
const prevDecimals = inputEl.data('prevDecimals') || 0;
inputEl.data('prevDecimals', decimals);
if (inputEl.val() === '') return;
const rawNum = new BigNumber(`${inputEl.val().replaceAll(',', '')}e${prevDecimals}`);
if (decimals != 0) {
inputEl.val(rawNum.dividedBy(new BigNumber(`1e${decimals}`)).toFormat());
} else {
inputEl.val(rawNum.dividedBy(new BigNumber(`1e${decimals}`)).toFixed());
}
inputEl.data('rawVal', rawNum.toFixed());
}
});
}
// loads a new contract into the page, downloads its ABI and generates all UI
async function _loadContract(contractAddress) {
try {
const abi = await dataFetchAbi(contractAddress, _currentNetwork);
dataInitializeContractInstance(contractAddress, _currentNetwork, abi);
_mainError();
_generateUiForAbi(abi);
} catch (e) {
_mainError(e.message);
_generateUiForAbi();
}
}
// event handler when user clicks the main "Load" button to load a new contract
function _loadContractOnClick() {
const contract = $('#contract-address').val();
const network = $('#contract-network').find('.selected').data('value');
window.location.search = $.param({
contract,
network
});
}
// event handler when user clicks the main sections menu (read, write, events, etc)
function _mainMenuOnClick(event) {
$(event.delegateTarget).parent().find('.active').removeClass('active');
$(event.delegateTarget).addClass('active');
$('.main-menu-content').hide();
const contentSelector = `#${$(event.delegateTarget).data('target')}`;
$(contentSelector).show();
}
// if error is non-empty, shows the main error dialog
// if error is empty, hides the main error dialog
function _mainError(errorMessage) {
if (errorMessage) {
$('#contract-loaded').hide();
$('#main-error > p').text(errorMessage);
$('#main-error').fadeIn(100);
} else {
$('#main-error').hide();
$('#contract-loaded').fadeIn(100);
}
}
// populate all the UI for a newly loaded contract (all functions, events, etc)
function _generateUiForAbi(abi) {
if (abi) {
for (let i=0; i<abi.length; i++) {
const field = abi[i];
if (field.type === 'function') {
if (field.stateMutability === 'view') {
_addUiForFunctionField(field, '#abi-read-functions');
} else {
_addUiForFunctionField(field, '#abi-write-functions');
}
}
if (field.type === 'event') {
// todo
}
_initUiComponents();
}
} else {
$('#abi-fields > div').empty();
}
}
// populate the UI for a single contract function
function _addUiForFunctionField(field, toSelector) {
const functionEl = $('#templates > #abi-function').clone();
$(toSelector).append(functionEl);
$(functionEl).find('[data-tpl=name]').text(field.name);
$(functionEl).find('[data-tpl=signature]').text(dataEncodeFunctionSignature(field, _currentNetwork));
$(functionEl).find('[data-tpl=query]').click(field, _queryFunctionOnClick);
for (let i=0; i<field.inputs.length; i++) {
const functionArgEl = _getElementForFunctionArg(field.inputs[i]);
_appendInputFunctionArg(field.inputs[i], functionEl, functionArgEl);
}
}
// event handler when user clicks the "Query" button of a function to execute a function call
async function _queryFunctionOnClick(event) {
const field = event.data;
const functionEl = $(event.delegateTarget).parent();
$(functionEl).find('[data-tpl=outputs]').empty();
const inputArray = _validateQueryInputArray(field, functionEl);
if (!Array.isArray(inputArray)) return;
try {
const result = await dataQueryFunction(field, inputArray);
for (let i=0; i<field.outputs.length; i++) {
const outputValue = result; // todo fix for multiple
const functionArgEl = _getElementForFunctionArg(field.outputs[i]);
_appendOutputFunctionArg(field.outputs[i], outputValue, functionEl, functionArgEl);
}
} catch (e) {
_appendOutputFunctionError(e.message, functionEl);
}
_initUiComponents();
}
// prepare the HTML element for a single function argument
function _getElementForFunctionArg(fieldArg) {
let templateSelector = '#function-arg';
if (fieldArg.type === 'uint256') {
templateSelector = '#function-arg-decimals';
}
const functionArgEl = $(`#templates > ${templateSelector}`).clone();
$(functionArgEl).find('[data-tpl=name]').text(fieldArg.name);
$(functionArgEl).find('[data-tpl=type]').text(fieldArg.type);
if (templateSelector === '#function-arg-decimals') {
$(functionArgEl).find('[data-tpl=menu]').find('.item').text((_, text) => {
return text.replace('type', fieldArg.type);
});
}
return functionArgEl;
}
// populate the UI for a single function argument for input
function _appendInputFunctionArg(fieldArg, functionEl, functionArgEl) {
$(functionEl).find('[data-tpl=inputs]').append(functionArgEl);
let placeholder = '';
if (fieldArg.type === 'address') {
placeholder = '20 byte hex string starting with 0x';
}
if (fieldArg.type === 'uint256') {
placeholder = 'unsigned decimal';
}
$(functionArgEl).find('[data-tpl=input]').attr('placeholder', placeholder);
}
// populate the UI for a single function argument for output
function _appendOutputFunctionArg(fieldArg, outputValue, functionEl, functionArgEl) {
$(functionArgEl).hide();
$(functionEl).find('[data-tpl=outputs]').append(functionArgEl);
$(functionArgEl).find('[data-tpl=input]').val(outputValue);
$(functionArgEl).find('[data-tpl=input]').attr('readonly', '1');
if (fieldArg.type === 'address') {
$(functionArgEl).find('[data-tpl=input]').parent().wrap(`<a href="${_getLinkForAddress(outputValue)}"></a>`);
}
$(functionArgEl).slideDown(100);
}
// validate all input arguments and show error UI before executing a function call
// returns array if the inputs are valid, false if not
function _validateQueryInputArray(field, functionEl) {
const inputArray = [];
let error = false;
$(functionEl).find('[data-tpl=inputs]').find('[data-tpl=input]').each((i, inputEl) => {
const fieldArg = field.inputs[i];
const inputValue = $(inputEl).data('rawVal') || $(inputEl).val();
try {
dataValidateType(fieldArg.type, inputValue);
} catch (e) {
const errorMessage = `Input '${fieldArg.name}' is invalid: ${e.message}`;
_appendOutputFunctionError(errorMessage);
errorMessage = true;
}
inputArray.push(inputValue);
});
if (error) return false;
return inputArray;
}
// populate the UI showing an error in the function output
function _appendOutputFunctionError(errorMessage, functionEl) {
const errorEl = $('#templates > #function-arg-error').clone();
$(errorEl).hide();
$(errorEl).find('[data-tpl=text]').text(errorMessage);
$(functionEl).find('[data-tpl=outputs]').append(errorEl);
$(errorEl).slideDown(100);
}
// get the URL for etherscan for exploring an Ethereum/BSC address further
function _getLinkForAddress(address) {
if (_currentNetwork === 'bsc') return `https://bscscan.com/address/${address}`;
return `https://etherscan.io/address/${address}`;
}
// add missing jquery implementation for $.deparam functionality (the opposite of $.param)
(function($){
$.deparam = $.deparam || function(uri){
if(uri === undefined){
uri = window.location.search;
}
var queryString = {};
uri.replace(
new RegExp(
"([^?=&]+)(=([^&#]*))?", "g"),
function($0, $1, $2, $3) { queryString[$1] = $3; }
);
return queryString;
};
})(jQuery);