mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Adding bottom toolbar to logs tab Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Making bottom toolbar responsive Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Using generic search input clear button Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Fixing log test selectors Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>
137 lines
3.4 KiB
TypeScript
137 lines
3.4 KiB
TypeScript
import React from "react";
|
|
import { observable, reaction } from "mobx";
|
|
import { disposeOnUnmount, observer } from "mobx-react";
|
|
|
|
import { searchStore } from "../../../common/search-store";
|
|
import { autobind } from "../../utils";
|
|
import { IDockTab } from "./dock.store";
|
|
import { InfoPanel } from "./info-panel";
|
|
import { LogResourceSelector } from "./log-resource-selector";
|
|
import { LogList } from "./log-list";
|
|
import { IPodLogsData, podLogsStore } from "./log.store";
|
|
import { LogSearch } from "./log-search";
|
|
import { LogControls } from "./log-controls";
|
|
|
|
interface Props {
|
|
className?: string
|
|
tab: IDockTab
|
|
}
|
|
|
|
@observer
|
|
export class Logs extends React.Component<Props> {
|
|
@observable isLoading = true;
|
|
|
|
private logListElement = React.createRef<LogList>(); // A reference for VirtualList component
|
|
|
|
componentDidMount() {
|
|
disposeOnUnmount(this,
|
|
reaction(() => this.props.tab.id, this.reload, { fireImmediately: true })
|
|
);
|
|
}
|
|
|
|
get tabData() {
|
|
return podLogsStore.getData(this.tabId);
|
|
}
|
|
|
|
get tabId() {
|
|
return this.props.tab.id;
|
|
}
|
|
|
|
@autobind()
|
|
save(data: Partial<IPodLogsData>) {
|
|
podLogsStore.setData(this.tabId, { ...this.tabData, ...data });
|
|
}
|
|
|
|
load = async () => {
|
|
this.isLoading = true;
|
|
await podLogsStore.load(this.tabId);
|
|
this.isLoading = false;
|
|
};
|
|
|
|
reload = async () => {
|
|
podLogsStore.clearLogs(this.tabId);
|
|
await this.load();
|
|
};
|
|
|
|
/**
|
|
* A function for various actions after search is happened
|
|
* @param query {string} A text from search field
|
|
*/
|
|
@autobind()
|
|
onSearch() {
|
|
this.toOverlay();
|
|
}
|
|
|
|
/**
|
|
* Scrolling to active overlay (search word highlight)
|
|
*/
|
|
@autobind()
|
|
toOverlay() {
|
|
const { activeOverlayLine } = searchStore;
|
|
|
|
if (!this.logListElement.current || activeOverlayLine === undefined) return;
|
|
// Scroll vertically
|
|
this.logListElement.current.scrollToItem(activeOverlayLine, "center");
|
|
// Scroll horizontally in timeout since virtual list need some time to prepare its contents
|
|
setTimeout(() => {
|
|
const overlay = document.querySelector(".PodLogs .list span.active");
|
|
|
|
if (!overlay) return;
|
|
overlay.scrollIntoViewIfNeeded();
|
|
}, 100);
|
|
}
|
|
|
|
renderResourceSelector() {
|
|
const logs = podLogsStore.logs;
|
|
const searchLogs = this.tabData.showTimestamps ? logs : podLogsStore.logsWithoutTimestamps;
|
|
const controls = (
|
|
<div className="flex gaps">
|
|
<LogResourceSelector
|
|
tabData={this.tabData}
|
|
save={this.save}
|
|
reload={this.reload}
|
|
/>
|
|
<LogSearch
|
|
onSearch={this.onSearch}
|
|
logs={searchLogs}
|
|
toPrevOverlay={this.toOverlay}
|
|
toNextOverlay={this.toOverlay}
|
|
/>
|
|
</div>
|
|
);
|
|
|
|
return (
|
|
<InfoPanel
|
|
tabId={this.props.tab.id}
|
|
controls={controls}
|
|
showSubmitClose={false}
|
|
showButtons={false}
|
|
showStatusPanel={false}
|
|
/>
|
|
);
|
|
}
|
|
|
|
render() {
|
|
const logs = podLogsStore.logs;
|
|
|
|
return (
|
|
<div className="PodLogs flex column">
|
|
{this.renderResourceSelector()}
|
|
<LogList
|
|
logs={logs}
|
|
id={this.tabId}
|
|
isLoading={this.isLoading}
|
|
load={this.load}
|
|
ref={this.logListElement}
|
|
/>
|
|
<LogControls
|
|
logs={logs}
|
|
tabData={this.tabData}
|
|
save={this.save}
|
|
reload={this.reload}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
}
|