blob: 212514d1f8e8343ff6449d9cab0086c038f9b043 [file] [log] [blame]
<template>
<div :class="marginClass">
<div ref="toolbar" class="kvm-toolbar">
<b-row class="d-flex">
<b-col class="d-flex flex-column justify-content-end" cols="4">
<dl class="mb-2" sm="2" md="2">
<dt class="d-inline font-weight-bold mr-1">
{{ $t('pageKvm.status') }}:
</dt>
<dd class="d-inline">
<status-icon :status="serverStatusIcon" />
<span class="d-none d-md-inline"> {{ serverStatus }}</span>
</dd>
</dl>
</b-col>
<b-col class="d-flex justify-content-end pr-1">
<b-button
v-if="isConnected"
variant="link"
type="button"
@click="sendCtrlAltDel"
>
<icon-arrow-down />
{{ $t('pageKvm.buttonCtrlAltDelete') }}
</b-button>
<b-button
v-if="!isFullWindow"
variant="link"
type="button"
@click="openConsoleWindow()"
>
<icon-launch />
{{ $t('pageKvm.openNewTab') }}
</b-button>
</b-col>
</b-row>
</div>
<div id="terminal-kvm" ref="panel" :class="terminalClass"></div>
</div>
</template>
<script>
import RFB from '@novnc/novnc/core/rfb';
import StatusIcon from '@/components/Global/StatusIcon';
import IconLaunch from '@carbon/icons-vue/es/launch/20';
import IconArrowDown from '@carbon/icons-vue/es/arrow--down/16';
import { throttle } from 'lodash';
import { mapState } from 'vuex';
const Connecting = 0;
const Connected = 1;
const Disconnected = 2;
export default {
name: 'KvmConsole',
components: { StatusIcon, IconLaunch, IconArrowDown },
props: {
isFullWindow: {
type: Boolean,
default: true,
},
},
data() {
return {
isConsoleWindow: null,
rfb: null,
isConnected: false,
terminalClass: this.isFullWindow ? 'full-window' : '',
marginClass: this.isFullWindow ? 'margin-left-full-window' : '',
status: Connecting,
convasRef: null,
resizeKvmWindow: null,
};
},
computed: {
...mapState('authentication', ['consoleWindow']),
serverStatusIcon() {
if (this.status === Connected) {
return 'success';
} else if (this.status === Disconnected) {
return 'danger';
}
return 'secondary';
},
serverStatus() {
if (this.status === Connected) {
return this.$t('pageKvm.connected');
} else if (this.status === Disconnected) {
return this.$t('pageKvm.disconnected');
}
return this.$t('pageKvm.connecting');
},
},
watch: {
consoleWindow() {
if (this.consoleWindow == false) this.isConsoleWindow.close();
},
},
created() {
this.$store.dispatch('global/getSystemInfo');
},
mounted() {
this.openTerminal();
},
beforeDestroy() {
window.removeEventListener('resize', this.resizeKvmWindow);
this.closeTerminal();
},
methods: {
sendCtrlAltDel() {
this.rfb.sendCtrlAltDel();
},
closeTerminal() {
this.rfb.disconnect();
this.rfb = null;
},
openTerminal() {
const token = this.$store.getters['authentication/token'];
this.rfb = new RFB(
this.$refs.panel,
`wss://${window.location.host}/kvm/0`,
{ wsProtocols: [token] }
);
this.rfb.scaleViewport = true;
this.rfb.clipViewport = true;
const that = this;
this.resizeKvmWindow = throttle(() => {
setTimeout(that.setWidthToolbar, 0);
}, 1000);
window.addEventListener('resize', this.resizeKvmWindow);
this.rfb.addEventListener('connect', () => {
that.isConnected = true;
that.status = Connected;
that.setWidthToolbar();
});
this.rfb.addEventListener('disconnect', () => {
this.isConnected = false;
that.status = Disconnected;
});
},
setWidthToolbar() {
if (
this.$refs.panel.children &&
this.$refs.panel.children.length > 0 &&
this.$refs.panel.children[0].children.length > 0
) {
this.$refs.toolbar.style.width =
this.$refs.panel.children[0].children[0].clientWidth - 10 + 'px';
}
},
openConsoleWindow() {
// If isConsoleWindow is not null
// Check the newly opened window is closed or not
if (this.isConsoleWindow) {
// If window is not closed set focus to new window
// If window is closed, do open new window
if (!this.isConsoleWindow.closed) {
this.isConsoleWindow.focus();
return;
} else {
this.openNewWindow();
}
} else {
// If isConsoleWindow is null, open new window
this.openNewWindow();
}
},
openNewWindow() {
this.isConsoleWindow = window.open(
'#/console/kvm',
'kvmConsoleWindow',
'directories=no,titlebar=no,toolbar=no,location=no,status=no,menubar=no,scrollbars=no,resizable=yes,width=700,height=550'
);
},
},
};
</script>
<style scoped lang="scss">
.button-ctrl-alt-delete {
float: right;
}
.kvm-status {
padding-top: $spacer / 2;
padding-left: $spacer / 4;
display: inline-block;
}
.margin-left-full-window {
margin-left: 5px;
}
</style>