From 92f50ce3a8792631a6c3800e211ed9580a46dfcf Mon Sep 17 00:00:00 2001 From: LianaHus Date: Thu, 28 May 2020 17:22:34 +0200 Subject: [PATCH 001/173] refactored debuggerUI. using SourceHighlight --- apps/remix-ide/src/app/tabs/debugger-tab.js | 2 +- .../src/app/tabs/debugger/debuggerUI.js | 75 ++++++++++--------- 2 files changed, 39 insertions(+), 38 deletions(-) diff --git a/apps/remix-ide/src/app/tabs/debugger-tab.js b/apps/remix-ide/src/app/tabs/debugger-tab.js index 84959edad2..c3fe77dbcd 100644 --- a/apps/remix-ide/src/app/tabs/debugger-tab.js +++ b/apps/remix-ide/src/app/tabs/debugger-tab.js @@ -56,8 +56,8 @@ class DebuggerTab extends ViewPlugin { }) this.debuggerUI = new DebuggerUI( + this, this.el.querySelector('#debugger'), - this.blockchain, (address, receipt) => { const target = (address && remixLib.helpers.trace.isContractCreation(address)) ? receipt.contractAddress : address return this.call('fetchAndCompile', 'resolve', target || receipt.contractAddress || receipt.to, '.debug', this.blockchain.web3()) diff --git a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js index 8359762412..72541d60c6 100644 --- a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js +++ b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js @@ -30,9 +30,8 @@ var css = csjs` class DebuggerUI { - constructor (container, blockchain, fetchContractAndCompile) { - this.registry = globalRegistry - this.blockchain = blockchain + constructor (debuggerModule, component, fetchContractAndCompile) { + this.debuggerModule = debuggerModule this.fetchContractAndCompile = fetchContractAndCompile this.event = new EventManager() @@ -48,62 +47,62 @@ class DebuggerUI { this.view - container.appendChild(this.render()) + component.appendChild(this.render()) this.setEditor() } setEditor () { - const self = this - this.editor = this.registry.get('editor').api + this.editor = globalRegistry.get('editor').api - self.editor.event.register('breakpointCleared', (fileName, row) => { - if (self.debugger) self.debugger.breakPointManager.remove({fileName: fileName, row: row}) + this.editor.event.register('breakpointCleared', (fileName, row) => { + if (this.debugger) this.debugger.breakPointManager.remove({fileName: fileName, row: row}) }) - self.editor.event.register('breakpointAdded', (fileName, row) => { - if (self.debugger) self.debugger.breakPointManager.add({fileName: fileName, row: row}) + this.editor.event.register('breakpointAdded', (fileName, row) => { + if (this.debugger) this.debugger.breakPointManager.add({fileName: fileName, row: row}) }) - self.editor.event.register('contentChanged', function () { - if (self.debugger) self.debugger.unload() + this.editor.event.register('contentChanged', () => { + if (this.debugger) this.debugger.unload() }) } listenToEvents () { - const self = this - if (!self.debugger) return + if (!this.debugger) return - this.debugger.event.register('debuggerStatus', function (isActive) { - self.sourceHighlighter.currentSourceLocation(null) - self.isActive = isActive + this.debugger.event.register('debuggerStatus', async (isActive) => { + await this.debuggerModule.call('editor', 'discardHighlight') + this.isActive = isActive }) - this.debugger.event.register('newSourceLocation', async function (lineColumnPos, rawLocation) { - const contracts = await self.fetchContractAndCompile( - self.currentReceipt.contractAddress || self.currentReceipt.to, - self.currentReceipt) + this.debugger.event.register('newSourceLocation', async (lineColumnPos, rawLocation) => { + const contracts = await this.fetchContractAndCompile( + this.currentReceipt.contractAddress || this.currentReceipt.to, + this.currentReceipt) if (contracts) { const path = contracts.getSourceName(rawLocation.file) - if (path) self.sourceHighlighter.currentSourceLocationFromfileName(lineColumnPos, path) + if (path) { + await this.debuggerModule.call('editor', 'discardHighlight') + await this.debuggerModule.call('editor', 'highlight', lineColumnPos, path) + } } }) - this.debugger.event.register('debuggerUnloaded', self.unLoad.bind(this)) + this.debugger.event.register('debuggerUnloaded', () => unLoad) } startTxBrowser () { - const self = this let txBrowser = new TxBrowser() this.txBrowser = txBrowser - txBrowser.event.register('requestDebug', function (blockNumber, txNumber, tx) { - if (self.debugger) self.debugger.unload() - self.startDebugging(blockNumber, txNumber, tx) + txBrowser.event.register('requestDebug', (blockNumber, txNumber, tx) => { + if (this.debugger) this.debugger.unload() + this.startDebugging(blockNumber, txNumber, tx) }) - txBrowser.event.register('unloadRequested', this, function (blockNumber, txIndex, tx) { - if (self.debugger) self.debugger.unload() + txBrowser.event.register('unloadRequested', this, (blockNumber, txIndex, tx) => { + if (this.debugger) this.debugger.unload() }) } @@ -113,13 +112,13 @@ class DebuggerUI { getDebugWeb3 () { return new Promise((resolve, reject) => { - this.blockchain.detectNetwork((error, network) => { + this.debuggerModule.blockchain.detectNetwork((error, network) => { let web3 if (error || !network) { - web3 = init.web3DebugNode(this.blockchain.web3()) + web3 = init.web3DebugNode(this.debuggerModule.blockchain.web3()) } else { const webDebugNode = init.web3DebugNode(network.name) - web3 = !webDebugNode ? this.blockchain.web3() : webDebugNode + web3 = !webDebugNode ? this.debuggerModule.blockchain.web3() : webDebugNode } init.extendWeb3(web3) resolve(web3) @@ -134,7 +133,7 @@ class DebuggerUI { this.currentReceipt = await web3.eth.getTransactionReceipt(txNumber) this.debugger = new Debugger({ web3, - offsetToLineColumnConverter: this.registry.get('offsettolinecolumnconverter').api, + offsetToLineColumnConverter: globalRegistry.get('offsettolinecolumnconverter').api, compilationResult: async (address) => { try { return await this.fetchContractAndCompile(address, this.currentReceipt) @@ -164,7 +163,7 @@ class DebuggerUI { this.currentReceipt = await web3.eth.getTransactionReceipt(hash) const debug = new Debugger({ web3, - offsetToLineColumnConverter: this.registry.get('offsettolinecolumnconverter').api, + offsetToLineColumnConverter: globalRegistry.get('offsettolinecolumnconverter').api, compilationResult: async (address) => { try { return await this.fetchContractAndCompile(address, this.currentReceipt) @@ -190,15 +189,17 @@ class DebuggerUI { this.debuggerHeadPanelsView = yo`
` this.stepManagerView = yo`
` - var view = yo`
+ var view = yo` +
${this.txBrowser.render()} ${this.debuggerHeadPanelsView} ${this.stepManagerView}
-
${this.statusMessage}
+
${this.statusMessage}
${this.debuggerPanelsView} -
` +
+ ` if (!this.view) { this.view = view } From 216aed0d0622713fdedee93d328df03910a29d31 Mon Sep 17 00:00:00 2001 From: LianaHus Date: Thu, 28 May 2020 18:11:10 +0200 Subject: [PATCH 002/173] fixed unLoad --- apps/remix-ide/src/app/tabs/debugger/debuggerUI.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js index 72541d60c6..dd9c50bd0d 100644 --- a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js +++ b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js @@ -89,7 +89,7 @@ class DebuggerUI { } }) - this.debugger.event.register('debuggerUnloaded', () => unLoad) + this.debugger.event.register('debuggerUnloaded', () => this.unLoad()) } startTxBrowser () { From 3fbb135ba0dc0e39b5a361fcabbf1ecf03f55710 Mon Sep 17 00:00:00 2001 From: LianaHus Date: Thu, 28 May 2020 19:40:24 +0200 Subject: [PATCH 003/173] remove highlights on unload --- apps/remix-ide/src/app/tabs/debugger-tab.js | 8 +++++++- apps/remix-ide/src/app/tabs/debugger/debuggerUI.js | 3 ++- 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/apps/remix-ide/src/app/tabs/debugger-tab.js b/apps/remix-ide/src/app/tabs/debugger-tab.js index c3fe77dbcd..71f1299fe8 100644 --- a/apps/remix-ide/src/app/tabs/debugger-tab.js +++ b/apps/remix-ide/src/app/tabs/debugger-tab.js @@ -61,7 +61,8 @@ class DebuggerTab extends ViewPlugin { (address, receipt) => { const target = (address && remixLib.helpers.trace.isContractCreation(address)) ? receipt.contractAddress : address return this.call('fetchAndCompile', 'resolve', target || receipt.contractAddress || receipt.to, '.debug', this.blockchain.web3()) - }) + } + ) this.call('manager', 'activatePlugin', 'source-verification') // this.call('manager', 'activatePlugin', 'udapp') @@ -69,6 +70,11 @@ class DebuggerTab extends ViewPlugin { return this.el } + deactivate () { + this.debuggerUI.unLoad() + super.deactivate() + } + debug (hash) { if (this.debuggerUI) this.debuggerUI.debug(hash) } diff --git a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js index dd9c50bd0d..2bbf18aaff 100644 --- a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js +++ b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js @@ -206,7 +206,8 @@ class DebuggerUI { return view } - unLoad () { + async unLoad () { + await this.debuggerModule.call('editor', 'discardHighlight') yo.update(this.debuggerHeadPanelsView, yo`
`) yo.update(this.debuggerPanelsView, yo`
`) yo.update(this.stepManagerView, yo`
`) From 85736a1e0bddcb38aa8d65cdb0759d040fb2f83a Mon Sep 17 00:00:00 2001 From: LianaHus Date: Fri, 29 May 2020 14:46:41 +0200 Subject: [PATCH 004/173] remove only highlight --- apps/remix-ide/src/app/tabs/debugger-tab.js | 2 +- apps/remix-ide/src/app/tabs/debugger/debuggerUI.js | 5 ++++- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/apps/remix-ide/src/app/tabs/debugger-tab.js b/apps/remix-ide/src/app/tabs/debugger-tab.js index 71f1299fe8..a2ed6f1088 100644 --- a/apps/remix-ide/src/app/tabs/debugger-tab.js +++ b/apps/remix-ide/src/app/tabs/debugger-tab.js @@ -71,7 +71,7 @@ class DebuggerTab extends ViewPlugin { } deactivate () { - this.debuggerUI.unLoad() + this.debuggerUI.deletHighlights() super.deactivate() } diff --git a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js index 2bbf18aaff..9d8d5702bc 100644 --- a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js +++ b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js @@ -207,7 +207,6 @@ class DebuggerUI { } async unLoad () { - await this.debuggerModule.call('editor', 'discardHighlight') yo.update(this.debuggerHeadPanelsView, yo`
`) yo.update(this.debuggerPanelsView, yo`
`) yo.update(this.stepManagerView, yo`
`) @@ -220,6 +219,10 @@ class DebuggerUI { this.event.trigger('traceUnloaded') } + async deletHighlights () { + await this.debuggerModule.call('editor', 'discardHighlight') + } + renderDebugger () { yo.update(this.debuggerHeadPanelsView, this.vmDebugger.renderHead()) yo.update(this.debuggerPanelsView, this.vmDebugger.render()) From b3fe4d21c0c034923ada72d97eeb8abeb90bf326 Mon Sep 17 00:00:00 2001 From: LianaHus Date: Sun, 31 May 2020 14:03:31 +0200 Subject: [PATCH 005/173] typo --- apps/remix-ide/src/app/tabs/debugger-tab.js | 2 +- apps/remix-ide/src/app/tabs/debugger/debuggerUI.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/apps/remix-ide/src/app/tabs/debugger-tab.js b/apps/remix-ide/src/app/tabs/debugger-tab.js index a2ed6f1088..42db476705 100644 --- a/apps/remix-ide/src/app/tabs/debugger-tab.js +++ b/apps/remix-ide/src/app/tabs/debugger-tab.js @@ -71,7 +71,7 @@ class DebuggerTab extends ViewPlugin { } deactivate () { - this.debuggerUI.deletHighlights() + this.debuggerUI.deleteHighlights() super.deactivate() } diff --git a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js index 9d8d5702bc..f45723a9a2 100644 --- a/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js +++ b/apps/remix-ide/src/app/tabs/debugger/debuggerUI.js @@ -219,7 +219,7 @@ class DebuggerUI { this.event.trigger('traceUnloaded') } - async deletHighlights () { + async deleteHighlights () { await this.debuggerModule.call('editor', 'discardHighlight') } From dfa8d5cc33a2e7f8a34b868764480669e2e9b6a6 Mon Sep 17 00:00:00 2001 From: LianaHus Date: Mon, 25 May 2020 17:13:48 +0200 Subject: [PATCH 006/173] new light theme --- apps/remix-ide/src/app/tabs/theme-module.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/remix-ide/src/app/tabs/theme-module.js b/apps/remix-ide/src/app/tabs/theme-module.js index 36ee8df523..0535941bad 100644 --- a/apps/remix-ide/src/app/tabs/theme-module.js +++ b/apps/remix-ide/src/app/tabs/theme-module.js @@ -5,7 +5,7 @@ import yo from 'yo-yo' const themes = [ {name: 'Dark', quality: 'dark', url: 'https://res.cloudinary.com/dvtmp0niu/raw/upload/v1584965247/remix-dark_tmkdla.css'}, - {name: 'Light', quality: 'light', url: 'https://res.cloudinary.com/dvtmp0niu/raw/upload/v1584966540/remix-light_t0c780.css'}, + {name: 'Light', quality: 'light', url: 'https://res.cloudinary.com/lianahus/raw/upload/v1590154457/remix-themes/remix-light_lg1arb.css'}, {name: 'Cerulean', quality: 'light', url: 'https://bootswatch.com/4/cerulean/bootstrap.min.css'}, {name: 'Flatly', quality: 'light', url: 'https://bootswatch.com/4/flatly/bootstrap.min.css'}, From 4448e895e36a74b555e36f7a683051a92dd71f6e Mon Sep 17 00:00:00 2001 From: LianaHus Date: Thu, 28 May 2020 01:12:54 +0200 Subject: [PATCH 007/173] new Ace editor dark theme --- apps/remix-ide/src/app/editor/editor.js | 2 +- assets/js/editor/darkTheme.js | 180 ++++++++++++++++++++++++ 2 files changed, 181 insertions(+), 1 deletion(-) create mode 100644 assets/js/editor/darkTheme.js diff --git a/apps/remix-ide/src/app/editor/editor.js b/apps/remix-ide/src/app/editor/editor.js index bf0055562b..3d429697d4 100644 --- a/apps/remix-ide/src/app/editor/editor.js +++ b/apps/remix-ide/src/app/editor/editor.js @@ -20,8 +20,8 @@ require('brace/mode/javascript') require('brace/mode/python') require('brace/mode/json') require('brace/mode/rust') -require('brace/theme/chaos') require('brace/theme/chrome') +require('../../../assets/js/editor/darkTheme') const css = csjs` .ace-editor { diff --git a/assets/js/editor/darkTheme.js b/assets/js/editor/darkTheme.js new file mode 100644 index 0000000000..9c7e68f0be --- /dev/null +++ b/assets/js/editor/darkTheme.js @@ -0,0 +1,180 @@ +ace.define("ace/theme/chaos",["require","exports","module","ace/lib/dom"], function(acequire, exports, module) { + + exports.isDark = true; + exports.cssClass = "ace-chaos"; + exports.cssText = ".ace-chaos .ace_gutter {\ + background: #2a2c3f;\ + color: #8789a1;\ + border-right: 1px solid #282828;\ + }\ + .ace-chaos .ace_gutter-cell.ace_warning {\ + background-image: none;\ + background: #FC0;\ + border-left: none;\ + padding-left: 0;\ + color: #000;\ + }\ + .ace-chaos .ace_gutter-cell.ace_error {\ + background-position: -6px center;\ + background-image: none;\ + background: #F10;\ + border-left: none;\ + padding-left: 0;\ + color: #000;\ + }\ + .ace-chaos .ace_print-margin {\ + border-left: 1px solid #555;\ + right: 0;\ + background: #1D1D1D;\ + }\ + .ace-chaos {\ + background-color: #222336;\ + color: #a2a3bd;\ + }\ + .ace-chaos .ace_cursor {\ + border-left: 2px solid #FFFFFF;\ + }\ + .ace-chaos .ace_cursor.ace_overwrite {\ + border-left: 0px;\ + border-bottom: 1px solid #FFFFFF;\ + }\ + .ace-chaos .ace_marker-layer .ace_selection {\ + background: #494836;\ + }\ + .ace-chaos .ace_marker-layer .ace_step {\ + background: rgb(198, 219, 174);\ + }\ + .ace-chaos .ace_marker-layer .ace_bracket {\ + margin: -1px 0 0 -1px;\ + border: 1px solid #FCE94F;\ + }\ + .ace-chaos .ace_marker-layer .ace_active-line {\ + background: #363950;\ + }\ + .ace-chaos .ace_gutter-active-line {\ + background-color: #363950;\ + }\ + .ace-chaos .ace_invisible {\ + color: #404040;\ + }\ + .ace-chaos .ace_rparen {\ + color: #d4d7ed;\ + }\ + .ace-chaos .ace_lparen {\ + color: #d4d7ed;\ + }\ + .ace-chaos .ace_keyword {\ + color:#ffa76d;\ + }\ + .ace-chaos .ace_keyword.ace_operator {\ + color:#eceeff;\ + }\ + .ace-chaos .ace_constant {\ + color:#1EDAFB;\ + }\ + .ace-chaos .ace_constant.ace_language {\ + color:#FDC251;\ + }\ + .ace-chaos .ace_constant.ace_library {\ + color:#8DFF0A;\ + }\ + .ace-chaos .ace_constant.ace_numeric {\ + color:#eceeff;\ + }\ + .ace-chaos .ace_invalid {\ + color:#FFFFFF;\ + background-color:#990000;\ + }\ + .ace-chaos .ace_invalid.ace_deprecated {\ + color:#FFFFFF;\ + background-color:#990000;\ + }\ + .ace-chaos .ace_support {\ + color: #999;\ + }\ + .ace-chaos .ace_support.ace_function {\ + color:#3fe2a7;\ + }\ + .ace-chaos .ace_function {\ + color:#3fe2a7;\ + }\ + .ace-chaos .ace_string {\ + color:#eceeff;\ + }\ + .ace-chaos .ace_comment {\ + color:#a7a7a7;\ + font-style:italic;\ + padding-bottom: 0px;\ + }\ + .ace-chaos .ace_type {\ + color:#75ceef;\ + }]\ + .ace-chaos .ace_visibility (\ + color:#f7d777;\ + )\ + .ace-chaos .ace_identifier {\ + color:#bec1dd;\ + }\ + .ace-chaos .ace_modifier {\ + color:#efff2f;\ + }\ + .ace-chaos .ace-boolean {\ + color:#ff86ac;\ + }\ + .ace-chaos .ace_statemutability {\ + color:#ff8181;\ + }\ + .ace-chaos .ace_variable {\ + color:#e0bb83;\ + }\ + .ace-chaos .ace_meta.ace_tag {\ + color:#BE53E6;\ + }\ + .ace-chaos .ace_entity.ace_other.ace_attribute-name {\ + color:#4aa8fd;\ + }\ + .ace-chaos .ace_markup.ace_underline {\ + text-decoration: underline;\ + }\ + .ace-chaos .ace_fold-widget {\ + text-align: center;\ + }\ + .ace-chaos .ace_fold-widget:hover {\ + color: #777;\ + }\ + .ace-chaos .ace_fold-widget.ace_start,\ + .ace-chaos .ace_fold-widget.ace_end,\ + .ace-chaos .ace_fold-widget.ace_closed{\ + background: none;\ + border: none;\ + box-shadow: none;\ + }\ + .ace-chaos .ace_fold-widget.ace_start:after {\ + content: '▾'\ + }\ + .ace-chaos .ace_fold-widget.ace_end:after {\ + content: '▴'\ + }\ + .ace-chaos .ace_fold-widget.ace_closed:after {\ + content: '‣'\ + }\ + .ace-chaos .ace_indent-guide {\ + border-right:1px dotted #333;\ + margin-right:-1px;\ + }\ + .ace-chaos .ace_fold { \ + background: #222; \ + border-radius: 3px; \ + color: #7AF; \ + border: none; \ + }\ + .ace-chaos .ace_fold:hover {\ + background: #CCC; \ + color: #000;\ + }\ + "; + + var dom = acequire("../lib/dom"); + dom.importCssString(exports.cssText, exports.cssClass); + + }); \ No newline at end of file From 724577b7ccacdaa285bb76d3fad0b3b7704d9285 Mon Sep 17 00:00:00 2001 From: LianaHus Date: Thu, 28 May 2020 02:31:59 +0200 Subject: [PATCH 008/173] converted base64 formats to webp --- .../src/app/components/local-plugin.js | 2 +- .../app/components/plugin-manager-component.js | 2 +- apps/remix-ide/src/app/files/fileManager.js | 2 +- apps/remix-ide/src/app/panels/file-panel.js | 2 +- .../src/app/panels/styles/terminal-styles.js | 1 - apps/remix-ide/src/app/tabs/analysis-tab.js | 2 +- apps/remix-ide/src/app/tabs/compile-tab.js | 9 +++------ apps/remix-ide/src/app/tabs/debugger-tab.js | 2 +- apps/remix-ide/src/app/tabs/settings-tab.js | 2 +- apps/remix-ide/src/app/tabs/test-tab.js | 2 +- apps/remix-ide/src/app/udapp/run-tab.js | 2 +- .../src/app/ui/landing-page/landing-page.js | 2 +- assets/img/deployAndRun.webp | Bin 0 -> 32324 bytes assets/img/fileManager.webp | Bin 0 -> 12312 bytes assets/img/ipfs.webp | Bin 0 -> 22016 bytes assets/img/localPlugin.webp | Bin 0 -> 44272 bytes assets/img/pluginManager.webp | Bin 0 -> 16914 bytes assets/img/remixLogo.webp | Bin 0 -> 4520 bytes assets/img/settings.webp | Bin 0 -> 23508 bytes assets/img/solidity.webp | Bin 0 -> 9248 bytes assets/img/staticAnalysis.webp | Bin 0 -> 15158 bytes assets/img/swarm.webp | Bin 0 -> 28736 bytes assets/img/unitTesting.webp | Bin 0 -> 408 bytes assets/js/editor/darkTheme.js | 2 +- 24 files changed, 14 insertions(+), 18 deletions(-) create mode 100644 assets/img/deployAndRun.webp create mode 100644 assets/img/fileManager.webp create mode 100644 assets/img/ipfs.webp create mode 100644 assets/img/localPlugin.webp create mode 100644 assets/img/pluginManager.webp create mode 100644 assets/img/remixLogo.webp create mode 100644 assets/img/settings.webp create mode 100644 assets/img/solidity.webp create mode 100644 assets/img/staticAnalysis.webp create mode 100644 assets/img/swarm.webp create mode 100644 assets/img/unitTesting.webp diff --git a/apps/remix-ide/src/app/components/local-plugin.js b/apps/remix-ide/src/app/components/local-plugin.js index f71e10d7bb..39d8c5fc68 100644 --- a/apps/remix-ide/src/app/components/local-plugin.js +++ b/apps/remix-ide/src/app/components/local-plugin.js @@ -37,7 +37,7 @@ module.exports = class LocalPlugin { */ create () { const profile = { - icon: '', + icon: 'assets/img/localPlugin.webp', methods: [], location: 'sidePanel', type: 'iframe', diff --git a/apps/remix-ide/src/app/components/plugin-manager-component.js b/apps/remix-ide/src/app/components/plugin-manager-component.js index 2b95d250e7..3772b55a9b 100644 --- a/apps/remix-ide/src/app/components/plugin-manager-component.js +++ b/apps/remix-ide/src/app/components/plugin-manager-component.js @@ -62,7 +62,7 @@ const profile = { displayName: 'Plugin manager', methods: [], events: [], - icon: '', + icon: 'assets/img/pluginManager.webp', description: 'Start/stop services, modules and plugins', kind: 'settings', location: 'sidePanel', diff --git a/apps/remix-ide/src/app/files/fileManager.js b/apps/remix-ide/src/app/files/fileManager.js index 2d75ef6f67..fe74117a3a 100644 --- a/apps/remix-ide/src/app/files/fileManager.js +++ b/apps/remix-ide/src/app/files/fileManager.js @@ -20,7 +20,7 @@ const profile = { name: 'fileManager', displayName: 'File manager', description: 'Service - read/write to any files or folders, require giving permissions', - icon: '', + icon: 'assets/img/fileManager.webp', permission: true, version: packageJson.version, methods: ['file', 'exists', 'open', 'writeFile', 'readFile', 'copyFile', 'rename', 'readdir', 'remove', 'getCurrentFile', 'getFile', 'getFolder', 'setFile', 'switchFile'], diff --git a/apps/remix-ide/src/app/panels/file-panel.js b/apps/remix-ide/src/app/panels/file-panel.js index 9ded1bcaac..e0c42c749e 100644 --- a/apps/remix-ide/src/app/panels/file-panel.js +++ b/apps/remix-ide/src/app/panels/file-panel.js @@ -32,7 +32,7 @@ const profile = { displayName: 'File explorers', methods: [], events: [], - icon: '', + icon: 'assets/img/fileManager.webp', description: ' - ', kind: 'fileexplorer', location: 'sidePanel', diff --git a/apps/remix-ide/src/app/panels/styles/terminal-styles.js b/apps/remix-ide/src/app/panels/styles/terminal-styles.js index 46dd30cf22..1aed65852b 100644 --- a/apps/remix-ide/src/app/panels/styles/terminal-styles.js +++ b/apps/remix-ide/src/app/panels/styles/terminal-styles.js @@ -46,7 +46,6 @@ var css = csjs` overflow-y : auto; font-family : monospace; margin : 0px; - background-image : url('') background-repeat : no-repeat; background-position : center 15%; background-size : auto calc(75% - 1.7em); diff --git a/apps/remix-ide/src/app/tabs/analysis-tab.js b/apps/remix-ide/src/app/tabs/analysis-tab.js index e1cec92d26..78b45b9c20 100644 --- a/apps/remix-ide/src/app/tabs/analysis-tab.js +++ b/apps/remix-ide/src/app/tabs/analysis-tab.js @@ -11,7 +11,7 @@ const profile = { displayName: 'Solidity static analysis', methods: [], events: [], - icon: '', + icon: 'assets/img/staticAnalysis.webp', description: 'Checks the contract code for security vulnerabilities and bad practices.', kind: 'analysis', location: 'sidePanel', diff --git a/apps/remix-ide/src/app/tabs/compile-tab.js b/apps/remix-ide/src/app/tabs/compile-tab.js index 1efb67415d..a5d8e4de10 100644 --- a/apps/remix-ide/src/app/tabs/compile-tab.js +++ b/apps/remix-ide/src/app/tabs/compile-tab.js @@ -24,7 +24,7 @@ import publishToStorage from '../../publishToStorage' const profile = { name: 'solidity', displayName: 'Solidity compiler', - icon: '', + icon: 'assets/img/solidity.webp', description: 'Compile solidity contracts', kind: 'compiler', permission: true, @@ -243,9 +243,6 @@ class CompileTab extends ViewPlugin { ` // define swarm logo - const swarmImg = '' - - const ipfsImg = '' let result = contractList.length ? yo`
@@ -257,11 +254,11 @@ class CompileTab extends ViewPlugin {
- diff --git a/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js b/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js index b91f933bf3..9c3fbfb50a 100644 --- a/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js +++ b/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js @@ -318,17 +318,21 @@ class CompilerContainer { this.compileIfAutoCompileOn() } + /* + The following functions are handlers for internal events. + */ + onchangeOptimize () { this.compileTabLogic.setOptimize(!!this._view.optimize.checked) this.compileIfAutoCompileOn() } - onchangeLanguage (event) { - this.compileTabLogic.setLanguage(event.target.value) + onchangeLanguage () { + this.compileTabLogic.setLanguage(this._view.languageSelector.value) this.compileIfAutoCompileOn() } - onchangeEvmVersion (_) { + onchangeEvmVersion () { let s = this._view.evmVersionSelector let v = s.value if (v === 'default') { @@ -338,12 +342,37 @@ class CompilerContainer { this.compileIfAutoCompileOn() } - onchangeLoadVersion (event) { + onchangeLoadVersion () { this.data.selectedVersion = this._view.versionSelector.value this._updateVersionSelector() this._updateLanguageSelector() } + /* + The following functions map with the above event handlers. + They are an external API for modifying the compiler configuration. + */ + + setOptimize (enabled) { + this._view.optimize.checked = enabled + this.onchangeOptimize() + } + + setLanguage (lang) { + this._view.languageSelector.value = lang + this.onchangeLanguage() + } + + setEvmVersion (version) { + this._view.evmVersionSelector.value = version || 'default' + this.onchangeEvmVersion() + } + + setVersion (version) { + this._view.versionSelector.value = `soljson-v${version}.js` + this.onchangeLoadVersion() + } + _shouldBeAdded (version) { return !version.includes('nightly') || (version.includes('nightly') && this._view.includeNightlies.checked) diff --git a/apps/remix-ide/test-browser/commands/verifyContracts.js b/apps/remix-ide/test-browser/commands/verifyContracts.js index 14f006504e..3275cfb65a 100644 --- a/apps/remix-ide/test-browser/commands/verifyContracts.js +++ b/apps/remix-ide/test-browser/commands/verifyContracts.js @@ -1,7 +1,7 @@ const EventEmitter = require('events') class VerifyContracts extends EventEmitter { - command (compiledContractNames, opts = { wait: 1000 }) { + command (compiledContractNames, opts = { wait: 1000, version: null }) { this.api.perform((done) => { verifyContracts(this.api, compiledContractNames, opts, () => { done() @@ -17,6 +17,17 @@ function getCompiledContracts (browser, opts, callback) { .clickLaunchIcon('solidity') .pause(opts.wait) .waitForElementPresent('*[data-id="compiledContracts"] option') + .perform((done) => { + if (opts.version) { + browser + .click('*[data-id="compilation-details"]') + .waitForElementPresent('*[data-id="treeViewLicompiler"]') + .click('*[data-id="treeViewLicompiler"]') + .waitForElementPresent('*[data-id="treeViewTogglecompiler/version"]') + .assert.containsText('*[data-id="treeViewLicompiler/version"]', `version: ${opts.version}`) + .perform(done) + } else done() + }) .execute(function () { var contracts = document.querySelectorAll('*[data-id="compiledContracts"] option') if (!contracts) { diff --git a/package.json b/package.json index 63a2ed15e8..5d36e605f7 100644 --- a/package.json +++ b/package.json @@ -246,8 +246,8 @@ "watchify": "^3.9.0", "web3": "1.2.4", "webworkify": "^1.2.1", -"yo-yo": "github:ioedeveloper/yo-yo", -"yo-yoify": "^3.7.3", -"webworkify-webpack": "^2.1.5" + "yo-yo": "github:ioedeveloper/yo-yo", + "yo-yoify": "^3.7.3", + "webworkify-webpack": "^2.1.5" } } diff --git a/test-browser/tests/compiler_api.test.js b/test-browser/tests/compiler_api.test.js new file mode 100644 index 0000000000..2c267b8177 --- /dev/null +++ b/test-browser/tests/compiler_api.test.js @@ -0,0 +1,101 @@ +'use strict' +var examples = require('../../src/app/editor/example-contracts') +var init = require('../helpers/init') +var sauce = require('./sauce') + +var sources = [ + {'browser/Untitled.sol': {content: examples.ballot.content}} +] + +module.exports = { + before: function (browser, done) { + init(browser, done) + }, + '@sources': function () { + return sources + }, + + 'Should compile using "compileWithParamaters" API': function (browser) { + browser + .addFile('test_jsCompile.js', { content: jsCompile }) + .executeScript('remix.exeCurrent()') + .pause(5000) + .journalChildIncludes('"languageversion": "0.6.8+commit.0bbfe453"') + }, + + 'Should update the compiler configuration with "setCompilerConfig" API': function (browser) { + browser + .addFile('test_updateConfiguration.js', { content: updateConfiguration }) + .executeScript('remix.exeCurrent()') + .pause(5000) + .addFile('test_updateConfiguration.sol', { content: simpleContract }) + .verifyContracts(['StorageTestUpdateConfiguration'], {wait: 5000, version: '0.6.8+commit.0bbfe453'}) + .end() + }, + + tearDown: sauce +} + +const simpleContract = `pragma solidity >=0.4.22 <0.7.0; + +/** +* @title Storage +* @dev Store & retreive value in a variable +*/ +contract StorageTestUpdateConfiguration { + + uint256 number; + + /** + * @dev Store value in variable + * @param num value to store + */ + function store(uint256 num) public { + number = num; + } + + /** + * @dev Return value + * @return value of 'number' + */ + function retreive() public view returns (uint256){ + return number; + } +} + + ` + +const jsCompile = `(async () => { + + try { + const contract = { + "storage.sol": {content : \`${simpleContract}\` } + } + console.log('compile') + const params = { + optimize: false, + evmVersion: null, + language: 'Solidity', + version: '0.6.8+commit.0bbfe453', + compilerUrl: 'https://solc-bin.ethereum.org/bin/soljson-v0.6.8+commit.0bbfe453.js' + } + const result = await remix.call('solidity', 'compileWithParameters', contract, params) + console.log('result ', result) + } catch (e) { + console.log(e.message) + } +})()` + +const updateConfiguration = `(async () => { + try { + const params = { + optimize: false, + evmVersion: null, + language: 'Solidity', + version: '0.6.8+commit.0bbfe453' + } + await remix.call('solidity', 'setCompilerConfig', params) + } catch (e) { + console.log(e.message) + } +})()` From b9279a9b9dd149031bf92a9ea1568a19a4b3655f Mon Sep 17 00:00:00 2001 From: yann300 Date: Wed, 3 Jun 2020 11:36:08 +0200 Subject: [PATCH 059/173] run tab listen on yulp --- apps/remix-ide/src/app/compiler/compiler-artefacts.js | 4 ++++ apps/remix-ide/src/app/compiler/compiler-helpers.js | 1 - apps/remix-ide/src/app/tabs/runTab/model/dropdownlogic.js | 3 +++ 3 files changed, 7 insertions(+), 1 deletion(-) diff --git a/apps/remix-ide/src/app/compiler/compiler-artefacts.js b/apps/remix-ide/src/app/compiler/compiler-artefacts.js index ecc25cbe93..f0f88bb069 100644 --- a/apps/remix-ide/src/app/compiler/compiler-artefacts.js +++ b/apps/remix-ide/src/app/compiler/compiler-artefacts.js @@ -32,6 +32,10 @@ module.exports = class CompilerArtefacts extends Plugin { this.on('lexon', 'compilationFinished', (file, source, languageVersion, data) => { this.compilersArtefacts['__last'] = new CompilerAbstract(languageVersion, data, source) }) + + this.on('yupl', 'compilationFinished', (file, source, languageVersion, data) => { + this.compilersArtefacts['__last'] = new CompilerAbstract(languageVersion, data, source) + }) } addResolvedContract (address, compilerData) { diff --git a/apps/remix-ide/src/app/compiler/compiler-helpers.js b/apps/remix-ide/src/app/compiler/compiler-helpers.js index 5bb0594e96..e820872bbd 100644 --- a/apps/remix-ide/src/app/compiler/compiler-helpers.js +++ b/apps/remix-ide/src/app/compiler/compiler-helpers.js @@ -12,7 +12,6 @@ export const compile = async (compilationTargets, settings) => { compiler.set('language', settings.language) compiler.loadVersion(canUseWorker(settings.version), urlFromVersion(settings.version)) compiler.event.register('compilationFinished', (success, compilationData, source) => { - console.log(success, compilationData) if (!success) return reject(compilationData) resolve(new CompilerAbstract(settings.version, compilationData, source)) }) diff --git a/apps/remix-ide/src/app/tabs/runTab/model/dropdownlogic.js b/apps/remix-ide/src/app/tabs/runTab/model/dropdownlogic.js index c2dc932aa1..0a42977a40 100644 --- a/apps/remix-ide/src/app/tabs/runTab/model/dropdownlogic.js +++ b/apps/remix-ide/src/app/tabs/runTab/model/dropdownlogic.js @@ -34,6 +34,9 @@ class DropdownLogic { this.runView.on('lexon', 'compilationFinished', (file, source, languageVersion, data) => broadcastCompilationResult(file, source, languageVersion, data) ) + this.runView.on('yulp', 'compilationFinished', (file, source, languageVersion, data) => + broadcastCompilationResult(file, source, languageVersion, data) + ) } loadContractFromAddress (address, confirmCb, cb) { From 769db84b8e008e6ea6f2f422ef5b85d5897f0a37 Mon Sep 17 00:00:00 2001 From: yann300 Date: Wed, 3 Jun 2020 11:50:15 +0200 Subject: [PATCH 060/173] fix e2e test --- apps/remix-ide/test-browser/commands/verifyContracts.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/apps/remix-ide/test-browser/commands/verifyContracts.js b/apps/remix-ide/test-browser/commands/verifyContracts.js index 3275cfb65a..c16dbfabe0 100644 --- a/apps/remix-ide/test-browser/commands/verifyContracts.js +++ b/apps/remix-ide/test-browser/commands/verifyContracts.js @@ -21,9 +21,10 @@ function getCompiledContracts (browser, opts, callback) { if (opts.version) { browser .click('*[data-id="compilation-details"]') - .waitForElementPresent('*[data-id="treeViewLicompiler"]') - .click('*[data-id="treeViewLicompiler"]') - .waitForElementPresent('*[data-id="treeViewTogglecompiler/version"]') + .waitForElementVisible('*[data-id="treeViewDivcompiler"]') + .pause(2000) + .click('*[data-id="treeViewDivcompiler"]') + .waitForElementVisible('*[data-id="treeViewLicompiler/version"]') .assert.containsText('*[data-id="treeViewLicompiler/version"]', `version: ${opts.version}`) .perform(done) } else done() From c2ebfd7404773e902bebf5f8f3b22f53d6e65bf0 Mon Sep 17 00:00:00 2001 From: yann300 Date: Tue, 9 Jun 2020 15:27:47 +0200 Subject: [PATCH 061/173] typo --- apps/remix-ide/src/app/compiler/compiler-artefacts.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/remix-ide/src/app/compiler/compiler-artefacts.js b/apps/remix-ide/src/app/compiler/compiler-artefacts.js index f0f88bb069..3fcc312c2b 100644 --- a/apps/remix-ide/src/app/compiler/compiler-artefacts.js +++ b/apps/remix-ide/src/app/compiler/compiler-artefacts.js @@ -33,7 +33,7 @@ module.exports = class CompilerArtefacts extends Plugin { this.compilersArtefacts['__last'] = new CompilerAbstract(languageVersion, data, source) }) - this.on('yupl', 'compilationFinished', (file, source, languageVersion, data) => { + this.on('yulp', 'compilationFinished', (file, source, languageVersion, data) => { this.compilersArtefacts['__last'] = new CompilerAbstract(languageVersion, data, source) }) } From fe589269ec03f1613a38b3cbac89b7a53a655934 Mon Sep 17 00:00:00 2001 From: yann300 Date: Tue, 9 Jun 2020 15:31:53 +0200 Subject: [PATCH 062/173] refactor to setConfiguration --- apps/remix-ide/src/app/tabs/compile-tab.js | 5 +---- .../remix-ide/src/app/tabs/compileTab/compilerContainer.js | 7 +++++++ 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/apps/remix-ide/src/app/tabs/compile-tab.js b/apps/remix-ide/src/app/tabs/compile-tab.js index e027e1e5ce..b41dec9bbf 100644 --- a/apps/remix-ide/src/app/tabs/compile-tab.js +++ b/apps/remix-ide/src/app/tabs/compile-tab.js @@ -246,10 +246,7 @@ class CompileTab extends ViewPlugin { setCompilerConfig (settings) { return new Promise((resolve, reject) => { addTooltip(yo`
${this.currentRequest.from} is updating the Solidity compiler configuration.
${JSON.stringify(settings, null, '\t')}
`) - this.compilerContainer.setLanguage(settings.language) - this.compilerContainer.setEvmVersion(settings.evmVersion) - this.compilerContainer.setOptimize(settings.optimize) - this.compilerContainer.setVersion(settings.version) + this.compilerContainer.setConfiguration(settings) // @todo(#2875) should use loading compiler return value to check whether the compiler is loaded instead of "setInterval" let timeout = 0 const id = setInterval(() => { diff --git a/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js b/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js index 9c3fbfb50a..d5b6449d35 100644 --- a/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js +++ b/apps/remix-ide/src/app/tabs/compileTab/compilerContainer.js @@ -353,6 +353,13 @@ class CompilerContainer { They are an external API for modifying the compiler configuration. */ + setConfiguration (settings) { + this.setLanguage(settings.language) + this.setEvmVersion(settings.evmVersion) + this.setOptimize(settings.optimize) + this.setVersion(settings.version) + } + setOptimize (enabled) { this._view.optimize.checked = enabled this.onchangeOptimize() From 04d97a4d98b3d7553f0a5c9cbaaa3efd73d8208a Mon Sep 17 00:00:00 2001 From: Jonas Hals Date: Fri, 19 Jun 2020 20:40:34 +0200 Subject: [PATCH 063/173] Fix Docker run instructions --- apps/remix-ide/README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/apps/remix-ide/README.md b/apps/remix-ide/README.md index 4e32391488..2c472b1732 100644 --- a/apps/remix-ide/README.md +++ b/apps/remix-ide/README.md @@ -57,13 +57,13 @@ If you want to run latest changes that are merged into master branch then run: ``` docker pull remixproject/remix-ide:latest -docker run -p 8080:80 remixproject-remix-ide:latest +docker run -p 8080:80 remixproject/remix-ide:latest ``` If you want to run latest remix-live release run. ``` docker pull remixproject/remix-ide:remix_live -docker run -p 8080:80 remixproject-remix-ide:remix_live +docker run -p 8080:80 remixproject/remix-ide:remix_live ``` ### Run with docker-compose: From 8e6f1e53ad6b312c86d3bc5ef5190f74f8bc7357 Mon Sep 17 00:00:00 2001 From: yann300 Date: Wed, 17 Jun 2020 16:45:47 +0200 Subject: [PATCH 064/173] remove uneeded compilerUrl param --- apps/remix-ide/src/app/tabs/compile-tab.js | 4 ++-- test-browser/tests/compiler_api.test.js | 3 +-- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/apps/remix-ide/src/app/tabs/compile-tab.js b/apps/remix-ide/src/app/tabs/compile-tab.js index b41dec9bbf..9a61b6d0c1 100644 --- a/apps/remix-ide/src/app/tabs/compile-tab.js +++ b/apps/remix-ide/src/app/tabs/compile-tab.js @@ -218,7 +218,7 @@ class CompileTab extends ViewPlugin { * The module UI will *not* be updated, the compilation result is returned * This function is used by remix-plugin compiler API. * @param {object} map of source files. - * @param {object} settings {evmVersion, optimize, compilerUrl, version, language} + * @param {object} settings {evmVersion, optimize, version, language} */ async compileWithParameters (compilationTargets, settings) { return await compile(compilationTargets, settings) @@ -241,7 +241,7 @@ class CompileTab extends ViewPlugin { /** * set the compiler configuration * This function is used by remix-plugin compiler API. - * @param {object} settings {evmVersion, optimize, compilerUrl, version, language} + * @param {object} settings {evmVersion, optimize, version, language} */ setCompilerConfig (settings) { return new Promise((resolve, reject) => { diff --git a/test-browser/tests/compiler_api.test.js b/test-browser/tests/compiler_api.test.js index 2c267b8177..d38f7b64b7 100644 --- a/test-browser/tests/compiler_api.test.js +++ b/test-browser/tests/compiler_api.test.js @@ -76,8 +76,7 @@ const jsCompile = `(async () => { optimize: false, evmVersion: null, language: 'Solidity', - version: '0.6.8+commit.0bbfe453', - compilerUrl: 'https://solc-bin.ethereum.org/bin/soljson-v0.6.8+commit.0bbfe453.js' + version: '0.6.8+commit.0bbfe453' } const result = await remix.call('solidity', 'compileWithParameters', contract, params) console.log('result ', result) From 2baea632ef31d183d437002c5fd8d8b961f544c5 Mon Sep 17 00:00:00 2001 From: LianaHus Date: Mon, 15 Jun 2020 12:54:22 +0200 Subject: [PATCH 065/173] terminal title bar UI adjustments according to last design --- .../src/app/panels/styles/terminal-styles.js | 34 +++---------------- apps/remix-ide/src/app/panels/terminal.js | 13 +++---- 2 files changed, 11 insertions(+), 36 deletions(-) diff --git a/apps/remix-ide/src/app/panels/styles/terminal-styles.js b/apps/remix-ide/src/app/panels/styles/terminal-styles.js index 1aed65852b..25d2d598ac 100644 --- a/apps/remix-ide/src/app/panels/styles/terminal-styles.js +++ b/apps/remix-ide/src/app/panels/styles/terminal-styles.js @@ -20,20 +20,7 @@ var css = csjs` max-height : 35px; min-height : 35px; } - .clear { - margin-right : 20px; - width : 10px; - cursor : pointer; - display : flex; - } - .clear:hover { - color : var(--secondary); - } .toggleTerminal { - margin-right : 20px; - margin-left : 2px; - font-size : 14px; - font-weight : bold; cursor : pointer; } .toggleTerminal:hover { @@ -95,9 +82,7 @@ var css = csjs` padding-bottom : 1px; } .filter { - padding-right : 0px; - margin-right : 0px; - height : 100%; + height : 80%; white-space : nowrap; overflow : hidden; text-overflow : ellipsis; @@ -126,20 +111,6 @@ var css = csjs` border-left : 1px solid var(--secondary) height : 65%; } - .listenOnNetworkLabel { - white-space : nowrap; - } - .pendingTx { - border-radius : 50%; - margin-right : 30px; - min-width : 13px; - height : 13px; - display : flex; - justify-content : center; - align-items : center; - font-size : 14px; - user-select : none; - } .dragbarHorizontal { position : absolute; top : 0; @@ -149,6 +120,9 @@ var css = csjs` cursor : ns-resize; z-index : 999; } + .listenOnNetwork { + min-height: auto; + } .ghostbar { position : absolute; height : 6px; diff --git a/apps/remix-ide/src/app/panels/terminal.js b/apps/remix-ide/src/app/panels/terminal.js index bf48f6677f..bcdf5f9f1b 100644 --- a/apps/remix-ide/src/app/panels/terminal.js +++ b/apps/remix-ide/src/app/panels/terminal.js @@ -147,11 +147,11 @@ class Terminal extends Plugin { self._view.icon = yo` ` + class="mx-2 ${css.toggleTerminal} fas fa-angle-double-down" data-id="terminalToggleIcon">` self._view.dragbar = yo`
` - self._view.pendingTxCount = yo`
0
` + self._view.pendingTxCount = yo`
0
` self._view.inputSearch = yo` ${self._view.icon} -
+
${self._view.pendingTxCount}
-
+