blob: 8438b35ce46488f1980c3b97dec5aa62608f8e0f [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="hostStatusIcon" />
<span class="d-none d-md-inline"> {{ hostStatus }}</span>
</dd>
</dl>
</b-col>
<b-col class="d-flex justify-content-end">
<b-button
v-if="isConnected"
variant="link"
type="button"
class="pr-0 button-launch"
@click="sendCtrlAltDel"
>
<icon-arrow-down aria-hidden="true" />
{{ $t('pageKvm.buttonCtrlAltDelete') }}
</b-button>
<b-button
v-if="!isFullWindow"
variant="link"
type="button"
class="pr-0 button-launch"
@click="openConsoleWindow()"
>
<icon-launch aria-hidden="true" />
<span class="d-none d-md-inline">
{{ $t('pageKvm.openNewTab') }}
</span>
</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';
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 {
rfb: null,
isConnected: false,
terminalClass: this.isFullWindow ? 'full-window' : '',
marginClass: this.isFullWindow ? 'margin-left-full-window' : '',
status: Connecting,
convasRef: null
};
},
computed: {
hostStatusIcon() {
if (this.status === Connected) {
return 'success';
} else if (this.status === Disconnected) {
return 'danger';
}
return 'secondary';
},
hostStatus() {
if (this.status === Connected) {
return this.$t('pageKvm.connected');
} else if (this.status === Disconnected) {
return this.$t('pageKvm.disconnected');
}
return this.$t('pageKvm.connecting');
}
},
mounted() {
this.openTerminal();
},
methods: {
sendCtrlAltDel() {
this.rfb.sendCtrlAltDel();
},
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;
window.addEventListener('resize', () => {
setTimeout(that.setWidthToolbar, 0);
});
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() {
window.open(
'#/console/kvm',
'_blank',
'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>