1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00
lens/src/renderer/components/cluster-settings/components/cluster-local-terminal-settings.tsx
Janne Savolainen 892b105a6b
Add missing gutter between sections in cluster settings (#6631)
Signed-off-by: Janne Savolainen <janne.savolainen@live.fi>

Signed-off-by: Janne Savolainen <janne.savolainen@live.fi>
2022-11-23 12:16:46 +03:00

170 lines
5.5 KiB
TypeScript

/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import React, { useEffect, useState } from "react";
import { observer } from "mobx-react";
import type { Cluster } from "../../../../common/cluster/cluster";
import { Input } from "../../input";
import { SubTitle } from "../../layout/sub-title";
import type { ShowNotification } from "../../notifications";
import { Icon } from "../../icon";
import { PathPicker } from "../../path-picker";
import { isWindows } from "../../../../common/vars";
import { withInjectables } from "@ogre-tools/injectable-react";
import showErrorNotificationInjectable from "../../notifications/show-error-notification.injectable";
import type { ValidateDirectory } from "../../../../common/fs/validate-directory.injectable";
import validateDirectoryInjectable from "../../../../common/fs/validate-directory.injectable";
import type { ResolveTilde } from "../../../../common/path/resolve-tilde.injectable";
import resolveTildeInjectable from "../../../../common/path/resolve-tilde.injectable";
import Gutter from "../../gutter/gutter";
export interface ClusterLocalTerminalSettingProps {
cluster: Cluster;
}
interface Dependencies {
showErrorNotification: ShowNotification;
validateDirectory: ValidateDirectory;
resolveTilde: ResolveTilde;
}
const NonInjectedClusterLocalTerminalSetting = observer(({
cluster,
showErrorNotification,
validateDirectory,
resolveTilde,
}: Dependencies & ClusterLocalTerminalSettingProps) => {
if (!cluster) {
return null;
}
const [directory, setDirectory] = useState<string>(cluster.preferences?.terminalCWD || "");
const [defaultNamespace, setDefaultNamespaces] = useState<string>(cluster.preferences?.defaultNamespace || "");
const [placeholderDefaultNamespace, setPlaceholderDefaultNamespace] = useState("default");
useEffect(() => {
(async () => {
const kubeconfig = await cluster.getKubeconfig();
const { namespace } = kubeconfig.getContextObject(cluster.contextName) ?? {};
if (namespace) {
setPlaceholderDefaultNamespace(namespace);
}
})();
setDirectory(cluster.preferences?.terminalCWD || "");
setDefaultNamespaces(cluster.preferences?.defaultNamespace || "");
}, [cluster]);
const commitDirectory = async (directory: string) => {
cluster.preferences ??= {};
if (!directory) {
cluster.preferences.terminalCWD = undefined;
} else {
const dir = resolveTilde(directory);
const result = await validateDirectory(dir);
if (!result.callWasSuccessful) {
showErrorNotification(
<>
<b>Terminal Working Directory</b>
<p>
{"Your changes were not saved because "}
{result.error}
</p>
</>,
);
} else {
cluster.preferences.terminalCWD = dir;
setDirectory(dir);
}
}
};
const commitDefaultNamespace = () => {
cluster.preferences ??= {};
cluster.preferences.defaultNamespace = defaultNamespace || undefined;
};
const setAndCommitDirectory = (newPath: string) => {
setDirectory(newPath);
commitDirectory(newPath);
};
const openFilePicker = () => {
PathPicker.pick({
label: "Choose Working Directory",
buttonLabel: "Pick",
properties: ["openDirectory", "showHiddenFiles"],
onPick: ([directory]) => setAndCommitDirectory(directory),
});
};
return (
<>
<section className="working-directory">
<SubTitle title="Working Directory"/>
<Input
theme="round-black"
value={directory}
data-testid="working-directory"
onChange={setDirectory}
onBlur={() => commitDirectory(directory)}
placeholder={isWindows ? "$USERPROFILE" : "$HOME"}
iconRight={(
<>
{
directory && (
<Icon
material="close"
title="Clear"
onClick={() => setAndCommitDirectory("")}
smallest
style={{ marginRight: "var(--margin)" }}
/>
)
}
<Icon
material="folder"
title="Pick from filesystem"
onClick={openFilePicker}
smallest
/>
</>
)}
/>
<small className="hint">
An explicit start path where the terminal will be launched,
{" "}
this is used as the current working directory (cwd) for the shell process.
</small>
</section>
<Gutter />
<section className="default-namespace">
<SubTitle title="Default Namespace"/>
<Input
theme="round-black"
data-testid="default-namespace"
value={defaultNamespace}
onChange={setDefaultNamespaces}
onBlur={commitDefaultNamespace}
placeholder={placeholderDefaultNamespace}
/>
<small className="hint">
Default namespace used for kubectl.
</small>
</section>
</>
);
});
export const ClusterLocalTerminalSetting = withInjectables<Dependencies, ClusterLocalTerminalSettingProps>(NonInjectedClusterLocalTerminalSetting, {
getProps: (di, props) => ({
...props,
showErrorNotification: di.inject(showErrorNotificationInjectable),
validateDirectory: di.inject(validateDirectoryInjectable),
resolveTilde: di.inject(resolveTildeInjectable),
}),
});