diff --git a/ReleaseNotes.md b/ReleaseNotes.md index 668f08e..8777288 100644 --- a/ReleaseNotes.md +++ b/ReleaseNotes.md @@ -1,9 +1,25 @@ -## Version 0.1.5 +## Version 0.1.6 --- Release highlights: -* Added correct icon -* made runner start and stop the same button which just toggles. +* Added extra tasks +* Modified the terminal to use [Xterm.js](https://github.com/xtermjs/xterm.js) --- + +# Extra tasks - Clean + +It is now possible to run dotnet clean against each loaded application. + +# Xterm.js + +The terminal has now been replaced with Node-PTY and Xterm.JS. + +These two technologies are what powers the integrated terminal in many IDE's and editors such as VS-Code. + +This now provides a much richer log output along with performance improvements and the ability to view a terminal in full screen. + +![New terminal]() + +![new terminal full screen]() \ No newline at end of file diff --git a/app/Components/runner/Runner.js b/app/Components/runner/Runner.js index f20cf63..c936442 100644 --- a/app/Components/runner/Runner.js +++ b/app/Components/runner/Runner.js @@ -1,4 +1,12 @@ -const { killDotnetProcessAsync, startDotnetProcess } = require('../../tasks'); +const { killDotnetProcessAsync, startDotnetProcess, startCleanProcess } = require('../../tasks'); +const Terminal = require('xterm').Terminal; +const fit = require('xterm/lib/addons/fit/fit'); +const fullScreen = require('xterm/lib/addons/fullscreen/fullscreen'); +const debounce = require('../../utils/debounce'); + +Terminal.applyAddon(fit); +Terminal.applyAddon(fullScreen); + const WebComponentBase = require('../WebComponentBase'); module.exports = class RunnerElement extends WebComponentBase { @@ -14,6 +22,8 @@ module.exports = class RunnerElement extends WebComponentBase { this._name = ''; + this._terminalProcess; + this.setState(RunnerElement.states.stopped); } @@ -34,10 +44,48 @@ module.exports = class RunnerElement extends WebComponentBase { this.setState(this.state); const clearLog = shadow.querySelector('.clear-log'); + const clean = shadow.querySelector('.clean'); + const full = shadow.querySelector('.full'); + const terminal = shadow.querySelector('.terminals'); clearLog.addEventListener('click', () => this.clearData()); + clean.addEventListener('click', () => this.clean()); + + terminal.addEventListener('click', (e) => { + if (e.target.classList.contains('full-screen')) { + terminal.classList.remove('full-screen'); + this._terminalProcess.fit(); + } + }); + + full.addEventListener('click', () => { + terminal.classList.add('full-screen'); + + this.resize(); + }); + + this._terminalProcess = new Terminal(); + this._terminalProcess.setOption('disableStdin', true); + this._terminalProcess.setOption('fontFamily', "Consolas, 'Courier New', monospace"); + + this._terminalProcess.open(shadow.querySelector('.terminals')); shadow.querySelector('.action').addEventListener('click', this._onToggle.bind(this)); + + // Terminal wont fit itself on resize. + window.addEventListener('resize', debounce(this.resize.bind(this), { + delay: 100, + executeOnFirstRun: true + })); + + this.resize(); + } + + resize() { + this._terminalProcess.fit(); + + if (this._runningProccess) + this._runningProccess.resize(this._terminalProcess.cols, this._terminalProcess.rows); } _enableAction() { @@ -67,6 +115,20 @@ module.exports = class RunnerElement extends WebComponentBase { } } + _enableClean() { + if (!this.shadowRoot) + return; + + this.shadowRoot.querySelector('.clean').removeAttribute('disabled'); + } + + _disableClean() { + if (!this.shadowRoot) + return; + + this.shadowRoot.querySelector('.clean').setAttribute('disabled', 'disabled'); + } + onStart() { if (this.state === RunnerElement.states.running || this.state === RunnerElement.states.starting) return; @@ -75,43 +137,43 @@ module.exports = class RunnerElement extends WebComponentBase { this.setState(RunnerElement.states.starting); - this._runningProccess = startDotnetProcess(this.cwd, true, this.runCommandArguments); + this._terminalProcess.writeln("Starting..."); - this._runningProccess.on('close', () => { + this._runningProccess = startDotnetProcess(this.cwd, true, this.runCommandArguments, this._terminalProcess.cols, this._terminalProcess.rows); + + this._runningProccess.on('exit', () => { this.setState(RunnerElement.states.stopped); this._runningProccess = undefined; }); - this._runningProccess.stdout.on('data', (d) => this.onData(d.toString())); - this._runningProccess.stderr.on('data', (d) => this.onData(d.toString(), true)); - } + this._runningProccess.once('data', () => this.setState(RunnerElement.states.running)); - onData(d, errorData) { - if (this.state === RunnerElement.states.starting) - this.setState(RunnerElement.states.running); + this._runningProccess.on('data', (d) => { + this._terminalProcess.write(d); + }); + } - const el = document.createElement('span'); - const terminal = this.shadowRoot.querySelector('.terminal'); + clean() { + if (this.state === RunnerElement.states.running || this.state === RunnerElement.states.starting) + return; - el.classList.add('log-item'); + this._runningProccess = startCleanProcess(this.cwd); - if (errorData) - el.classList.add('error'); - - el.textContent = d; + this._runningProccess.on('exit', () => { + this.setState(RunnerElement.states.stopped); - terminal.appendChild(el); + this._runningProccess = undefined; + }); - terminal.scrollTop = terminal.scrollHeight; + this._runningProccess.on('data', (d) => { + this.setState(RunnerElement.states.running); + this._terminalProcess.write(d); + }); } clearData() { - const terminal = this.shadowRoot.querySelector('.terminal'); - - while(terminal.firstChild) { - terminal.removeChild(terminal.firstChild); - } + this._terminalProcess.clear(); } onTerminate() { @@ -136,6 +198,7 @@ module.exports = class RunnerElement extends WebComponentBase { setState(state) { this._disableAction(); + this._disableClean(); this.state = state; @@ -166,6 +229,7 @@ module.exports = class RunnerElement extends WebComponentBase { stateEl.textContent = 'Stopped'; stateEl.className = 'state badge badge-secondary'; this._enableAction(); + this._enableClean(); actionBtn.textContent = 'Start'; break; } diff --git a/app/Components/runner/runner.css b/app/Components/runner/runner.css index 6966643..41232d7 100644 --- a/app/Components/runner/runner.css +++ b/app/Components/runner/runner.css @@ -9,19 +9,42 @@ span.badge { padding: 5px; } -.terminal { - background-color: rgb(0, 36, 81); - color: #cccccc; - white-space: pre-line; +:host(:hover) .full-screen-toggle { + opacity: 1; +} + +.terminals { border: 1px solid rgba(128, 128, 128, 0.35); - border-radius: 4px; - padding: 5px; - word-break: break-word; - max-height: 200px; - overflow: auto; - flex: 1; } .log-item.error { color: red; -} \ No newline at end of file +} + +.terminals.full-screen { + position: absolute; + top: 0px; + bottom: 0px; + left: 0px; + right: 0px; + z-index: 200; +} + +.terminals.full-screen::before { + content: 'Close'; + display: block; + position: absolute; + top: 0px; + z-index: 5; + right: 20px; + color: white; + cursor: pointer; + border-radius: 3px; + border: 1px solid white; + padding: 5px; + margin: 3px; +} + +.terminals { + position: relative; +} diff --git a/app/Components/runner/runner.html b/app/Components/runner/runner.html index f9d42e9..0ec4f21 100644 --- a/app/Components/runner/runner.html +++ b/app/Components/runner/runner.html @@ -1,10 +1,22 @@ + + + +

