🔨 执行日志.
This commit is contained in:
@@ -93,5 +93,5 @@ export function getExecLogTailToken(request: ExecTailRequest) {
|
|||||||
* 下载执行日志文件
|
* 下载执行日志文件
|
||||||
*/
|
*/
|
||||||
export function downloadExecLogFile(id: number) {
|
export function downloadExecLogFile(id: number) {
|
||||||
return axios.get<Blob>('/asset/exec/download-log', { unwrap: true, params: { id } });
|
return axios.get('/asset/exec/download-log', { unwrap: true, params: { id } });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -332,6 +332,7 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding-left: 20px;
|
padding-left: 20px;
|
||||||
|
color: var(--color-text-1);
|
||||||
|
|
||||||
&-logo {
|
&-logo {
|
||||||
width: 32px;
|
width: 32px;
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ export interface LogDomRef {
|
|||||||
export interface LogAppenderConf {
|
export interface LogAppenderConf {
|
||||||
id: number;
|
id: number;
|
||||||
el: HTMLElement;
|
el: HTMLElement;
|
||||||
|
fixed: boolean;
|
||||||
terminal: Terminal;
|
terminal: Terminal;
|
||||||
addons: LogAddons;
|
addons: LogAddons;
|
||||||
}
|
}
|
||||||
@@ -49,8 +50,38 @@ export interface ILogAppender {
|
|||||||
// 初始化
|
// 初始化
|
||||||
init(refs: Array<LogDomRef>): Promise<void>;
|
init(refs: Array<LogDomRef>): Promise<void>;
|
||||||
|
|
||||||
// 自适应
|
// 设置当前元素
|
||||||
fit(): void;
|
setCurrent(id: number): void;
|
||||||
|
|
||||||
|
// 查找关键字
|
||||||
|
find(word: string, next: boolean, options: any): void;
|
||||||
|
|
||||||
|
// 聚焦
|
||||||
|
focus(): void;
|
||||||
|
|
||||||
|
// 设置固定
|
||||||
|
setFixed(fixed: boolean): void;
|
||||||
|
|
||||||
|
// 去顶部
|
||||||
|
toTop(): void;
|
||||||
|
|
||||||
|
// 去底部
|
||||||
|
toBottom(): void;
|
||||||
|
|
||||||
|
// 添加字体大小
|
||||||
|
addFontSize(addSize: number): void;
|
||||||
|
|
||||||
|
// 复制
|
||||||
|
copy(): void;
|
||||||
|
|
||||||
|
// 复制全部
|
||||||
|
copyAll(): void;
|
||||||
|
|
||||||
|
// 选中全部
|
||||||
|
selectAll(): void;
|
||||||
|
|
||||||
|
// 清空
|
||||||
|
clear(): void;
|
||||||
|
|
||||||
// 关闭 client
|
// 关闭 client
|
||||||
closeClient(): void;
|
closeClient(): void;
|
||||||
|
|||||||
@@ -1,38 +1,38 @@
|
|||||||
import type { ExecTailRequest } from '@/api/exec/exec';
|
|
||||||
import { getExecLogTailToken } from '@/api/exec/exec';
|
|
||||||
import type { ILogAppender, LogAddons, LogAppenderConf, LogDomRef } from './appender.const';
|
import type { ILogAppender, LogAddons, LogAppenderConf, LogDomRef } from './appender.const';
|
||||||
|
import type { ExecTailRequest } from '@/api/exec/exec';
|
||||||
import { AppenderOptions } from './appender.const';
|
import { AppenderOptions } from './appender.const';
|
||||||
|
import { getExecLogTailToken } from '@/api/exec/exec';
|
||||||
import { webSocketBaseUrl } from '@/utils/env';
|
import { webSocketBaseUrl } from '@/utils/env';
|
||||||
import { Message } from '@arco-design/web-vue';
|
import { Message } from '@arco-design/web-vue';
|
||||||
import { createWebSocket } from '@/utils';
|
import { createWebSocket } from '@/utils';
|
||||||
|
import { useDebounceFn } from '@vueuse/core';
|
||||||
|
import { addEventListen, removeEventListen } from '@/utils/event';
|
||||||
|
import { copy as copyText } from '@/hooks/copy';
|
||||||
import { Terminal } from 'xterm';
|
import { Terminal } from 'xterm';
|
||||||
import { FitAddon } from 'xterm-addon-fit';
|
import { FitAddon } from 'xterm-addon-fit';
|
||||||
import { SearchAddon } from 'xterm-addon-search';
|
import { SearchAddon } from 'xterm-addon-search';
|
||||||
import { CanvasAddon } from 'xterm-addon-canvas';
|
import { CanvasAddon } from 'xterm-addon-canvas';
|
||||||
import { useDebounceFn } from '@vueuse/core';
|
|
||||||
import { addEventListen, removeEventListen } from '@/utils/event';
|
|
||||||
|
|
||||||
// todo SEARCH addon setfixed
|
|
||||||
// todo font-size totop copy tobottom selectall clear
|
|
||||||
// todo 批量执行的 warn
|
|
||||||
|
|
||||||
// 执行日志 appender 实现
|
// 执行日志 appender 实现
|
||||||
export default class LogAppender implements ILogAppender {
|
export default class LogAppender implements ILogAppender {
|
||||||
|
|
||||||
private config: ExecTailRequest;
|
private current: LogAppenderConf;
|
||||||
|
|
||||||
private client?: WebSocket;
|
private client?: WebSocket;
|
||||||
|
|
||||||
|
private readonly config: ExecTailRequest;
|
||||||
|
|
||||||
private readonly appenderRel: Record<string, LogAppenderConf>;
|
private readonly appenderRel: Record<string, LogAppenderConf>;
|
||||||
|
|
||||||
private keepAliveTask?: number;
|
private keepAliveTask?: number;
|
||||||
|
|
||||||
private readonly fitFn: () => {};
|
private readonly fitAllFn: () => {};
|
||||||
|
|
||||||
constructor(config: ExecTailRequest) {
|
constructor(config: ExecTailRequest) {
|
||||||
|
this.current = undefined as unknown as LogAppenderConf;
|
||||||
this.config = config;
|
this.config = config;
|
||||||
this.appenderRel = {};
|
this.appenderRel = {};
|
||||||
this.fitFn = useDebounceFn(this.fit).bind(this);
|
this.fitAllFn = useDebounceFn(this.fitAll).bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 初始化
|
// 初始化
|
||||||
@@ -49,6 +49,8 @@ export default class LogAppender implements ILogAppender {
|
|||||||
for (let logDomRef of logDomRefs) {
|
for (let logDomRef of logDomRefs) {
|
||||||
// 初始化 terminal
|
// 初始化 terminal
|
||||||
const terminal = new Terminal(AppenderOptions);
|
const terminal = new Terminal(AppenderOptions);
|
||||||
|
// 初始化快捷键
|
||||||
|
this.initCustomKey(terminal);
|
||||||
// 初始化插件
|
// 初始化插件
|
||||||
const addons = this.initAddons(terminal);
|
const addons = this.initAddons(terminal);
|
||||||
// 打开终端
|
// 打开终端
|
||||||
@@ -57,12 +59,46 @@ export default class LogAppender implements ILogAppender {
|
|||||||
addons.fit.fit();
|
addons.fit.fit();
|
||||||
this.appenderRel[logDomRef.id] = {
|
this.appenderRel[logDomRef.id] = {
|
||||||
...logDomRef,
|
...logDomRef,
|
||||||
|
fixed: false,
|
||||||
terminal,
|
terminal,
|
||||||
addons
|
addons
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
// 设置当前对象
|
||||||
|
this.current = this.appenderRel[logDomRefs[0].id];
|
||||||
// 注册自适应事件
|
// 注册自适应事件
|
||||||
addEventListen(window, 'resize', this.fitFn);
|
addEventListen(window, 'resize', this.fitAllFn);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化快捷键操作
|
||||||
|
initCustomKey(terminal: Terminal) {
|
||||||
|
terminal.attachCustomKeyEventHandler((e: KeyboardEvent) => {
|
||||||
|
if (e.type !== 'keydown') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (e.ctrlKey && e.code === 'KeyC') {
|
||||||
|
// 复制
|
||||||
|
e.preventDefault();
|
||||||
|
this.copy();
|
||||||
|
return false;
|
||||||
|
} else if (e.ctrlKey && e.code === 'KeyL') {
|
||||||
|
// 清空
|
||||||
|
e.preventDefault();
|
||||||
|
this.clear();
|
||||||
|
return false;
|
||||||
|
} else if (e.ctrlKey && e.code === 'KeyA') {
|
||||||
|
// 全选
|
||||||
|
e.preventDefault();
|
||||||
|
this.selectAll();
|
||||||
|
return false;
|
||||||
|
} else if (e.ctrlKey && e.code === 'KeyF') {
|
||||||
|
// 搜索
|
||||||
|
e.preventDefault();
|
||||||
|
// TODO open search
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 初始化插件
|
// 初始化插件
|
||||||
@@ -102,10 +138,92 @@ export default class LogAppender implements ILogAppender {
|
|||||||
}, 15000);
|
}, 15000);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 自适应
|
// 设置当前元素
|
||||||
fit(): void {
|
setCurrent(id: number): void {
|
||||||
|
const rel = this.appenderRel[id];
|
||||||
|
if (!rel) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.current = rel;
|
||||||
|
// 自适应
|
||||||
|
rel.addons.fit.fit();
|
||||||
|
// 非固定跳转到最底部
|
||||||
|
if (!rel.fixed) {
|
||||||
|
rel.terminal.scrollToBottom();
|
||||||
|
}
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查找关键字
|
||||||
|
find(word: string, next: boolean, options: any) {
|
||||||
|
if (next) {
|
||||||
|
this.current.addons.search.findNext(word, options);
|
||||||
|
} else {
|
||||||
|
this.current.addons.search.findPrevious(word, options);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 设置固定
|
||||||
|
setFixed(fixed: boolean): void {
|
||||||
|
this.current.fixed = fixed;
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 去顶部
|
||||||
|
toTop(): void {
|
||||||
|
this.current.terminal.scrollToTop();
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 去底部
|
||||||
|
toBottom(): void {
|
||||||
|
this.current.terminal.scrollToBottom();
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加字体大小
|
||||||
|
addFontSize(addSize: number): void {
|
||||||
|
this.current.terminal.options['fontSize'] = this.current.terminal.options['fontSize'] as number + addSize;
|
||||||
|
this.current.addons.fit.fit();
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 复制
|
||||||
|
copy(): void {
|
||||||
|
copyText(this.current.terminal.getSelection(), '已复制');
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 复制全部
|
||||||
|
copyAll(): void {
|
||||||
|
this.selectAll();
|
||||||
|
this.copy();
|
||||||
|
this.current.terminal.clearSelection();
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 选中全部
|
||||||
|
selectAll(): void {
|
||||||
|
this.current.terminal.selectAll();
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 清空
|
||||||
|
clear(): void {
|
||||||
|
this.current.terminal.clear();
|
||||||
|
this.current.terminal.clearSelection();
|
||||||
|
this.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 聚焦
|
||||||
|
focus(): void {
|
||||||
|
this.current.terminal.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 自适应全部
|
||||||
|
fitAll(): void {
|
||||||
Object.values(this.appenderRel).forEach(s => {
|
Object.values(this.appenderRel).forEach(s => {
|
||||||
s.addons?.fit?.fit();
|
s.addons.fit.fit();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -129,7 +247,7 @@ export default class LogAppender implements ILogAppender {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
// 移除自适应事件
|
// 移除自适应事件
|
||||||
removeEventListen(window, 'resize', this.fitFn);
|
removeEventListen(window, 'resize', this.fitAllFn);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 关闭
|
// 关闭
|
||||||
|
|||||||
@@ -53,7 +53,7 @@
|
|||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
export default {
|
export default {
|
||||||
name: 'sshSearchModal'
|
name: 'xtermSearchModal'
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -130,26 +130,26 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
background: var(--search-bg);
|
background: var(--bg);
|
||||||
transition: background-color .2s;
|
transition: background-color .2s;
|
||||||
|
|
||||||
&:focus-within, &:hover {
|
&:focus-within, &:hover {
|
||||||
background: var(--search-bg-focus);
|
background: var(--bg-focus);
|
||||||
|
|
||||||
.search-input {
|
.search-input {
|
||||||
color: var(--search-color-text-focus);
|
color: var(--color-text-focus);
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon-wrapper {
|
.icon-wrapper {
|
||||||
color: var(--search-color-text-focus);
|
color: var(--color-text-focus);
|
||||||
transition: background-color .2s;
|
transition: background-color .2s;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background: var(--search-bg-icon-hover-focus);
|
background: var(--bg-icon-hover-focus);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
background: var(--search-bg-icon-selected-focus);
|
background: var(--bg-icon-selected-focus);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -163,7 +163,7 @@
|
|||||||
outline: none;
|
outline: none;
|
||||||
height: 18px;
|
height: 18px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
color: var(--search-color-text);
|
color: var(--color-text);
|
||||||
}
|
}
|
||||||
|
|
||||||
.word-option {
|
.word-option {
|
||||||
@@ -183,18 +183,18 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
color: var(--search-color-text);
|
color: var(--color-text);
|
||||||
|
|
||||||
&:not(:first-child) {
|
&:not(:first-child) {
|
||||||
margin-left: 2px;
|
margin-left: 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background: var(--search-bg-icon-hover);
|
background: var(--bg-icon-hover);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
background: var(--search-bg-icon-selected);
|
background: var(--bg-icon-selected);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -87,7 +87,7 @@ export function getParentPath(path: string) {
|
|||||||
/**
|
/**
|
||||||
* 下载文件
|
* 下载文件
|
||||||
*/
|
*/
|
||||||
export function downloadFile(res: any, fileName: string) {
|
export function downloadFile(res: any, fileName: string = '') {
|
||||||
const blob = new Blob([res.data]);
|
const blob = new Blob([res.data]);
|
||||||
const tempLink = document.createElement('a');
|
const tempLink = document.createElement('a');
|
||||||
const blobURL = window.URL.createObjectURL(blob);
|
const blobURL = window.URL.createObjectURL(blob);
|
||||||
|
|||||||
@@ -7,40 +7,159 @@
|
|||||||
<!-- 面板头部 -->
|
<!-- 面板头部 -->
|
||||||
<div class="log-header">
|
<div class="log-header">
|
||||||
<!-- 左侧信息 -->
|
<!-- 左侧信息 -->
|
||||||
<div class="log-header-left">
|
<a-space class="log-header-left" :size="12">
|
||||||
<a-space :size="12">
|
<!-- 状态 -->
|
||||||
<!-- 状态 -->
|
<a-tag :color="getDictValue(execHostStatusKey, host.status, 'color')">
|
||||||
<a-tag :color="getDictValue(execHostStatusKey, host.status, 'color')">
|
{{ getDictValue(execHostStatusKey, host.status) }}
|
||||||
{{ getDictValue(execHostStatusKey, host.status) }}
|
</a-tag>
|
||||||
</a-tag>
|
<!-- exitStatus -->
|
||||||
<!-- exitStatus -->
|
<a-tag v-if="host.exitStatus || host.exitStatus === 0"
|
||||||
<a-tag v-if="host.exitStatus || host.exitStatus === 0"
|
:color="host.exitStatus === 0 ? 'arcoblue' : 'orangered'"
|
||||||
:color="host.exitStatus === 0 ? 'arcoblue' : 'orangered'"
|
title="exit status">
|
||||||
title="exit status">
|
<template #icon>
|
||||||
<template #icon>
|
<icon-check v-if="host.exitStatus === 0" />
|
||||||
<icon-check v-if="host.exitStatus === 0" />
|
<icon-exclamation v-else />
|
||||||
<icon-exclamation v-else />
|
</template>
|
||||||
</template>
|
<span class="tag-value">{{ host.exitStatus }}</span>
|
||||||
<span class="tag-value">{{ host.exitStatus }}</span>
|
</a-tag>
|
||||||
</a-tag>
|
<!-- 持续时间 -->
|
||||||
<!-- 持续时间 -->
|
<a-tag color="arcoblue" title="持续时间">
|
||||||
<a-tag color="arcoblue" title="持续时间">
|
<template #icon>
|
||||||
<template #icon>
|
<icon-loading v-if="host.status === execHostStatus.WAITING || host.status === execHostStatus.RUNNING" />
|
||||||
<icon-loading v-if="host.status === execHostStatus.WAITING || host.status === execHostStatus.RUNNING" />
|
<icon-clock-circle v-else />
|
||||||
<icon-clock-circle v-else />
|
</template>
|
||||||
</template>
|
<span class="tag-value">{{ formatDuration(host.startTime, host.finishTime) || '0s' }}</span>
|
||||||
<span class="tag-value">{{ formatDuration(host.startTime, host.finishTime) || '0s' }}</span>
|
</a-tag>
|
||||||
</a-tag>
|
</a-space>
|
||||||
</a-space>
|
|
||||||
</div>
|
|
||||||
<!-- 右侧操作 -->
|
<!-- 右侧操作 -->
|
||||||
<div class="log-header-right">TODO</div>
|
<a-space class="log-header-right" :size="12">
|
||||||
</div>
|
<!-- 搜索 -->
|
||||||
<!-- 日志面板 -->
|
<span class="log-action click-icon-wrapper"
|
||||||
<div class="log-wrapper">
|
title="搜索"
|
||||||
<div class="log-appender"
|
@click="() => appender?.addFontSize(1)">
|
||||||
:ref="e => addRef(host.id, e) as unknown as VNodeRef" />
|
<icon-find-replace />
|
||||||
|
</span>
|
||||||
|
<!-- 增大字号 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="增大字号"
|
||||||
|
@click="() => appender?.addFontSize(1)">
|
||||||
|
<icon-zoom-in />
|
||||||
|
</span>
|
||||||
|
<!-- 减小字号 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="减小字号"
|
||||||
|
@click="() => appender?.addFontSize(-1)">
|
||||||
|
<icon-zoom-out />
|
||||||
|
</span>
|
||||||
|
<!-- 去顶部 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="去顶部"
|
||||||
|
@click="() => appender?.toTop()">
|
||||||
|
<icon-up />
|
||||||
|
</span>
|
||||||
|
<!-- 去底部 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="去底部"
|
||||||
|
@click="() => appender?.toBottom()">
|
||||||
|
<icon-down />
|
||||||
|
</span>
|
||||||
|
<!-- 全选 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="全选"
|
||||||
|
@click="() => appender?.selectAll()">
|
||||||
|
<icon-expand />
|
||||||
|
</span>
|
||||||
|
<!-- 复制 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="复制"
|
||||||
|
@click="() => appender?.copy()">
|
||||||
|
<icon-copy />
|
||||||
|
</span>
|
||||||
|
<!-- 复制全部 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="复制全部"
|
||||||
|
@click="() => appender?.copyAll()">
|
||||||
|
<icon-brush />
|
||||||
|
</span>
|
||||||
|
<!-- 清空 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="清空"
|
||||||
|
@click="() => appender?.clear()">
|
||||||
|
<icon-delete />
|
||||||
|
</span>
|
||||||
|
<!-- 下载 -->
|
||||||
|
<span class="log-action click-icon-wrapper"
|
||||||
|
title="下载"
|
||||||
|
@click="downloadLogFile(host.id)">
|
||||||
|
<icon-download />
|
||||||
|
</span>
|
||||||
|
<!-- 设置固定 -->
|
||||||
|
<a-switch type="round"
|
||||||
|
checked-text="固定"
|
||||||
|
unchecked-text="跟随"
|
||||||
|
@change="(e: any) => appender?.setFixed(e as boolean)" />
|
||||||
|
</a-space>
|
||||||
</div>
|
</div>
|
||||||
|
<!-- 右键菜单 -->
|
||||||
|
<a-dropdown :popup-max-height="false"
|
||||||
|
trigger="contextMenu"
|
||||||
|
position="bl"
|
||||||
|
alignPoint>
|
||||||
|
<!-- 日志面板 -->
|
||||||
|
<div class="log-wrapper">
|
||||||
|
<!-- terminal -->
|
||||||
|
<div class="log-appender"
|
||||||
|
:ref="e => addRef(host.id, e as HTMLElement) as unknown as VNodeRef" />
|
||||||
|
<!-- 搜索框 -->
|
||||||
|
<xterm-search-modal ref="searchModal"
|
||||||
|
class="search-modal"
|
||||||
|
@find="searchWords"
|
||||||
|
@close="searchClose" />
|
||||||
|
</div>
|
||||||
|
<!-- 右键菜单 -->
|
||||||
|
<template #content>
|
||||||
|
<!-- 去顶部 -->
|
||||||
|
<a-doption style="line-height: 30px; padding: 0 8px;"
|
||||||
|
@click="() => appender?.toTop()">
|
||||||
|
<template #icon>
|
||||||
|
<icon-up />
|
||||||
|
</template>
|
||||||
|
<span>去顶部</span>
|
||||||
|
</a-doption>
|
||||||
|
<!-- 去底部 -->
|
||||||
|
<a-doption style="line-height: 30px; padding: 0 8px;"
|
||||||
|
@click="() => appender?.toBottom()">
|
||||||
|
<template #icon>
|
||||||
|
<icon-down />
|
||||||
|
</template>
|
||||||
|
<span>去底部</span>
|
||||||
|
</a-doption>
|
||||||
|
<!-- 全选 -->
|
||||||
|
<a-doption style="line-height: 30px; padding: 0 8px;"
|
||||||
|
@click="() => appender?.selectAll()">
|
||||||
|
<template #icon>
|
||||||
|
<icon-expand />
|
||||||
|
</template>
|
||||||
|
<span>全选</span>
|
||||||
|
</a-doption>
|
||||||
|
<!-- 复制 -->
|
||||||
|
<a-doption style="line-height: 30px; padding: 0 8px;"
|
||||||
|
@click="() => appender?.copy()">
|
||||||
|
<template #icon>
|
||||||
|
<icon-copy />
|
||||||
|
</template>
|
||||||
|
<span>复制</span>
|
||||||
|
</a-doption>
|
||||||
|
<!-- 清空 -->
|
||||||
|
<a-doption style="line-height: 30px; padding: 0 8px;"
|
||||||
|
@click="() => appender?.clear()">
|
||||||
|
<template #icon>
|
||||||
|
<icon-delete />
|
||||||
|
</template>
|
||||||
|
<span>清空</span>
|
||||||
|
</a-doption>
|
||||||
|
</template>
|
||||||
|
</a-dropdown>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -56,10 +175,13 @@
|
|||||||
import type { ExecCommandResponse } from '@/api/exec/exec';
|
import type { ExecCommandResponse } from '@/api/exec/exec';
|
||||||
import type { LogDomRef, ILogAppender } from '@/components/xtrem/log-appender/appender.const';
|
import type { LogDomRef, ILogAppender } from '@/components/xtrem/log-appender/appender.const';
|
||||||
import { nextTick, ref, watch } from 'vue';
|
import { nextTick, ref, watch } from 'vue';
|
||||||
|
import { downloadExecLogFile } from '@/api/exec/exec';
|
||||||
|
import { downloadFile } from '@/utils/file';
|
||||||
import { formatDuration } from '@/utils';
|
import { formatDuration } from '@/utils';
|
||||||
import { execHostStatus, execHostStatusKey } from '@/views/exec/exec-log/types/const';
|
import { execHostStatus, execHostStatusKey } from '@/views/exec/exec-log/types/const';
|
||||||
import { useDictStore } from '@/store';
|
import { useDictStore } from '@/store';
|
||||||
import LogAppender from '@/components/xtrem/log-appender/log-appender';
|
import LogAppender from '@/components/xtrem/log-appender/log-appender';
|
||||||
|
import XtermSearchModal from '@/components/xtrem/search-modal/index.vue';
|
||||||
import 'xterm/css/xterm.css';
|
import 'xterm/css/xterm.css';
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
@@ -71,11 +193,12 @@
|
|||||||
|
|
||||||
const logRefs = ref<Array<LogDomRef>>([]);
|
const logRefs = ref<Array<LogDomRef>>([]);
|
||||||
const appender = ref<ILogAppender>();
|
const appender = ref<ILogAppender>();
|
||||||
|
const searchModal = ref();
|
||||||
|
|
||||||
// 切换标签自适应
|
// 切换标签
|
||||||
watch(() => props.current, () => {
|
watch(() => props.current, (val) => {
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
appender.value?.fit();
|
appender.value?.setCurrent(val);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -107,10 +230,27 @@
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// 搜索关键字
|
||||||
|
const searchWords = (word: string, next: boolean, options: any) => {
|
||||||
|
appender.value?.find(word, next, options);
|
||||||
|
};
|
||||||
|
|
||||||
|
// 关闭搜索框
|
||||||
|
const searchClose = () => {
|
||||||
|
appender.value?.focus();
|
||||||
|
};
|
||||||
|
|
||||||
|
// 下载文件
|
||||||
|
const downloadLogFile = async (id: number) => {
|
||||||
|
const data = await downloadExecLogFile(id);
|
||||||
|
downloadFile(data);
|
||||||
|
};
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="less" scoped>
|
<style lang="less" scoped>
|
||||||
@header-height: 38px;
|
@header-height: 40px;
|
||||||
|
|
||||||
.log-view {
|
.log-view {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -136,8 +276,10 @@
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
&-right {
|
.log-action {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
font-size: 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -157,4 +299,16 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.search-modal {
|
||||||
|
--bg-focus: rgba(255, 255, 255, .85);
|
||||||
|
--bg: rgba(255, 255, 255, .95);
|
||||||
|
--color-text: #0E0E0E;
|
||||||
|
--color-text-focus: #0F0F0F;
|
||||||
|
--bg-icon-hover: rgba(12, 12, 12, .04);
|
||||||
|
--bg-icon-hover-focus: rgba(12, 12, 12, .08);
|
||||||
|
--bg-icon-selected: rgba(12, 12, 12, .06);
|
||||||
|
--bg-icon-selected-focus: rgba(12, 12, 12, .10);
|
||||||
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -138,9 +138,10 @@
|
|||||||
@host-real-width: @host-width + 16px;
|
@host-real-width: @host-width + 16px;
|
||||||
|
|
||||||
.log-panel-container {
|
.log-panel-container {
|
||||||
width: 100%;
|
width: calc(100% - 32px);
|
||||||
height: 100%;
|
height: calc(100% - 32px);
|
||||||
display: flex;
|
display: flex;
|
||||||
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.host-container, .log-container {
|
.host-container, .log-container {
|
||||||
|
|||||||
@@ -44,9 +44,10 @@
|
|||||||
<!-- 终端实例 -->
|
<!-- 终端实例 -->
|
||||||
<div class="ssh-inst" ref="terminalRef" />
|
<div class="ssh-inst" ref="terminalRef" />
|
||||||
<!-- 搜索模态框 -->
|
<!-- 搜索模态框 -->
|
||||||
<ssh-search-modal ref="searchModal"
|
<xterm-search-modal ref="searchModal"
|
||||||
@find="findWords"
|
class="search-modal"
|
||||||
@close="focus" />
|
@find="findWords"
|
||||||
|
@close="focus" />
|
||||||
</div>
|
</div>
|
||||||
</ssh-context-menu>
|
</ssh-context-menu>
|
||||||
<!-- 命令编辑器 -->
|
<!-- 命令编辑器 -->
|
||||||
@@ -74,8 +75,8 @@
|
|||||||
import { ActionBarItems, connectStatusKey } from '../../types/terminal.const';
|
import { ActionBarItems, connectStatusKey } from '../../types/terminal.const';
|
||||||
import ShellEditorModal from '@/components/view/shell-editor/modal/index.vue';
|
import ShellEditorModal from '@/components/view/shell-editor/modal/index.vue';
|
||||||
import IconActions from '../layout/icon-actions.vue';
|
import IconActions from '../layout/icon-actions.vue';
|
||||||
import SshSearchModal from './ssh-search-modal.vue';
|
|
||||||
import SshContextMenu from './ssh-context-menu.vue';
|
import SshContextMenu from './ssh-context-menu.vue';
|
||||||
|
import XtermSearchModal from '@/components/xtrem/search-modal/index.vue';
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
tab: TerminalTabItem
|
tab: TerminalTabItem
|
||||||
@@ -251,4 +252,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.search-modal {
|
||||||
|
--bg-focus: var(--search-bg-focus);
|
||||||
|
--bg: var(--search-bg);
|
||||||
|
--color-text: var(--search-color-text);
|
||||||
|
--color-text-focus: var(--search-color-text-focus);
|
||||||
|
--bg-icon-hover: var(--search-bg-icon-hover);
|
||||||
|
--bg-icon-hover-focus: var(--search-bg-icon-hover-focus);
|
||||||
|
--bg-icon-selected: var(--search-bg-icon-selected);
|
||||||
|
--bg-icon-selected-focus: var(--search-bg-icon-selected-focus);
|
||||||
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
Reference in New Issue
Block a user