Merge branch 'master' into optimizecircle

pull/5370/head
bunsenstraat 2 years ago committed by GitHub
commit 3037ea2865
  1. 20
      .github/workflows/run-sut.yml
  2. 9
      apps/remix-ide-e2e/src/tests/editorAutoComplete.test.ts
  3. 8
      apps/remix-ide-e2e/src/tests/editorHoverContext.test.ts
  4. 7
      apps/remix-ide-e2e/src/tests/editorReferences.test.ts
  5. 7
      apps/remix-ide-e2e/src/tests/editor_error_marker.test.ts
  6. 7
      apps/remix-ide-e2e/src/tests/editor_line_text.test.ts
  7. 15
      apps/remix-ide-e2e/src/tests/erc721.test.ts
  8. 8
      apps/remix-ide-e2e/src/tests/workspace.test.ts
  9. 147
      apps/remix-ide/contracts/ballot.sol
  10. 28
      apps/remix-ide/contracts/tests/Ballot_test.sol
  11. 4
      apps/remix-ide/src/app.js
  12. 28
      apps/remix-ide/src/app/plugins/parser/code-parser.tsx
  13. 2
      apps/remix-ide/src/assets/js/loader.js
  14. 24
      libs/remix-lib/src/execution/txHelper.ts
  15. 15
      libs/remix-lib/test/txHelper.ts
  16. 2
      libs/remix-ui/app/src/lib/remix-app/style/remix-app.css
  17. 21
      libs/remix-ui/editor/src/lib/providers/completion/completionGlobals.ts
  18. 19
      libs/remix-ui/settings/src/lib/remix-ui-settings.tsx
  19. 5
      libs/remix-ui/vertical-icons-panel/src/lib/components/IconList.tsx
  20. 5
      libs/remix-ui/workspace/src/lib/actions/index.ts
  21. 162
      libs/remix-ui/workspace/src/lib/remix-ui-workspace.tsx
  22. 3
      libs/remix-ws-templates/package.json
  23. 14
      libs/remix-ws-templates/src/templates/ozerc20/contracts/SampleERC20.sol
  24. 7
      libs/remix-ws-templates/src/templates/ozerc20/index.ts
  25. 2
      libs/remix-ws-templates/src/templates/ozerc20/scripts/deploy_with_ethers.ts
  26. 2
      libs/remix-ws-templates/src/templates/ozerc20/scripts/deploy_with_web3.ts
  27. 18
      libs/remix-ws-templates/src/templates/ozerc20/tests/MyToken_test.sol
  28. 18
      libs/remix-ws-templates/src/templates/ozerc20/tests/SampleERC20_test.sol
  29. 14
      libs/remix-ws-templates/src/templates/ozerc721/contracts/SampleERC721.sol
  30. 7
      libs/remix-ws-templates/src/templates/ozerc721/index.ts
  31. 2
      libs/remix-ws-templates/src/templates/ozerc721/scripts/deploy_with_ethers.ts
  32. 2
      libs/remix-ws-templates/src/templates/ozerc721/scripts/deploy_with_web3.ts
  33. 18
      libs/remix-ws-templates/src/templates/ozerc721/tests/MyToken_test.sol
  34. 18
      libs/remix-ws-templates/src/templates/ozerc721/tests/SampleERC721_test.sol
  35. 2
      package.json
  36. 12
      yarn.lock

@ -0,0 +1,20 @@
name: Running Solidity Unit Tests
on: [push]
jobs:
run_sol_contracts_job:
runs-on: ubuntu-latest
name: A job to run solidity unit tests on github actions CI
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Environment Setup
uses: actions/setup-node@v3
with:
node-version: 14.17.6
- name: Run SUT Action
uses: EthereumRemix/sol-test@v1
with:
test-path: 'apps/remix-ide/contracts/tests'
compiler-version: '0.8.15'

@ -12,6 +12,15 @@ module.exports = {
before: function (browser: NightwatchBrowser, done: VoidFunction) { before: function (browser: NightwatchBrowser, done: VoidFunction) {
init(browser, done, 'http://127.0.0.1:8080', false) init(browser, done, 'http://127.0.0.1:8080', false)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should add test and base files #group1': function (browser: NightwatchBrowser) { 'Should add test and base files #group1': function (browser: NightwatchBrowser) {
browser.addFile(examples.testContract.name, examples.testContract) browser.addFile(examples.testContract.name, examples.testContract)
.addFile(examples.baseContract.name, examples.baseContract) .addFile(examples.baseContract.name, examples.baseContract)

@ -18,6 +18,14 @@ module.exports = {
init(browser, done, 'http://127.0.0.1:8080', false) init(browser, done, 'http://127.0.0.1:8080', false)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should load the test file': function (browser: NightwatchBrowser) { 'Should load the test file': function (browser: NightwatchBrowser) {
browser.openFile('contracts') browser.openFile('contracts')
.openFile('contracts/3_Ballot.sol') .openFile('contracts/3_Ballot.sol')

@ -20,6 +20,13 @@ module.exports = {
init(browser, done, 'http://127.0.0.1:8080', false) init(browser, done, 'http://127.0.0.1:8080', false)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should load the test file': function (browser: NightwatchBrowser) { 'Should load the test file': function (browser: NightwatchBrowser) {
browser.openFile('contracts') browser.openFile('contracts')
.openFile('contracts/3_Ballot.sol') .openFile('contracts/3_Ballot.sol')

@ -8,6 +8,13 @@ module.exports = {
before: function (browser: NightwatchBrowser, done: VoidFunction) { before: function (browser: NightwatchBrowser, done: VoidFunction) {
init(browser, done, 'http://127.0.0.1:8080', true) init(browser, done, 'http://127.0.0.1:8080', true)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should add error marker': function (browser: NightwatchBrowser) { 'Should add error marker': function (browser: NightwatchBrowser) {
browser browser
.openFile('contracts') .openFile('contracts')

@ -8,6 +8,13 @@ module.exports = {
before: function (browser: NightwatchBrowser, done: VoidFunction) { before: function (browser: NightwatchBrowser, done: VoidFunction) {
init(browser, done, 'http://127.0.0.1:8080', true) init(browser, done, 'http://127.0.0.1:8080', true)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should add line texts': function (browser: NightwatchBrowser) { 'Should add line texts': function (browser: NightwatchBrowser) {
browser browser
.openFile('contracts') .openFile('contracts')

@ -26,20 +26,17 @@ module.exports = {
.execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() }) .execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() })
.pause(100) .pause(100)
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/SampleERC721.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/MyToken.sol"]')
.openFile('contracts/SampleERC721.sol') .openFile('contracts/MyToken.sol')
.verifyContracts(['SampleERC721']) .verifyContracts(['MyToken'])
// deploy contract // deploy contract
.clickLaunchIcon('udapp') .clickLaunchIcon('udapp')
.selectContract('SampleERC721') .selectContract('MyToken')
.createContract('E,E') .createContract('')
.testFunction('last', .testFunction('last',
{ {
status: 'true Transaction mined and execution succeed', status: 'true Transaction mined and execution succeed',
'decoded input': { 'decoded input': {}
'string tokenName': 'E',
'string tokenSymbol': 'E'
}
}).end() }).end()
} }
} }

@ -124,7 +124,7 @@ module.exports = {
.execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() }) .execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() })
.pause(100) .pause(100)
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/SampleERC20.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/MyToken.sol"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]')
// check js and ts files are not transformed // check js and ts files are not transformed
@ -156,7 +156,7 @@ module.exports = {
'Incorrect content') 'Incorrect content')
}) })
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/SampleERC20_test.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/MyToken_test.sol"]')
}, },
'Should create ERC721 workspace with files #group1': function (browser: NightwatchBrowser) { 'Should create ERC721 workspace with files #group1': function (browser: NightwatchBrowser) {
@ -172,7 +172,7 @@ module.exports = {
.execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() }) .execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() })
.pause(100) .pause(100)
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/SampleERC721.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/MyToken.sol"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]')
// check js and ts files are not transformed // check js and ts files are not transformed
@ -204,7 +204,7 @@ module.exports = {
'Incorrect content') 'Incorrect content')
}) })
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/SampleERC721_test.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/MyToken_test.sol"]')
}, },
// WORKSPACE TEMPLATES E2E END // WORKSPACE TEMPLATES E2E END