+ +
@@ -12,4 +24,5 @@

-
\ No newline at end of file +
+
\ No newline at end of file diff --git a/app/DotnetRunnerApp.js b/app/DotnetRunnerApp.js index e9d6401..a4e09f3 100644 --- a/app/DotnetRunnerApp.js +++ b/app/DotnetRunnerApp.js @@ -1,4 +1,4 @@ -const { BrowserWindow, Menu } = require('electron'); +const { BrowserWindow, Menu, ipcMain } = require('electron'); const { getApplications } = require('./data/applicationStore'); const { shell } = require('electron'); const ipcMessages = require('./ipcMessages'); @@ -82,6 +82,7 @@ module.exports = class DotnetRunnerApp { { label: 'Options', submenu: [{ + id: 'config-apps', label: 'Configure Applicaions', click: this._preferencesOnCLick.bind(this) }, { @@ -89,20 +90,25 @@ module.exports = class DotnetRunnerApp { click: this._displayPreferences.bind(this) }] }, { + id: 'tasks', label: 'Tasks', submenu: [{ + id: 'start-all', label: 'Start all', accelerator: 'Ctrl+s', click: this._startAllApps.bind(this) }, { + id: 'stop-all', label: 'Stop all', accelerator: 'Ctrl+Shift+S', click: this._stopAllApps.bind(this) }, { + id: 'clear-all', label: 'Clear all', accelerator: 'Ctrl+Shift+C', click: this._clearAllApps.bind(this) }, { + id: 'purge-all', label: 'Purge', accelerator: 'Ctrl+Shift+P', click: this._purge.bind(this) @@ -142,15 +148,24 @@ module.exports = class DotnetRunnerApp { } _startAllApps() { + this._disableTasks(); + this._disableConfigAppMenu(); this._sendMessage(ipcMessages.startAllApplications); + ipcMain.once(ipcMessages.taskComplete, this._onTaskComplete.bind(this)); } _stopAllApps() { + this._disableTasks(); + this._disableConfigAppMenu(); this._sendMessage(ipcMessages.stopAllApplications); + ipcMain.once(ipcMessages.taskComplete, this._onTaskComplete.bind(this)); } _clearAllApps() { + this._disableTasks(); + this._disableConfigAppMenu(); this._sendMessage(ipcMessages.clearAllApplicationLogs); + ipcMain.once(ipcMessages.taskComplete, this._onTaskComplete.bind(this)); } _purge() { @@ -161,6 +176,43 @@ module.exports = class DotnetRunnerApp { this._mainWindow.send(message); } + _disableTasks() { + const menu = this._menu.getApplicationMenu().getMenuItemById('tasks'); + + if (!menu || !menu.submenu || !menu.submenu.items) + return; + + menu.submenu.items.forEach(x => x.enabled = false); + } + + _enableTasks() { + const menu = this._menu.getApplicationMenu().getMenuItemById('tasks'); + + if (!menu || !menu.submenu || !menu.submenu.items) + return; + + menu.submenu.items.forEach(x => x.enabled = true); + } + + _disableConfigAppMenu() { + const menu = this._menu.getApplicationMenu().getMenuItemById('config-apps') || {}; + + menu.enabled = false; + } + + _enableConfigAppMenu() { + const menu = this._menu.getApplicationMenu().getMenuItemById('config-apps') || {}; + + menu.enabled = true; + } + + _onTaskComplete(e, task) { + if (task === ipcMessages.startAllApplications || task === ipcMessages.stopAllApplications || task === ipcMessages.clearAllApplicationLogs) + this._enableConfigAppMenu(); + + this._enableTasks(); + } + /** * @returns {Menu} */ diff --git a/app/SplashScreenApp.js b/app/SplashScreenApp.js index 5fa08bf..008c907 100644 --- a/app/SplashScreenApp.js +++ b/app/SplashScreenApp.js @@ -72,7 +72,6 @@ module.exports = class SplashScreenApp { show: false }); - this._splashWindow .loadFile('app/browserWindows/splashScreen/splashScreen.html'); diff --git a/app/browserWindows/main/main.css b/app/browserWindows/main/main.css index 54261cb..b975e92 100644 --- a/app/browserWindows/main/main.css +++ b/app/browserWindows/main/main.css @@ -14,7 +14,11 @@ .footer { padding: 5px; - border-top: solid 1px rgba(128, 128, 128, 0.35);; + border-top: solid 1px rgba(128, 128, 128, 0.35); +} + +.footer-content { + min-height: 36px; } html, body { @@ -28,4 +32,15 @@ body { .runningProcessCount, #github { cursor: pointer; +} + +.batch-run-message.hide { + display: none; +} + +div.batch-run-message { + padding: 5px; + width: 500px; + text-align: center; + margin: 0px; } \ No newline at end of file diff --git a/app/browserWindows/main/main.html b/app/browserWindows/main/main.html index dd87f50..0c3589a 100644 --- a/app/browserWindows/main/main.html +++ b/app/browserWindows/main/main.html @@ -29,9 +29,13 @@