mirror of https://github.com/nocodb/nocodb
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
377 lines
9.9 KiB
377 lines
9.9 KiB
4 years ago
|
<template>
|
||
|
<div>
|
||
|
<v-progress-linear
|
||
|
v-show="loading"
|
||
|
color="success"
|
||
|
indeterminate
|
||
|
absolute
|
||
|
height="4"
|
||
4 years ago
|
/>
|
||
4 years ago
|
|
||
|
<v-tabs
|
||
|
v-model="tab"
|
||
4 years ago
|
style="height: 100%"
|
||
|
height="40"
|
||
|
>
|
||
4 years ago
|
<v-tab
|
||
|
v-for="(n,i) in terminals"
|
||
4 years ago
|
:key="n"
|
||
|
class="caption"
|
||
|
>
|
||
4 years ago
|
<span @dblclick="dblClick"> Local ({{ i + 1 }})</span>
|
||
4 years ago
|
<v-icon
|
||
|
v-if="terminals.length > 1"
|
||
|
class="ml-2 grey--text text--lighten-1"
|
||
|
small
|
||
|
@click.native.stop="closeTerminal(i)"
|
||
|
>
|
||
|
mdi-close
|
||
4 years ago
|
</v-icon>
|
||
|
</v-tab>
|
||
|
|
||
4 years ago
|
<x-icon class="mx-2" small @click="addTerminal">
|
||
|
mdi-plus
|
||
|
</x-icon>
|
||
|
<span v-shortkey="['meta','t']" @shortkey="addTerminal" />
|
||
|
<span v-if="isModal" v-shortkey="['meta','w']" @shortkey="terminals.length && closeTerminal(tab)" />
|
||
|
|
||
|
<div v-if="!isDocker" class="flex-grow-1 d-flex text-right pr-4 justify-end ">
|
||
|
<x-icon
|
||
|
v-if="_isWindows"
|
||
|
icon-class="mr-4"
|
||
|
:disabled="loading"
|
||
|
tooltip="Set-ExecutionPolicy"
|
||
|
color="success success"
|
||
|
@click="setExecutionPolicy()"
|
||
|
>
|
||
4 years ago
|
mdi-consolidate
|
||
|
</x-icon>
|
||
|
|
||
|
<template v-if="!isNoApis">
|
||
4 years ago
|
<x-icon
|
||
|
v-if="isDashboard && !isGraphql"
|
||
|
icon-class="mr-4"
|
||
|
:disabled="loading"
|
||
|
tooltip="Generate REST Code (^ ⇧ M)"
|
||
|
color="success success"
|
||
|
@click="runXcGenApisRest()"
|
||
|
>
|
||
|
{{ isTsEnabled ? 'mdi-language-typescript' : 'mdi-language-javascript' }}
|
||
4 years ago
|
</x-icon>
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','m']" @shortkey="runXcGenApisRest()" />
|
||
|
|
||
|
<x-icon
|
||
|
v-if="isDashboard && isGraphql"
|
||
|
icon-class="mr-4"
|
||
|
:disabled="loading"
|
||
|
color="pink pink"
|
||
|
tooltip="Generate GraphQL Code (^ ⇧ Q)"
|
||
|
@click="runXcGenApisGql()"
|
||
|
>
|
||
|
{{ isTsEnabled ? 'mdi-language-typescript' : 'mdi-language-javascript' }}
|
||
4 years ago
|
</x-icon>
|
||
|
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','q']" @shortkey="runXcGenApisGql()" />
|
||
4 years ago
|
|
||
4 years ago
|
<x-icon
|
||
|
v-if="isDashboard && showDelete"
|
||
|
icon-class="mr-4"
|
||
|
:disabled="loading"
|
||
|
color="error error"
|
||
|
tooltip="Delete all code (^ ⇧ C)"
|
||
|
@click="codeClear()"
|
||
|
>
|
||
4 years ago
|
mdi-delete-variant
|
||
|
</x-icon>
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','c']" @shortkey="codeClear()" />
|
||
|
|
||
|
<x-icon
|
||
|
v-if="isDashboard"
|
||
|
icon-class="mr-4"
|
||
|
:disabled="loading"
|
||
|
size="30"
|
||
|
color="error error"
|
||
|
tooltip="npm install (^ ⇧ I)"
|
||
|
@click="runNpmInstall()"
|
||
|
>
|
||
4 years ago
|
mdi-npm
|
||
|
</x-icon>
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','i']" @shortkey="runNpmInstall()" />
|
||
|
|
||
|
<x-icon
|
||
|
color="success success"
|
||
|
:disabled="loading"
|
||
|
tooltip="Run Command (^ ⇧ R)"
|
||
|
icon-class="mr-2"
|
||
|
size="29"
|
||
|
@click="runScript"
|
||
|
>
|
||
4 years ago
|
mdi-play
|
||
|
</x-icon>
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','r']" @shortkey="runScript()" />
|
||
|
|
||
|
<x-icon
|
||
|
color="red red"
|
||
|
:disabled="loading"
|
||
|
tooltip="Stop Command (^ ⇧ S)"
|
||
|
icon-class="mr-2"
|
||
|
@click="stopScript"
|
||
|
>
|
||
4 years ago
|
mdi-stop
|
||
|
</x-icon>
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','s']" @shortkey="stopScript()" />
|
||
|
|
||
|
<x-icon
|
||
|
color="info info"
|
||
|
:disabled="loading"
|
||
|
tooltip="Clear (^ ⇧ E)"
|
||
|
icon-class="mr-2"
|
||
|
@click="clearScript"
|
||
|
>
|
||
4 years ago
|
mdi-notification-clear-all
|
||
|
</x-icon>
|
||
4 years ago
|
<span v-if="isDashboard" v-shortkey="['ctrl','shift','e']" @shortkey="stopScript()" />
|
||
4 years ago
|
<v-tooltip bottom>
|
||
4 years ago
|
<template #activator="{on}">
|
||
|
<img height="24" class="mt-2 mr-2 pointer" src="~/assets/img/brand/favicon-64.png" v-on="on" @click="installCliTool">
|
||
4 years ago
|
</template>
|
||
|
Installs NocoDB CLI<br>( npm install -g xc-cli )
|
||
|
</v-tooltip>
|
||
|
|
||
|
<div class="flex-shrink-1 ">
|
||
|
<v-select
|
||
|
v-model="selectedScript"
|
||
|
height="32"
|
||
|
class="caption envs"
|
||
|
dense
|
||
|
:items="scripts"
|
||
|
placeholder="package.json"
|
||
|
single-line
|
||
|
outlined
|
||
|
>
|
||
4 years ago
|
<template #selection="{item}">
|
||
|
<span
|
||
|
style="text-transform: uppercase"
|
||
|
>{{ item }}</span> <span class="grey--text">(env)</span>
|
||
4 years ago
|
</template>
|
||
|
|
||
|
<!-- <template v-slot:prepend-outer>-->
|
||
|
<!-- -->
|
||
|
<!-- </template>-->
|
||
|
</v-select>
|
||
|
</div>
|
||
|
</template>
|
||
|
</div>
|
||
|
|
||
|
<v-tab-item
|
||
|
v-for="i in terminals"
|
||
|
:key="i"
|
||
|
>
|
||
4 years ago
|
<div :key="i" ref="term" style="height:100%" class="terminal-window" />
|
||
4 years ago
|
</v-tab-item>
|
||
|
</v-tabs>
|
||
|
</div>
|
||
|
</template>
|
||
|
|
||
|
<script>
|
||
4 years ago
|
import { Terminal } from 'xterm'
|
||
|
import { FitAddon } from 'xterm-addon-fit'
|
||
|
import { WebLinksAddon } from 'xterm-addon-web-links'
|
||
|
import { mapGetters } from 'vuex'
|
||
3 years ago
|
import XIcon from './global/XIcon'
|
||
4 years ago
|
|
||
|
export default {
|
||
|
name: 'XTerm',
|
||
|
components: { XIcon },
|
||
|
props: {
|
||
|
isModal: {
|
||
|
default: false,
|
||
|
type: Boolean
|
||
|
}
|
||
|
},
|
||
|
computed: {
|
||
|
...mapGetters({
|
||
|
currentProjectFolder: 'project/currentProjectFolder',
|
||
|
sqlMgr: 'sqlMgr/sqlMgr',
|
||
|
phql: 'project/GtrProjectIsGraphql',
|
||
|
isNoApis: 'project/GtrProjectIsNoApis',
|
||
|
isDocker: 'project/GtrIsDocker'
|
||
|
}),
|
||
4 years ago
|
isTsEnabled() {
|
||
4 years ago
|
return false // return process.env.TS_ENABLED;
|
||
|
}
|
||
|
},
|
||
4 years ago
|
data() {
|
||
4 years ago
|
return {
|
||
|
tab: 0,
|
||
|
terminals: [Date.now()],
|
||
|
termRef: [],
|
||
|
scripts: [],
|
||
|
selectedScript: null,
|
||
|
loading: false,
|
||
|
showDelete: false
|
||
|
}
|
||
|
},
|
||
4 years ago
|
async mounted() {
|
||
4 years ago
|
// try {
|
||
|
// const packageData = await jsonfile.readFileSync(path.join(this.currentProjectFolder, 'package.json'));
|
||
|
// this.scripts = Object.keys(packageData.scripts);
|
||
|
// } catch (e) {
|
||
|
// console.log('package============', e)
|
||
|
// }
|
||
|
|
||
|
setTimeout(() => {
|
||
|
this.initTerminal(this.$refs.term[0], 0)
|
||
|
}, 200)
|
||
|
},
|
||
4 years ago
|
destroyed() {
|
||
4 years ago
|
let l = this.terminals.length
|
||
|
while (l--) { this.closeTerminal(l) }
|
||
|
},
|
||
|
methods: {
|
||
4 years ago
|
dblClick() {
|
||
4 years ago
|
},
|
||
4 years ago
|
async setExecutionPolicy() {
|
||
4 years ago
|
},
|
||
4 years ago
|
async handleKeyDown(event) {
|
||
4 years ago
|
},
|
||
|
|
||
4 years ago
|
async codeGenerateMvc() {
|
||
4 years ago
|
this.loading = true
|
||
|
},
|
||
4 years ago
|
async codeGenerateMvcGql() {
|
||
4 years ago
|
this.loading = true
|
||
|
},
|
||
4 years ago
|
async codeClear() {
|
||
4 years ago
|
this.loading = true
|
||
|
},
|
||
4 years ago
|
|
||
4 years ago
|
runXcGenApisRest() {
|
||
4 years ago
|
},
|
||
4 years ago
|
|
||
4 years ago
|
runXcGenApisGql() {
|
||
4 years ago
|
},
|
||
4 years ago
|
|
||
4 years ago
|
runNpmInstall() {
|
||
4 years ago
|
const { client } = this.termRef[this.tab]
|
||
|
client.emit('req', '\n\rnpm install;\n\r')
|
||
|
},
|
||
4 years ago
|
|
||
4 years ago
|
runScript() {
|
||
4 years ago
|
},
|
||
4 years ago
|
stopScript() {
|
||
4 years ago
|
},
|
||
4 years ago
|
clearScript() {
|
||
4 years ago
|
},
|
||
4 years ago
|
installCliTool() {
|
||
4 years ago
|
},
|
||
4 years ago
|
addTerminal() {
|
||
4 years ago
|
if (this.terminals.length > 4) {
|
||
|
this.$toast.info('Only 5 terminals can be opened').goAway()
|
||
|
return
|
||
|
}
|
||
|
this.terminals.push(Date.now())
|
||
|
this.tab = this.terminals.length - 1
|
||
|
this.$nextTick(() => {
|
||
|
this.initTerminal(this.$refs.term[this.terminals.length - 1], this.terminals.length - 1)
|
||
|
})
|
||
|
},
|
||
4 years ago
|
|
||
4 years ago
|
initTerminal($el, index) {
|
||
4 years ago
|
try {
|
||
|
// todo: change to hostname
|
||
|
const client = require('socket.io-client')('http://localhost:8081')
|
||
4 years ago
|
|
||
4 years ago
|
const term = new Terminal({
|
||
|
theme: {
|
||
|
background: 'black',
|
||
|
foreground: 'green'
|
||
|
}
|
||
|
})
|
||
4 years ago
|
|
||
4 years ago
|
term.loadAddon(new WebLinksAddon((e, url) => {
|
||
|
e.preventDefault()
|
||
|
}))
|
||
|
// //
|
||
|
// //
|
||
|
const fitAddon = new FitAddon()
|
||
|
term.loadAddon(fitAddon)
|
||
|
// //
|
||
|
term.open($el)
|
||
|
// //
|
||
|
fitAddon.fit()
|
||
|
term.onData((data) => {
|
||
|
client.emit('req', data)
|
||
|
})
|
||
4 years ago
|
|
||
4 years ago
|
client.on('res', (data) => {
|
||
|
term.write(`${data}`)
|
||
|
})
|
||
|
term.focus()
|
||
|
//
|
||
|
this.termRef[index] = {
|
||
|
term,
|
||
|
client
|
||
4 years ago
|
}
|
||
4 years ago
|
} catch (e) {
|
||
|
this.$toast.error('Error opening the terminal\n\nPlease use your system terminal').goAway(5000)
|
||
4 years ago
|
}
|
||
|
},
|
||
4 years ago
|
closeTerminal(index) {
|
||
4 years ago
|
try {
|
||
|
const proc = this.termRef.splice(index, 1)[0]
|
||
|
proc.term.dispose()
|
||
|
proc.client.off()
|
||
|
proc.client.disconnect()
|
||
|
this.terminals.splice(index, 1)
|
||
|
} catch (e) {
|
||
|
console.log(e)
|
||
|
throw e
|
||
|
}
|
||
4 years ago
|
}
|
||
|
}
|
||
4 years ago
|
}
|
||
4 years ago
|
</script>
|
||
|
|
||
|
<style scoped>
|
||
|
/deep/ .v-tabs-items {
|
||
|
height: calc(100% - 40px);
|
||
|
}
|
||
|
|
||
|
/deep/ .v-window__container {
|
||
|
height: 100%;
|
||
|
}
|
||
|
|
||
|
/deep/ .v-window__container .v-window-item {
|
||
|
height: 100%;
|
||
|
}
|
||
|
|
||
|
/deep/ .terminal-window > div {
|
||
|
padding: 0 5px;
|
||
|
}
|
||
|
</style>
|
||
|
<!--
|
||
|
/**
|
||
|
* @copyright Copyright (c) 2021, Xgene Cloud Ltd
|
||
|
*
|
||
|
* @author Naveen MR <oof1lab@gmail.com>
|
||
|
* @author Pranav C Balan <pranavxc@gmail.com>
|
||
|
*
|
||
|
* @license GNU AGPL version 3 or any later version
|
||
|
*
|
||
|
* This program is free software: you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU Affero General Public License as
|
||
|
* published by the Free Software Foundation, either version 3 of the
|
||
|
* License, or (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU Affero General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Affero General Public License
|
||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*
|
||
|
*/
|
||
|
-->
|