@ -1,10 +1,13 @@
pragma solidity ^0.4.0; // SPDX-License-Identifier: GPL-3.0
/// @title Voting with delegation. pragma solidity >=0.7.0 <0.9.0;
/**
* @title Ballot
* @dev Implements voting process along with vote delegation
*/
contract Ballot { contract Ballot {
// This declares a new complex type which will
// be used for variables later.
// It will represent a single voter.
struct Voter { struct Voter {
uint weight; // weight is accumulated by delegation uint weight; // weight is accumulated by delegation
bool voted; // if true, that person already voted bool voted; // if true, that person already voted
@ -12,33 +15,31 @@ contract Ballot {
uint vote; // index of the voted proposal uint vote; // index of the voted proposal
} }
// This is a type for a single proposal.
struct Proposal { struct Proposal {
// If you can limit the length to a certain number of bytes,
// always use one of bytes1 to bytes32 because they are much cheaper
bytes32 name; // short name (up to 32 bytes) bytes32 name; // short name (up to 32 bytes)
uint voteCount; // number of accumulated votes uint voteCount; // number of accumulated votes
} }
address public chairperson; address public chairperson;
// This declares a state variable that
// stores a \`Voter\` struct for each possible address.
mapping(address => Voter) public voters; mapping(address => Voter) public voters;
// A dynamically-sized array of \`Proposal\` structs.
Proposal[] public proposals; Proposal[] public proposals;
/// Create a new ballot to choose one of \`proposalNames\`. /**
function Ballot(bytes32[] proposalNames) { * @dev Create a new ballot to choose one of 'proposalNames'.
* @param proposalNames names of proposals
*/
constructor(bytes32[] memory proposalNames) {
chairperson = msg.sender; chairperson = msg.sender;
voters[chairperson].weight = 1; voters[chairperson].weight = 1;
// For each of the provided proposal names,
// create a new proposal object and add it
// to the end of the array.
for (uint i = 0; i < proposalNames.length; i++) { for (uint i = 0; i < proposalNames.length; i++) {
// \`Proposal({...})\` creates a temporary // 'Proposal({...})' creates a temporary
// Proposal object and \`proposals.push(...)\` // Proposal object and 'proposals.push(...)'
// appends it to the end of \`proposals\`. // appends it to the end of 'proposals'.
proposals.push(Proposal({ proposals.push(Proposal({
name: proposalNames[i], name: proposalNames[i],
voteCount: 0 voteCount: 0
@ -46,98 +47,92 @@ contract Ballot {
} }
} }
// Give \`voter\` the right to vote on this ballot. /**
// May only be called by \`chairperson\`. * @dev Give 'voter' the right to vote on this ballot. May only be called by 'chairperson'.
function giveRightToVote(address voter) { * @param voter address of voter
if (msg.sender != chairperson || voters[voter].voted) { */
// \`throw\` terminates and reverts all changes to function giveRightToVote(address voter) public {
// the state and to Ether balances. It is often require(
// a good idea to use this if functions are msg.sender == chairperson,
// called incorrectly. But watch out, this "Only chairperson can give right to vote."
// will also consume all provided gas. );
throw; require(
} !voters[voter].voted,
"The voter already voted."
);
require(voters[voter].weight == 0);
voters[voter].weight = 1; voters[voter].weight = 1;
} }
/// Delegate your vote to the voter \`to\`. /**
function delegate(address to) { * @dev Delegate your vote to the voter 'to'.
// assigns reference * @param to address to which vote is delegated
Voter sender = voters[msg.sender]; */
if (sender.voted) function delegate(address to) public {
throw; Voter storage sender = voters[msg.sender];
require(!sender.voted, "You already voted.");
// Forward the delegation as long as require(to != msg.sender, "Self-delegation is disallowed.");
// \`to\` also delegated.
// In general, such loops are very dangerous, while (voters[to].delegate != address(0)) {
// because if they run too long, they might
// need more gas than is available in a block.
// In this case, the delegation will not be executed,
// but in other situations, such loops might
// cause a contract to get "stuck" completely.
while (
voters[to].delegate != address(0) &&
voters[to].delegate != msg.sender
) {
to = voters[to].delegate; to = voters[to].delegate;
}
// We found a loop in the delegation, not allowed. // We found a loop in the delegation, not allowed.
if (to == msg.sender) { require(to != msg.sender, "Found loop in delegation.");
throw;
} }
// Since \`sender\` is a reference, this
// modifies \`voters[msg.sender].voted\`
sender.voted = true; sender.voted = true;
sender.delegate = to; sender.delegate = to;
Voter delegate = voters[to]; Voter storage delegate_ = voters[to];
if (delegate.voted) { if (delegate_.voted) {
// If the delegate already voted, // If the delegate already voted,
// directly add to the number of votes // directly add to the number of votes
proposals[delegate.vote].voteCount += sender.weight; proposals[delegate_.vote].voteCount += sender.weight;
} else { } else {
// If the delegate did not vote yet, // If the delegate did not vote yet,
// add to her weight. // add to her weight.
delegate.weight += sender.weight; delegate_.weight += sender.weight;
} }
} }
/// Give your vote (including votes delegated to you) /**
/// to proposal \`proposals[proposal].name\`. * @dev Give your vote (including votes delegated to you) to proposal 'proposals[proposal].name'.
function vote(uint proposal) { * @param proposal index of proposal in the proposals array
Voter sender = voters[msg.sender]; */
if (sender.voted) function vote(uint proposal) public {
throw; Voter storage sender = voters[msg.sender];
require(sender.weight != 0, "Has no right to vote");
require(!sender.voted, "Already voted.");
sender.voted = true; sender.voted = true;
sender.vote = proposal; sender.vote = proposal;
// If \`proposal\` is out of the range of the array, // If 'proposal' is out of the range of the array,
// this will throw automatically and revert all // this will throw automatically and revert all
// changes. // changes.
proposals[proposal].voteCount += sender.weight; proposals[proposal].voteCount += sender.weight;
} }
/// @dev Computes the winning proposal taking all /**
/// previous votes into account. * @dev Computes the winning proposal taking all previous votes into account.
function winningProposal() constant * @return winningProposal_ index of winning proposal in the proposals array
returns (uint winningProposal) */
function winningProposal() public view
returns (uint winningProposal_)
{ {
uint winningVoteCount = 0; uint winningVoteCount = 0;
for (uint p = 0; p < proposals.length; p++) { for (uint p = 0; p < proposals.length; p++) {
if (proposals[p].voteCount > winningVoteCount) { if (proposals[p].voteCount > winningVoteCount) {
winningVoteCount = proposals[p].voteCount; winningVoteCount = proposals[p].voteCount;
winningProposal = p; winningProposal_ = p;
} }
} }
} }
// Calls winningProposal() function to get the index /**
// of the winner contained in the proposals array and then * @dev Calls winningProposal() function to get the index of the winner contained in the proposals array and then
// returns the name of the winner * @return winnerName_ the name of the winner
function winnerName() constant */
returns (bytes32 winnerName) function winnerName() public view
returns (bytes32 winnerName_)
{ {
winnerName = proposals[winningProposal()].name; winnerName_ = proposals[winningProposal()].name;
} }
} }

@ -0,0 +1,28 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol"; // this import is automatically injected by Remix.
import "hardhat/console.sol";
import "../ballot.sol";
contract BallotTest {
bytes32[] proposalNames;
Ballot ballotToTest;
function beforeAll () public {
proposalNames.push(bytes32("candidate1"));
ballotToTest = new Ballot(proposalNames);
}
function checkWinningProposal () public {
console.log("Running checkWinningProposal");
ballotToTest.vote(0);
Assert.equal(ballotToTest.winningProposal(), uint(0), "proposal at index 0 should be the winning proposal");
Assert.equal(ballotToTest.winnerName(), bytes32("candidate1"), "candidate1 should be the winner name");
}
function checkWinninProposalWithReturnValue () public view returns (bool) {
return ballotToTest.winningProposal() == 0;
}
}

@ -354,10 +354,6 @@ class AppComponent {
const queryParams = new QueryParams() const queryParams = new QueryParams()
const params = queryParams.get() const params = queryParams.get()
if (isElectron()) {
this.appManager.activatePlugin('remixd')
}
try { try {
this.engine.register(await this.appManager.registeredPlugins()) this.engine.register(await this.appManager.registeredPlugins())
} catch (e) { } catch (e) {

@ -87,6 +87,18 @@ export class CodeParser extends Plugin {
} }
} }
async handleChangeEvents() {
const completionSettings = await this.call('config', 'getAppParameter', 'auto-completion')
if (completionSettings) {
await this.antlrService.getCurrentFileAST()
}
const showGasSettings = await this.call('config', 'getAppParameter', 'show-gas')
const showErrorSettings = await this.call('config', 'getAppParameter', 'display-errors')
if(showGasSettings || showErrorSettings || completionSettings) {
await this.compilerService.compile()
}
}
async onActivation() { async onActivation() {
this.gasService = new CodeParserGasService(this) this.gasService = new CodeParserGasService(this)
@ -102,8 +114,7 @@ export class CodeParser extends Plugin {
this.on('editor', 'didChangeFile', async (file) => { this.on('editor', 'didChangeFile', async (file) => {
await this.call('editor', 'discardLineTexts') await this.call('editor', 'discardLineTexts')
await this.antlrService.getCurrentFileAST() await this.handleChangeEvents()
await this.compilerService.compile()
}) })
this.on('filePanel', 'setWorkspace', async () => { this.on('filePanel', 'setWorkspace', async () => {
@ -113,8 +124,7 @@ export class CodeParser extends Plugin {
this.on('fileManager', 'currentFileChanged', async () => { this.on('fileManager', 'currentFileChanged', async () => {
await this.call('editor', 'discardLineTexts') await this.call('editor', 'discardLineTexts')
await this.antlrService.getCurrentFileAST() await this.handleChangeEvents()
await this.compilerService.compile()
}) })
this.on('solidity', 'loadingCompiler', async (url) => { this.on('solidity', 'loadingCompiler', async (url) => {
@ -188,10 +198,10 @@ export class CodeParser extends Plugin {
const index = {} const index = {}
const contractName: string = contractNode.name const contractName: string = contractNode.name
const callback = (node) => { const callback = (node) => {
if(inScope && node.scope !== contractNode.id if (inScope && node.scope !== contractNode.id
&& !(node.nodeType === 'EnumDefinition' || node.nodeType === 'EventDefinition' || node.nodeType === 'ModifierDefinition')) && !(node.nodeType === 'EnumDefinition' || node.nodeType === 'EventDefinition' || node.nodeType === 'ModifierDefinition'))
return return
if(inScope) node.isClassNode = true; if (inScope) node.isClassNode = true;
node.gasEstimate = this._getContractGasEstimate(node, contractName, fileName, compilatioResult) node.gasEstimate = this._getContractGasEstimate(node, contractName, fileName, compilatioResult)
node.functionName = node.name + this._getInputParams(node) node.functionName = node.name + this._getInputParams(node)
node.contractName = contractName node.contractName = contractName
@ -227,11 +237,11 @@ export class CodeParser extends Plugin {
if ((node.scope && node.scope === baseContract.id) if ((node.scope && node.scope === baseContract.id)
|| node.nodeType === 'EnumDefinition' || node.nodeType === 'EnumDefinition'
|| node.nodeType === 'EventDefinition' || node.nodeType === 'EventDefinition'
) { ) {
baseNodesWithBaseContractScope[node.id] = node baseNodesWithBaseContractScope[node.id] = node
} }
if(node.members){ if (node.members) {
for(const member of node.members){ for (const member of node.members) {
member.contractName = (baseContract as any).name member.contractName = (baseContract as any).name
member.contractId = (baseContract as any).id member.contractId = (baseContract as any).id
member.isBaseNode = true; member.isBaseNode = true;

@ -45,7 +45,7 @@ function isElectron() {
return false return false
} }
const versionUrl = isElectron() ? 'https://remix.ethereum.org/assets/version.json' : 'assets/version.json' const versionUrl = 'assets/version.json'
fetch(versionUrl, { cache: "no-store" }).then(response => { fetch(versionUrl, { cache: "no-store" }).then(response => {
response.text().then(function (data) { response.text().then(function (data) {
const version = JSON.parse(data); const version = JSON.parse(data);

@ -102,17 +102,25 @@ export function extractSize (type) {
return size ? size[2] : '' return size ? size[2] : ''
} }
export function getFunctionLiner (fn, detailTuple: boolean = true) {
/*
if detailsTuple is True, this will return something like fnName((uint, string))
if detailsTuple is False, this will return something like fnName(tuple)
*/
return fn.name + '(' + fn.inputs.map((value) => {
if (detailTuple && value.components) {
const fullType = makeFullTypeDefinition(value)
return fullType.replace(/tuple/g, '') // return of makeFullTypeDefinition might contain `tuple`, need to remove it cause `methodIdentifier` (fnName) does not include `tuple` keyword
} else {
return value.type
}
}).join(',') + ')'
}
export function getFunction (abi, fnName) { export function getFunction (abi, fnName) {
for (let i = 0; i < abi.length; i++) { for (let i = 0; i < abi.length; i++) {
const fn = abi[i] const fn = abi[i]
if (fn.type === 'function' && fnName === fn.name + '(' + fn.inputs.map((value) => { if (fn.type === 'function' && (fnName === getFunctionLiner(fn, true) || fnName === getFunctionLiner(fn, false))) {
if (value.components) {
const fullType = makeFullTypeDefinition(value)
return fullType.replace(/tuple/g, '') // return of makeFullTypeDefinition might contain `tuple`, need to remove it cause `methodIdentifier` (fnName) does not include `tuple` keyword
} else {
return value.type
}
}).join(',') + ')') {
return fn return fn
} }
} }

@ -3,7 +3,7 @@ import tape from 'tape'
import * as txHelper from '../src/execution/txHelper' import * as txHelper from '../src/execution/txHelper'
tape('getFunction', function (st) { tape('getFunction', function (st) {
st.plan(6) st.plan(11)
let fn = txHelper.getFunction(JSON.parse(abi), 'o((address,uint256))') let fn = txHelper.getFunction(JSON.parse(abi), 'o((address,uint256))')
st.equal(fn.name, 'o') st.equal(fn.name, 'o')
@ -21,6 +21,17 @@ tape('getFunction', function (st) {
fn = txHelper.getReceiveInterface(JSON.parse(abi)) fn = txHelper.getReceiveInterface(JSON.parse(abi))
st.equal(fn.type, 'receive') st.equal(fn.type, 'receive')
fn = txHelper.getFunction(testTupleAbi, 'setUser(tuple)') // some compiler version might resolve to tuple.
st.equal(fn.name, 'setUser')
st.equal(fn.inputs[0].type, 'tuple')
st.equal(fn.inputs[0].name, 'user')
fn = txHelper.getFunctionLiner(testTupleAbi[0], true)
st.equal(fn, 'setUser((string,uint256))')
fn = txHelper.getFunctionLiner(testTupleAbi[0], false)
st.equal(fn, 'setUser(tuple)')
}) })
const abi = `[ const abi = `[
@ -153,3 +164,5 @@ const abi = `[
"type": "receive" "type": "receive"
} }
]` ]`
const testTupleAbi = [{"inputs":[{"components":[{"internalType":"string","name":"name","type":"string"},{"internalType":"uint256","name":"age","type":"uint256"}],"internalType":"struct Example.User","name":"user","type":"tuple"}],"name":"setUser","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"userByAddress","outputs":[{"internalType":"string","name":"name","type":"string"},{"internalType":"uint256","name":"age","type":"uint256"}],"stateMutability":"view","type":"function"}]

@ -8,7 +8,7 @@ pre {
overflow-x: auto; overflow-x: auto;
} }
.remixIDE { .remixIDE {
width : 100vw; width : 100%;
height : 100vh; height : 100vh;
overflow : hidden; overflow : hidden;
flex-direction : row; flex-direction : row;

@ -148,6 +148,27 @@ export function getCompletionSnippets(range: IRange, monaco): monaco.languages.C
insertTextRules: monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet, insertTextRules: monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet,
range range
}, },
{
label: 'while loop',
kind: monaco.languages.CompletionItemKind.Snippet,
insertText: 'while (${1:condition}) \n{\n\t${2:code}\n};',
insertTextRules: monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet,
range
},
{
label: 'do while loop',
kind: monaco.languages.CompletionItemKind.Snippet,
insertText: 'do {\n\t${2:code}\n} \nwhile (${1:condition});',
insertTextRules: monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet,
range
},
{
label: 'for loop',
kind: monaco.languages.CompletionItemKind.Snippet,
insertText: 'for (${1:init}; ${2:condition}; ${3:increment}) \n{\n\t${4:code}\n};',
insertTextRules: monaco.languages.CompletionItemInsertTextRule.InsertAsSnippet,
range
},
{ {
label: 'pragma', label: 'pragma',
kind: monaco.languages.CompletionItemKind.Snippet, kind: monaco.languages.CompletionItemKind.Snippet,

@ -41,13 +41,13 @@ export const RemixUiSettings = (props: RemixUiSettingsProps) => {
if (javascriptVM === null || javascriptVM === undefined) ethereumVM(props.config, true, dispatch) if (javascriptVM === null || javascriptVM === undefined) ethereumVM(props.config, true, dispatch)
const useAutoComplete = props.config.get('settings/auto-completion') const useAutoComplete = props.config.get('settings/auto-completion')
if (useAutoComplete === null || useAutoComplete === undefined) useAutoCompletion(props.config, true, dispatch) if (useAutoComplete === null || useAutoComplete === undefined) useAutoCompletion(props.config, false, dispatch)
const displayErrors = props.config.get('settings/display-errors') const displayErrors = props.config.get('settings/display-errors')
if (displayErrors === null || displayErrors === undefined) useDisplayErrors(props.config, true, dispatch) if (displayErrors === null || displayErrors === undefined) useDisplayErrors(props.config, false, dispatch)
const useShowGas = props.config.get('settings/show-gas') const useShowGas = props.config.get('settings/show-gas')
if (useShowGas === null || useShowGas === undefined) useShowGasInEditor(props.config, true, dispatch) if (useShowGas === null || useShowGas === undefined) useShowGasInEditor(props.config, false, dispatch)
} }
useEffect(() => initValue(), [resetState, props.config]) useEffect(() => initValue(), [resetState, props.config])
useEffect(() => initValue(), []) useEffect(() => initValue(), [])
@ -148,9 +148,10 @@ export const RemixUiSettings = (props: RemixUiSettingsProps) => {
const isEditorWrapChecked = props.config.get('settings/text-wrap') || false const isEditorWrapChecked = props.config.get('settings/text-wrap') || false
const isPersonalChecked = props.config.get('settings/personal-mode') || false const isPersonalChecked = props.config.get('settings/personal-mode') || false
const isMatomoChecked = props.config.get('settings/matomo-analytics') || false const isMatomoChecked = props.config.get('settings/matomo-analytics') || false
const isAutoCompleteChecked = props.config.get('settings/auto-completion') === null ? true:props.config.get('settings/auto-completion')
const isShowGasInEditorChecked = props.config.get('settings/show-gas') === null ? true:props.config.get('settings/show-gas') const isAutoCompleteChecked = props.config.get('settings/auto-completion') || false
const displayErrorsChecked = props.config.get('settings/display-errors') === null ? true:props.config.get('settings/display-errors') const isShowGasInEditorChecked = props.config.get('settings/show-gas') || false
const displayErrorsChecked = props.config.get('settings/display-errors') || false
return ( return (
<div className="$border-top"> <div className="$border-top">
<div title="Reset to Default settings." className='d-flex justify-content-end pr-4'> <div title="Reset to Default settings." className='d-flex justify-content-end pr-4'>
@ -188,19 +189,19 @@ export const RemixUiSettings = (props: RemixUiSettingsProps) => {
</div> </div>
<div className='custom-control custom-checkbox mb-1'> <div className='custom-control custom-checkbox mb-1'>
<input onChange={onchangeUseAutoComplete} id="settingsUseAutoComplete" type="checkbox" className="custom-control-input" checked={isAutoCompleteChecked} /> <input onChange={onchangeUseAutoComplete} id="settingsUseAutoComplete" type="checkbox" className="custom-control-input" checked={isAutoCompleteChecked} />
<label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/auto-completion')}`} htmlFor="settingsUseAutoComplete"> <label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/auto-completion')}`} data-id="settingsAutoCompleteLabel" htmlFor="settingsUseAutoComplete">
<span>{useAutoCompleteText}</span> <span>{useAutoCompleteText}</span>
</label> </label>
</div> </div>
<div className='custom-control custom-checkbox mb-1'> <div className='custom-control custom-checkbox mb-1'>
<input onChange={onchangeShowGasInEditor} id="settingsUseShowGas" type="checkbox" className="custom-control-input" checked={isShowGasInEditorChecked} /> <input onChange={onchangeShowGasInEditor} id="settingsUseShowGas" type="checkbox" className="custom-control-input" checked={isShowGasInEditorChecked} />
<label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/show-gas')}`} htmlFor="settingsUseShowGas"> <label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/show-gas')}`} data-id="settingsShowGasLabel" htmlFor="settingsUseShowGas">
<span>{useShowGasInEditorText}</span> <span>{useShowGasInEditorText}</span>
</label> </label>
</div> </div>
<div className='custom-control custom-checkbox mb-1'> <div className='custom-control custom-checkbox mb-1'>
<input onChange={onchangeDisplayErrors} id="settingsDisplayErrors" type="checkbox" className="custom-control-input" checked={displayErrorsChecked} /> <input onChange={onchangeDisplayErrors} id="settingsDisplayErrors" type="checkbox" className="custom-control-input" checked={displayErrorsChecked} />
<label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/display-errors')}`} htmlFor="settingsDisplayErrors"> <label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/display-errors')}`} data-id="displayErrorsLabel" htmlFor="settingsDisplayErrors">
<span>{displayErrorsText}</span> <span>{displayErrorsText}</span>
</label> </label>
</div> </div>

@ -12,7 +12,7 @@ interface OtherIconsProps {
function IconList ({ verticalIconsPlugin, itemContextAction, icons, theme }: OtherIconsProps) { function IconList ({ verticalIconsPlugin, itemContextAction, icons, theme }: OtherIconsProps) {
return ( return (
<div id="otherIcons"> <div id="otherIcons" className="position-relative">
{ {
icons icons
.map(p => ( .map(p => (
@ -25,7 +25,8 @@ function IconList ({ verticalIconsPlugin, itemContextAction, icons, theme }: Oth
p.profile.name p.profile.name
} }
/> />
))} ))
}
</div> </div>
) )
} }

@ -8,6 +8,7 @@ import { createWorkspaceTemplate, getWorkspaces, loadWorkspacePreset, setPlugin,
import { QueryParams } from '@remix-project/remix-lib' import { QueryParams } from '@remix-project/remix-lib'
import { fetchContractFromEtherscan } from '@remix-project/core-plugin' // eslint-disable-line import { fetchContractFromEtherscan } from '@remix-project/core-plugin' // eslint-disable-line
import JSZip from 'jszip' import JSZip from 'jszip'
import isElectron from 'is-electron'
export * from './events' export * from './events'
export * from './workspace' export * from './workspace'
@ -111,6 +112,10 @@ export const initWorkspace = (filePanelPlugin) => async (reducerDispatch: React.
await basicWorkspaceInit(workspaces, workspaceProvider) await basicWorkspaceInit(workspaces, workspaceProvider)
} }
} else await basicWorkspaceInit(workspaces, workspaceProvider) } else await basicWorkspaceInit(workspaces, workspaceProvider)
} else if (isElectron()) {
plugin.call('notification', 'toast', `connecting to localhost...`)
await basicWorkspaceInit(workspaces, workspaceProvider)
await plugin.call('manager', 'activatePlugin', 'remixd')
} else if (localStorage.getItem("currentWorkspace")) { } else if (localStorage.getItem("currentWorkspace")) {
const index = workspaces.findIndex(element => element.name == localStorage.getItem("currentWorkspace")) const index = workspaces.findIndex(element => element.name == localStorage.getItem("currentWorkspace"))
if (index !== -1) { if (index !== -1) {

@ -271,6 +271,16 @@ export function Workspace () {
</Dropdown.Toggle> </Dropdown.Toggle>
<Dropdown.Menu as={CustomMenu} className='w-100 custom-dropdown-items' data-id="custom-dropdown-items"> <Dropdown.Menu as={CustomMenu} className='w-100 custom-dropdown-items' data-id="custom-dropdown-items">
<Dropdown.Item
onClick={() => {
createWorkspace()
}}
>
{
<span className="pl-3"> - create a new workspace - </span>
}
</Dropdown.Item>
<Dropdown.Item onClick={() => { switchWorkspace(LOCALHOST) }}>{currentWorkspace === LOCALHOST ? <span>&#10003; localhost </span> : <span className="pl-3"> { LOCALHOST } </span>}</Dropdown.Item>
{ {
global.fs.browser.workspaces.map(({ name, isGitRepo }, index) => ( global.fs.browser.workspaces.map(({ name, isGitRepo }, index) => (
<Dropdown.Item <Dropdown.Item
@ -290,7 +300,6 @@ export function Workspace () {
</Dropdown.Item> </Dropdown.Item>
)) ))
} }
<Dropdown.Item onClick={() => { switchWorkspace(LOCALHOST) }}>{currentWorkspace === LOCALHOST ? <span>&#10003; localhost </span> : <span className="pl-3"> { LOCALHOST } </span>}</Dropdown.Item>
{ ((global.fs.browser.workspaces.length <= 0) || currentWorkspace === NO_WORKSPACE) && <Dropdown.Item onClick={() => { switchWorkspace(NO_WORKSPACE) }}>{ <span className="pl-3">NO_WORKSPACE</span> }</Dropdown.Item> } { ((global.fs.browser.workspaces.length <= 0) || currentWorkspace === NO_WORKSPACE) && <Dropdown.Item onClick={() => { switchWorkspace(NO_WORKSPACE) }}>{ <span className="pl-3">NO_WORKSPACE</span> }</Dropdown.Item> }
</Dropdown.Menu> </Dropdown.Menu>
</Dropdown> </Dropdown>
@ -300,83 +309,80 @@ export function Workspace () {
<div className='h-100 remixui_fileExplorerTree' onFocus={() => { toggleDropdown(false) }}> <div className='h-100 remixui_fileExplorerTree' onFocus={() => { toggleDropdown(false) }}>
<div className='h-100'> <div className='h-100'>
{ (global.fs.browser.isRequestingWorkspace || global.fs.browser.isRequestingCloning) && <div className="text-center py-5"><i className="fas fa-spinner fa-pulse fa-2x"></i></div>} { (global.fs.browser.isRequestingWorkspace || global.fs.browser.isRequestingCloning) && <div className="text-center py-5"><i className="fas fa-spinner fa-pulse fa-2x"></i></div>}
{ !(global.fs.browser.isRequestingWorkspace || { !(global.fs.browser.isRequestingWorkspace || global.fs.browser.isRequestingCloning) &&
global.fs.browser.isRequestingCloning) && (global.fs.mode === 'browser') && (currentWorkspace !== NO_WORKSPACE) &&
(global.fs.mode === 'browser') && (currentWorkspace !== NO_WORKSPACE) && <div className='h-100 remixui_treeview' data-id='filePanelFileExplorerTree'>
<div className='h-100 remixui_treeview' data-id='filePanelFileExplorerTree'> <FileExplorer
<FileExplorer name={currentWorkspace}
name={currentWorkspace} menuItems={['createNewFile', 'createNewFolder', 'publishToGist', canUpload ? 'uploadFile' : '']}
menuItems={['createNewFile', 'createNewFolder', 'publishToGist', canUpload ? 'uploadFile' : '']} contextMenuItems={global.fs.browser.contextMenu.registeredMenuItems}
contextMenuItems={global.fs.browser.contextMenu.registeredMenuItems} removedContextMenuItems={global.fs.browser.contextMenu.removedMenuItems}
removedContextMenuItems={global.fs.browser.contextMenu.removedMenuItems} files={global.fs.browser.files}
files={global.fs.browser.files} fileState={global.fs.browser.fileState}
fileState={global.fs.browser.fileState} expandPath={global.fs.browser.expandPath}
expandPath={global.fs.browser.expandPath} focusEdit={global.fs.focusEdit}
focusEdit={global.fs.focusEdit} focusElement={global.fs.focusElement}
focusElement={global.fs.focusElement} dispatchCreateNewFile={global.dispatchCreateNewFile}
dispatchCreateNewFile={global.dispatchCreateNewFile} modal={global.modal}
modal={global.modal} dispatchCreateNewFolder={global.dispatchCreateNewFolder}
dispatchCreateNewFolder={global.dispatchCreateNewFolder} readonly={global.fs.readonly}
readonly={global.fs.readonly} toast={global.toast}
toast={global.toast} dispatchDeletePath={global.dispatchDeletePath}
dispatchDeletePath={global.dispatchDeletePath} dispatchRenamePath={global.dispatchRenamePath}
dispatchRenamePath={global.dispatchRenamePath} dispatchUploadFile={global.dispatchUploadFile}
dispatchUploadFile={global.dispatchUploadFile} dispatchCopyFile={global.dispatchCopyFile}
dispatchCopyFile={global.dispatchCopyFile} dispatchCopyFolder={global.dispatchCopyFolder}
dispatchCopyFolder={global.dispatchCopyFolder} dispatchPublishToGist={global.dispatchPublishToGist}
dispatchPublishToGist={global.dispatchPublishToGist} dispatchRunScript={global.dispatchRunScript}
dispatchRunScript={global.dispatchRunScript} dispatchEmitContextMenuEvent={global.dispatchEmitContextMenuEvent}
dispatchEmitContextMenuEvent={global.dispatchEmitContextMenuEvent} dispatchHandleClickFile={global.dispatchHandleClickFile}
dispatchHandleClickFile={global.dispatchHandleClickFile} dispatchSetFocusElement={global.dispatchSetFocusElement}
dispatchSetFocusElement={global.dispatchSetFocusElement} dispatchFetchDirectory={global.dispatchFetchDirectory}
dispatchFetchDirectory={global.dispatchFetchDirectory} dispatchRemoveInputField={global.dispatchRemoveInputField}
dispatchRemoveInputField={global.dispatchRemoveInputField} dispatchAddInputField={global.dispatchAddInputField}
dispatchAddInputField={global.dispatchAddInputField} dispatchHandleExpandPath={global.dispatchHandleExpandPath}
dispatchHandleExpandPath={global.dispatchHandleExpandPath} dispatchMoveFile={global.dispatchMoveFile}
dispatchMoveFile={global.dispatchMoveFile} dispatchMoveFolder={global.dispatchMoveFolder}
dispatchMoveFolder={global.dispatchMoveFolder} />
/> </div>
</div> }
} { global.fs.localhost.isRequestingLocalhost && <div className="text-center py-5"><i className="fas fa-spinner fa-pulse fa-2x"></i></div> }
{ { (global.fs.mode === 'localhost' && global.fs.localhost.isSuccessfulLocalhost) &&
global.fs.localhost.isRequestingLocalhost ? <div className="text-center py-5"><i className="fas fa-spinner fa-pulse fa-2x"></i></div> <div className='h-100 filesystemexplorer remixui_treeview'>
: <div className='h-100 filesystemexplorer remixui_treeview'> <FileExplorer
{ global.fs.mode === 'localhost' && global.fs.localhost.isSuccessfulLocalhost && name='localhost'
<FileExplorer menuItems={['createNewFile', 'createNewFolder']}
name='localhost' contextMenuItems={global.fs.localhost.contextMenu.registeredMenuItems}
menuItems={['createNewFile', 'createNewFolder']} removedContextMenuItems={global.fs.localhost.contextMenu.removedMenuItems}
contextMenuItems={global.fs.localhost.contextMenu.registeredMenuItems} files={global.fs.localhost.files}
removedContextMenuItems={global.fs.localhost.contextMenu.removedMenuItems} fileState={[]}
files={global.fs.localhost.files} expandPath={global.fs.localhost.expandPath}
fileState={[]} focusEdit={global.fs.focusEdit}
expandPath={global.fs.localhost.expandPath} focusElement={global.fs.focusElement}
focusEdit={global.fs.focusEdit} dispatchCreateNewFile={global.dispatchCreateNewFile}
focusElement={global.fs.focusElement} modal={global.modal}
dispatchCreateNewFile={global.dispatchCreateNewFile} dispatchCreateNewFolder={global.dispatchCreateNewFolder}
modal={global.modal} readonly={global.fs.readonly}
dispatchCreateNewFolder={global.dispatchCreateNewFolder} toast={global.toast}
readonly={global.fs.readonly} dispatchDeletePath={global.dispatchDeletePath}
toast={global.toast} dispatchRenamePath={global.dispatchRenamePath}
dispatchDeletePath={global.dispatchDeletePath} dispatchUploadFile={global.dispatchUploadFile}
dispatchRenamePath={global.dispatchRenamePath} dispatchCopyFile={global.dispatchCopyFile}
dispatchUploadFile={global.dispatchUploadFile} dispatchCopyFolder={global.dispatchCopyFolder}
dispatchCopyFile={global.dispatchCopyFile} dispatchPublishToGist={global.dispatchPublishToGist}
dispatchCopyFolder={global.dispatchCopyFolder} dispatchRunScript={global.dispatchRunScript}
dispatchPublishToGist={global.dispatchPublishToGist} dispatchEmitContextMenuEvent={global.dispatchEmitContextMenuEvent}
dispatchRunScript={global.dispatchRunScript} dispatchHandleClickFile={global.dispatchHandleClickFile}
dispatchEmitContextMenuEvent={global.dispatchEmitContextMenuEvent} dispatchSetFocusElement={global.dispatchSetFocusElement}
dispatchHandleClickFile={global.dispatchHandleClickFile} dispatchFetchDirectory={global.dispatchFetchDirectory}
dispatchSetFocusElement={global.dispatchSetFocusElement} dispatchRemoveInputField={global.dispatchRemoveInputField}
dispatchFetchDirectory={global.dispatchFetchDirectory} dispatchAddInputField={global.dispatchAddInputField}
dispatchRemoveInputField={global.dispatchRemoveInputField} dispatchHandleExpandPath={global.dispatchHandleExpandPath}
dispatchAddInputField={global.dispatchAddInputField} dispatchMoveFile={global.dispatchMoveFile}
dispatchHandleExpandPath={global.dispatchHandleExpandPath} dispatchMoveFolder={global.dispatchMoveFolder}
dispatchMoveFile={global.dispatchMoveFile} />
dispatchMoveFolder={global.dispatchMoveFolder} </div>
/> }
}
</div>
}
</div> </div>
</div> </div>
</div> </div>

@ -22,8 +22,11 @@
"homepage": "https://github.com/ethereum/remix-project/tree/master/libs/remix-ws-templates#readme", "homepage": "https://github.com/ethereum/remix-project/tree/master/libs/remix-ws-templates#readme",
"typings": "./src/index.d.ts", "typings": "./src/index.d.ts",
"dependencies": { "dependencies": {
"@openzeppelin/contracts": "^4.7.3",
"@openzeppelin/wizard": "^0.1.1",
"ethers": "^5.4.2", "ethers": "^5.4.2",
"web3": "^1.5.1" "web3": "^1.5.1"
}, },
"gitHead": "0c1957c9b2f890076a5062309bc81b41f93af57c" "gitHead": "0c1957c9b2f890076a5062309bc81b41f93af57c"
} }

@ -1,14 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
/**
* @title SampleERC20
* @dev Create a sample ERC20 standard token
*/
contract SampleERC20 is ERC20 {
constructor(string memory tokenName, string memory tokenSymbol) ERC20(tokenName, tokenSymbol) {}
}

@ -1,7 +1,8 @@
import { erc20 } from '@openzeppelin/wizard';
export default async () => { export default async () => {
return { return {
// @ts-ignore 'contracts/MyToken.sol': erc20.print(),
'contracts/SampleERC20.sol': (await import('raw-loader!./contracts/SampleERC20.sol')).default,
// @ts-ignore // @ts-ignore
'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default, 'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default,
// @ts-ignore // @ts-ignore
@ -11,6 +12,6 @@ export default async () => {
// @ts-ignore // @ts-ignore
'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default, 'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default,
// @ts-ignore // @ts-ignore
'tests/SampleERC20_test.sol': (await import('raw-loader!./tests/SampleERC20_test.sol')).default 'tests/MyToken_test.sol': (await import('raw-loader!./tests/MyToken_test.sol')).default
} }
} }

@ -2,7 +2,7 @@ import { deploy } from './ethers-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC20', ['testToken', 'TST']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -2,7 +2,7 @@ import { deploy } from './web3-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC20', ['testToken', 'TST']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -0,0 +1,18 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/MyToken.sol";
contract MyTokenTest {
MyToken s;
function beforeAll () public {
s = new MyToken();
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "MyToken", "token name did not match");
Assert.equal(s.symbol(), "MTK", "token symbol did not match");
}
}

@ -1,18 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/SampleERC20.sol";
contract SampleERC20Test {
SampleERC20 s;
function beforeAll () public {
s = new SampleERC20("TestToken", "TST");
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "TestToken", "token name did not match");
Assert.equal(s.symbol(), "TST", "token symbol did not match");
}
}

@ -1,14 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
/**
* @title SampleERC721
* @dev Create a sample ERC721 standard token
*/
contract SampleERC721 is ERC721 {
constructor(string memory tokenName, string memory tokenSymbol) ERC721(tokenName, tokenSymbol) {}
}

@ -1,7 +1,8 @@
import { erc721 } from '@openzeppelin/wizard';
export default async () => { export default async () => {
return { return {
// @ts-ignore 'contracts/MyToken.sol': erc721.print(),
'contracts/SampleERC721.sol': (await import('raw-loader!./contracts/SampleERC721.sol')).default,
// @ts-ignore // @ts-ignore
'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default, 'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default,
// @ts-ignore // @ts-ignore
@ -11,6 +12,6 @@ export default async () => {
// @ts-ignore // @ts-ignore
'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default, 'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default,
// @ts-ignore // @ts-ignore
'tests/SampleERC721_test.sol': (await import('raw-loader!./tests/SampleERC721_test.sol')).default 'tests/MyToken_test.sol': (await import('raw-loader!./tests/MyToken_test.sol')).default
} }
} }

@ -2,7 +2,7 @@ import { deploy } from './ethers-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC721', ['testNFT', 'TNFT']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -2,7 +2,7 @@ import { deploy } from './web3-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC721', ['testToken', 'TST']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -0,0 +1,18 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/MyToken.sol";
contract MyTokenTest {
MyToken s;
function beforeAll () public {
s = new MyToken();
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "MyToken", "token name did not match");
Assert.equal(s.symbol(), "MTK", "token symbol did not match");
}
}

@ -1,18 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/SampleERC721.sol";
contract SampleERC721Test {
SampleERC721 s;
function beforeAll () public {
s = new SampleERC721("TestNFT", "TNFT");
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "TestNFT", "token name did not match");
Assert.equal(s.symbol(), "TNFT", "token symbol did not match");
}
}

@ -154,6 +154,8 @@
"@ethersphere/bee-js": "^3.2.0", "@ethersphere/bee-js": "^3.2.0",
"@isomorphic-git/lightning-fs": "^4.4.1", "@isomorphic-git/lightning-fs": "^4.4.1",
"@monaco-editor/react": "4.4.5", "@monaco-editor/react": "4.4.5",
"@openzeppelin/contracts": "^4.7.3",
"@openzeppelin/wizard": "^0.1.1",
"@remixproject/engine": "^0.3.31", "@remixproject/engine": "^0.3.31",
"@remixproject/engine-web": "^0.3.31", "@remixproject/engine-web": "^0.3.31",
"@remixproject/plugin": "^0.3.31", "@remixproject/plugin": "^0.3.31",

@ -3912,6 +3912,18 @@
dependencies: dependencies:
"@octokit/openapi-types" "^11.2.0" "@octokit/openapi-types" "^11.2.0"
"@openzeppelin/contracts@^4.7.3":
version "4.7.3"
resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-4.7.3.tgz#939534757a81f8d69cc854c7692805684ff3111e"
integrity sha512-dGRS0agJzu8ybo44pCIf3xBaPQN/65AIXNgK8+4gzKd5kbvlqyxryUYVLJv7fK98Seyd2hDZzVEHSWAh0Bt1Yw==
"@openzeppelin/wizard@^0.1.1":
version "0.1.1"
resolved "https://registry.yarnpkg.com/@openzeppelin/wizard/-/wizard-0.1.1.tgz#8c183e2c5748869bc3a5317c0330aa36a9ad44fe"
integrity sha512-AGyvn3PIh1vCgAEoRKAXKhtlk4fkA8AHE7G4PyzLnYcASClYCWpSf43WLJCs6S/LORvTZADX1flvF8x2LciJIg==
dependencies:
array.prototype.flatmap "^1.2.4"
"@pmmmwh/react-refresh-webpack-plugin@^0.4.3": "@pmmmwh/react-refresh-webpack-plugin@^0.4.3":
version "0.4.3" version "0.4.3"
resolved "https://registry.yarnpkg.com/@pmmmwh/react-refresh-webpack-plugin/-/react-refresh-webpack-plugin-0.4.3.tgz#1eec460596d200c0236bf195b078a5d1df89b766" resolved "https://registry.yarnpkg.com/@pmmmwh/react-refresh-webpack-plugin/-/react-refresh-webpack-plugin-0.4.3.tgz#1eec460596d200c0236bf195b078a5d1df89b766"

Loading…
Cancel
Save