From cedebe2196c236636c6bf39ca8b54a748fb9375f Mon Sep 17 00:00:00 2001 From: Red J Adaya Date: Sat, 3 Feb 2024 12:22:30 +0800 Subject: [PATCH 1/2] split settings modals (#272) --- src/app/app.less | 158 +++ src/app/common/modals/index.tsx | 3 + src/app/common/modals/linesettings.less | 23 + src/app/common/modals/linesettings.tsx | 151 +++ src/app/common/modals/registry.tsx | 4 +- src/app/common/modals/screensettings.less | 52 + src/app/common/modals/screensettings.tsx | 365 +++++++ src/app/common/modals/sessionsettings.less | 19 + src/app/common/modals/sessionsettings.tsx | 147 +++ src/app/common/modals/settings.less | 1028 -------------------- src/app/common/modals/settings.tsx | 838 ---------------- 11 files changed, 921 insertions(+), 1867 deletions(-) create mode 100644 src/app/common/modals/linesettings.less create mode 100644 src/app/common/modals/linesettings.tsx create mode 100644 src/app/common/modals/screensettings.less create mode 100644 src/app/common/modals/screensettings.tsx create mode 100644 src/app/common/modals/sessionsettings.less create mode 100644 src/app/common/modals/sessionsettings.tsx delete mode 100644 src/app/common/modals/settings.less delete mode 100644 src/app/common/modals/settings.tsx diff --git a/src/app/app.less b/src/app/app.less index 9d9e84643..d7429b8dc 100644 --- a/src/app/app.less +++ b/src/app/app.less @@ -664,3 +664,161 @@ a.a-block { } } } + +.settings-field { + display: flex; + flex-direction: row; + align-items: center; + + &.settings-field.sub-field { + .settings-label { + font-weight: normal; + + text-align: right; + padding-right: 20px; + } + } + + &.settings-error { + color: @term-red; + margin-top: 20px; + padding: 10px; + border-radius: 5px; + background-color: #200707; + border: 1px solid @term-red; + font-weight: bold; + + .error-dismiss { + padding: 2px; + cursor: pointer; + } + } + + .settings-label { + font-weight: bold; + width: 12em; + display: flex; + flex-direction: row; + align-items: center; + .info-message { + margin-left: 5px; + } + } + + .settings-input { + display: flex; + flex-direction: row; + align-items: center; + color: @term-white; + + &.settings-clickable { + cursor: pointer; + } + + &.inline-edit.edit-active { + input.input { + padding: 0; + height: 20px; + } + + .button { + height: 20px; + } + } + + input { + padding: 4px; + border-radius: 3px; + } + + .control { + .icon { + width: 1.5em; + height: 1.5em; + margin: 0; + } + } + + .tab-color-icon.color-default path { + fill: @tab-green; + } + .tab-color-icon.color-green path { + fill: @tab-green; + } + .tab-color-icon.color-orange path { + fill: @tab-orange; + } + .tab-color-icon.color-red path { + fill: @tab-red; + } + .tab-color-icon.color-yellow path { + fill: @tab-yellow; + } + .tab-color-icon.color-blue path { + fill: @tab-blue; + } + .tab-color-icon.color-mint path { + fill: @tab-mint; + } + .tab-color-icon.color-cyan path { + fill: @tab-cyan; + } + .tab-color-icon.color-white path { + fill: @tab-white; + } + .tab-color-icon.color-violet path { + fill: @tab-violet; + } + .tab-color-icon.color-pink path { + fill: @tab-pink; + } + + .tab-colors, + .tab-icons { + display: flex; + flex-direction: row; + align-items: center; + + .tab-color-sep, + .tab-icon-sep { + padding-left: 10px; + padding-right: 10px; + font-weight: bold; + } + + .tab-color-icon, + .tab-icon-icon { + width: 1.1em; + vertical-align: middle; + } + + .tab-color-name, + .tab-icon-name { + margin-left: 1em; + } + + .tab-color-select, + .tab-icon-select { + cursor: pointer; + margin: 5px; + &:hover { + outline: 2px solid white; + } + } + } + + .action-text { + margin-left: 20px; + + color: @term-red; + } + + .settings-share-link { + width: 160px; + } + } + + &:not(:first-child) { + margin-top: 10px; + } +} diff --git a/src/app/common/modals/index.tsx b/src/app/common/modals/index.tsx index 50da27888..a95fc732f 100644 --- a/src/app/common/modals/index.tsx +++ b/src/app/common/modals/index.tsx @@ -6,3 +6,6 @@ export { CreateRemoteConnModal } from "./createremoteconn"; export { ViewRemoteConnDetailModal } from "./viewremoteconndetail"; export { EditRemoteConnModal } from "./editremoteconn"; export { TabSwitcherModal } from "./tabswitcher"; +export { SessionSettingsModal } from "./sessionsettings"; +export { ScreenSettingsModal } from "./screensettings"; +export { LineSettingsModal } from "./linesettings"; diff --git a/src/app/common/modals/linesettings.less b/src/app/common/modals/linesettings.less new file mode 100644 index 000000000..d7f2c03b0 --- /dev/null +++ b/src/app/common/modals/linesettings.less @@ -0,0 +1,23 @@ +@import "../../../app/common/themes/themes.less"; + +.line-settings-modal { + width: 640px; + + .wave-modal-content { + gap: 24px; + + .wave-modal-body { + display: flex; + padding: 0px 20px; + flex-direction: column; + align-items: flex-start; + gap: 4px; + align-self: stretch; + width: 100%; + + .settings-input .hotkey { + color: @text-secondary; + } + } + } +} diff --git a/src/app/common/modals/linesettings.tsx b/src/app/common/modals/linesettings.tsx new file mode 100644 index 000000000..b7c3d7dda --- /dev/null +++ b/src/app/common/modals/linesettings.tsx @@ -0,0 +1,151 @@ +// Copyright 2023, Command Line Inc. +// SPDX-License-Identifier: Apache-2.0 + +import * as React from "react"; +import * as mobxReact from "mobx-react"; +import * as mobx from "mobx"; +import { boundMethod } from "autobind-decorator"; +import { GlobalModel, GlobalCommandRunner } from "../../../model/model"; +import { SettingsError, Modal, Dropdown } from "../common"; +import { LineType, RendererPluginType } from "../../../types/types"; +import { PluginModel } from "../../../plugins/plugins"; +import { commandRtnHandler } from "../../../util/util"; + +import "./linesettings.less"; + +type OV = mobx.IObservableValue; + +@mobxReact.observer +class LineSettingsModal extends React.Component<{}, {}> { + rendererDropdownActive: OV = mobx.observable.box(false, { name: "lineSettings-rendererDropdownActive" }); + errorMessage: OV = mobx.observable.box(null, { name: "ScreenSettings-errorMessage" }); + linenum: number; + + constructor(props: any) { + super(props); + this.linenum = GlobalModel.lineSettingsModal.get(); + if (this.linenum == null) { + return; + } + } + + @boundMethod + closeModal(): void { + mobx.action(() => { + GlobalModel.lineSettingsModal.set(null); + })(); + GlobalModel.modalsModel.popModal(); + } + + @boundMethod + handleChangeArchived(val: boolean): void { + let line = this.getLine(); + if (line == null) { + return; + } + let prtn = GlobalCommandRunner.lineArchive(line.lineid, val); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + toggleRendererDropdown(): void { + mobx.action(() => { + this.rendererDropdownActive.set(!this.rendererDropdownActive.get()); + })(); + } + + getLine(): LineType { + let screen = GlobalModel.getActiveScreen(); + if (screen == null) { + return; + } + return screen.getLineByNum(this.linenum); + } + + @boundMethod + clickSetRenderer(renderer: string): void { + let line = this.getLine(); + if (line == null) { + return; + } + let prtn = GlobalCommandRunner.lineSet(line.lineid, { renderer: renderer }); + commandRtnHandler(prtn, this.errorMessage); + mobx.action(() => { + this.rendererDropdownActive.set(false); + })(); + } + + getOptions(plugins: RendererPluginType[]) { + // Add label and value to each object in the array + const options = plugins.map((item) => ({ + ...item, + label: item.name, + value: item.name, + })); + + // Create an additional object with label "terminal" and value null + const terminalItem = { + label: "terminal", + value: null, + name: null, + rendererType: null, + heightType: null, + dataType: null, + collapseType: null, + globalCss: null, + mimeTypes: null, + }; + + // Create an additional object with label "none" and value none + const noneItem = { + label: "none", + value: "none", + name: null, + rendererType: null, + heightType: null, + dataType: null, + collapseType: null, + globalCss: null, + mimeTypes: null, + }; + + // Combine the options with the terminal item + return [terminalItem, ...options, noneItem]; + } + + render() { + let line = this.getLine(); + if (line == null) { + setTimeout(() => { + this.closeModal(); + }, 0); + return null; + } + let plugins = PluginModel.rendererPlugins; + let renderer = line.renderer ?? "terminal"; + + return ( + + +
+
+
Renderer
+
+ +
+
+ +
+
+ + + ); + } +} + +export { LineSettingsModal }; diff --git a/src/app/common/modals/registry.tsx b/src/app/common/modals/registry.tsx index 7b01e740b..42d738bd1 100644 --- a/src/app/common/modals/registry.tsx +++ b/src/app/common/modals/registry.tsx @@ -9,8 +9,10 @@ import { ViewRemoteConnDetailModal, EditRemoteConnModal, TabSwitcherModal, + SessionSettingsModal, + ScreenSettingsModal, + LineSettingsModal, } from "../modals"; -import { ScreenSettingsModal, SessionSettingsModal, LineSettingsModal } from "./settings"; import * as constants from "../../appconst"; const modalsRegistry: { [key: string]: () => React.ReactElement } = { diff --git a/src/app/common/modals/screensettings.less b/src/app/common/modals/screensettings.less new file mode 100644 index 000000000..5d2d7674d --- /dev/null +++ b/src/app/common/modals/screensettings.less @@ -0,0 +1,52 @@ +@import "../../../app/common/themes/themes.less"; + +.screen-settings-modal { + width: 640px; + + .wave-modal-content { + gap: 24px; + + .wave-modal-body { + display: flex; + padding: 0px 20px; + flex-direction: column; + align-items: flex-start; + gap: 4px; + align-self: stretch; + width: 100%; + + .screen-settings-dropdown { + width: 412px; + + .lefticon { + position: absolute; + top: 50%; + left: 16px; + transform: translateY(-50%); + + .globe-icon { + width: 16px; + height: 16px; + flex-shrink: 0; + } + + .status-icon { + position: absolute; + left: 7px; + top: 8px; + } + } + } + + .archived-label, + .actions-label { + div:first-child { + margin-right: 5px; + } + div:last-child i { + font-size: 13px; + } + } + } + } +} diff --git a/src/app/common/modals/screensettings.tsx b/src/app/common/modals/screensettings.tsx new file mode 100644 index 000000000..2034ca575 --- /dev/null +++ b/src/app/common/modals/screensettings.tsx @@ -0,0 +1,365 @@ +// Copyright 2023, Command Line Inc. +// SPDX-License-Identifier: Apache-2.0 + +import * as React from "react"; +import * as mobxReact from "mobx-react"; +import * as mobx from "mobx"; +import { boundMethod } from "autobind-decorator"; +import { If, For } from "tsx-control-statements/components"; +import cn from "classnames"; +import { GlobalModel, GlobalCommandRunner, TabColors, TabIcons, Screen } from "../../../model/model"; +import { Toggle, InlineSettingsTextEdit, SettingsError, Modal, Dropdown, Tooltip } from "../common"; +import { RemoteType } from "../../../types/types"; +import * as util from "../../../util/util"; +import { commandRtnHandler } from "../../../util/util"; +import { ReactComponent as SquareIcon } from "../../assets/icons/tab/square.svg"; +import { ReactComponent as GlobeIcon } from "../../assets/icons/globe.svg"; +import { ReactComponent as StatusCircleIcon } from "../../assets/icons/statuscircle.svg"; + +import "./screensettings.less"; + +type OV = mobx.IObservableValue; + +const ScreenDeleteMessage = ` +Are you sure you want to delete this tab? + +All commands and output will be deleted. To hide the tab, and retain the commands and output, use 'archive'. +`.trim(); + +const WebShareConfirmMarkdown = ` +You are about to share a terminal tab on the web. Please make sure that you do +NOT share any private information, keys, passwords, or other sensitive information. +You are responsible for what you are sharing, be smart. +`.trim(); + +const WebStopShareConfirmMarkdown = ` +Are you sure you want to stop web-sharing this tab? +`.trim(); + +@mobxReact.observer +class ScreenSettingsModal extends React.Component<{}, {}> { + shareCopied: OV = mobx.observable.box(false, { name: "ScreenSettings-shareCopied" }); + errorMessage: OV = mobx.observable.box(null, { name: "ScreenSettings-errorMessage" }); + screen: Screen; + sessionId: string; + screenId: string; + remotes: RemoteType[]; + + constructor(props) { + super(props); + let screenSettingsModal = GlobalModel.screenSettingsModal.get(); + let { sessionId, screenId } = screenSettingsModal; + this.sessionId = sessionId; + this.screenId = screenId; + this.screen = GlobalModel.getScreenById(sessionId, screenId); + if (this.screen == null || sessionId == null || screenId == null) { + return; + } + this.remotes = GlobalModel.remotes; + } + + @boundMethod + getOptions(): { label: string; value: string }[] { + return this.remotes + .filter((r) => !r.archived) + .map((remote) => ({ + ...remote, + label: + remote.remotealias && !util.isBlank(remote.remotealias) + ? `${remote.remotecanonicalname}` + : remote.remotecanonicalname, + value: remote.remotecanonicalname, + })) + .sort((a, b) => { + let connValA = util.getRemoteConnVal(a); + let connValB = util.getRemoteConnVal(b); + if (connValA !== connValB) { + return connValA - connValB; + } + return a.remoteidx - b.remoteidx; + }); + } + + @boundMethod + closeModal(): void { + mobx.action(() => { + GlobalModel.screenSettingsModal.set(null); + })(); + GlobalModel.modalsModel.popModal(); + } + + @boundMethod + selectTabColor(color: string): void { + if (this.screen == null) { + return; + } + if (this.screen.getTabColor() == color) { + return; + } + let prtn = GlobalCommandRunner.screenSetSettings(this.screenId, { tabcolor: color }, false); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + selectTabIcon(icon: string): void { + if (this.screen.getTabIcon() == icon) { + return; + } + let prtn = GlobalCommandRunner.screenSetSettings(this.screen.screenId, { tabicon: icon }, false); + util.commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + handleChangeArchived(val: boolean): void { + if (this.screen == null) { + return; + } + if (this.screen.archived.get() == val) { + return; + } + let prtn = GlobalCommandRunner.screenArchive(this.screenId, val); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + handleChangeWebShare(val: boolean): void { + if (this.screen == null) { + return; + } + if (this.screen.isWebShared() == val) { + return; + } + let message = val ? WebShareConfirmMarkdown : WebStopShareConfirmMarkdown; + let alertRtn = GlobalModel.showAlert({ message: message, confirm: true, markdown: true }); + alertRtn.then((result) => { + if (!result) { + return; + } + let prtn = GlobalCommandRunner.screenWebShare(this.screen.screenId, val); + commandRtnHandler(prtn, this.errorMessage); + }); + } + + @boundMethod + copyShareLink(): void { + if (this.screen == null) { + return null; + } + let shareLink = this.screen.getWebShareUrl(); + if (shareLink == null) { + return; + } + navigator.clipboard.writeText(shareLink); + mobx.action(() => { + this.shareCopied.set(true); + })(); + setTimeout(() => { + mobx.action(() => { + this.shareCopied.set(false); + })(); + }, 600); + } + + @boundMethod + inlineUpdateName(val: string): void { + if (this.screen == null) { + return; + } + if (util.isStrEq(val, this.screen.name.get())) { + return; + } + let prtn = GlobalCommandRunner.screenSetSettings(this.screenId, { name: val }, false); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + inlineUpdateShareName(val: string): void { + if (this.screen == null) { + return; + } + if (util.isStrEq(val, this.screen.getShareName())) { + return; + } + let prtn = GlobalCommandRunner.screenSetSettings(this.screenId, { sharename: val }, false); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + dismissError(): void { + mobx.action(() => { + this.errorMessage.set(null); + })(); + } + + @boundMethod + handleDeleteScreen(): void { + if (this.screen == null) { + return; + } + if (this.screen.getScreenLines().lines.length == 0) { + GlobalCommandRunner.screenDelete(this.screenId, false); + GlobalModel.modalsModel.popModal(); + return; + } + let message = ScreenDeleteMessage; + let alertRtn = GlobalModel.showAlert({ message: message, confirm: true, markdown: true }); + alertRtn.then((result) => { + if (!result) { + return; + } + let prtn = GlobalCommandRunner.screenDelete(this.screenId, false); + commandRtnHandler(prtn, this.errorMessage); + GlobalModel.modalsModel.popModal(); + }); + } + + @boundMethod + selectRemote(cname: string): void { + let prtn = GlobalCommandRunner.screenSetRemote(cname, true, false); + util.commandRtnHandler(prtn, this.errorMessage); + } + + render() { + let screen = this.screen; + if (screen == null) { + return null; + } + let color: string = null; + let icon: string = null; + let index: number = 0; + let curRemote = GlobalModel.getRemote(GlobalModel.getActiveScreen().getCurRemoteInstance().remoteid); + + return ( + + +
+
+
Tab Id
+
{screen.screenId}
+
+
+
Name
+
+ +
+
+
+
Connection
+
+ + + +
+ ), + }} + /> +
+
+
+
Tab Color
+
+
+
+ + {screen.getTabColor()} +
+
|
+ +
this.selectTabColor(color)} + > + +
+
+
+
+
+
+
Tab Icon
+
+
+
+ + + + + + + {screen.getTabIcon()} +
+
|
+ +
this.selectTabIcon(icon)} + > + +
+
+
+
+
+
+
+
Archived
+ } + className="screen-settings-tooltip" + > + + +
+
+ +
+
+
+
+
Actions
+ } + className="screen-settings-tooltip" + > + + +
+
+
+ Delete Tab +
+
+
+ +
+ +
+ ); + } +} + +export { ScreenSettingsModal }; diff --git a/src/app/common/modals/sessionsettings.less b/src/app/common/modals/sessionsettings.less new file mode 100644 index 000000000..67e7a7e48 --- /dev/null +++ b/src/app/common/modals/sessionsettings.less @@ -0,0 +1,19 @@ +@import "../../../app/common/themes/themes.less"; + +.session-settings-modal { + width: 640px; + + .wave-modal-content { + gap: 24px; + + .wave-modal-body { + display: flex; + padding: 0px 20px; + flex-direction: column; + align-items: flex-start; + gap: 4px; + align-self: stretch; + width: 100%; + } + } +} diff --git a/src/app/common/modals/sessionsettings.tsx b/src/app/common/modals/sessionsettings.tsx new file mode 100644 index 000000000..f3ed4604f --- /dev/null +++ b/src/app/common/modals/sessionsettings.tsx @@ -0,0 +1,147 @@ +// Copyright 2023, Command Line Inc. +// SPDX-License-Identifier: Apache-2.0 + +import * as React from "react"; +import * as mobxReact from "mobx-react"; +import * as mobx from "mobx"; +import { boundMethod } from "autobind-decorator"; +import { GlobalModel, GlobalCommandRunner, Session } from "../../../model/model"; +import { Toggle, InlineSettingsTextEdit, SettingsError, InfoMessage, Modal } from "../common"; +import * as util from "../../../util/util"; +import { commandRtnHandler } from "../../../util/util"; + +import "./sessionsettings.less"; + +type OV = mobx.IObservableValue; + +const SessionDeleteMessage = ` +Are you sure you want to delete this workspace? + +All commands and output will be deleted. To hide the workspace, and retain the commands and output, use 'archive'. +`.trim(); + +@mobxReact.observer +class SessionSettingsModal extends React.Component<{}, {}> { + errorMessage: OV = mobx.observable.box(null, { name: "ScreenSettings-errorMessage" }); + session: Session; + sessionId: string; + + constructor(props: any) { + super(props); + this.sessionId = GlobalModel.sessionSettingsModal.get(); + this.session = GlobalModel.getSessionById(this.sessionId); + if (this.session == null) { + return; + } + } + + @boundMethod + closeModal(): void { + mobx.action(() => { + GlobalModel.sessionSettingsModal.set(null); + })(); + GlobalModel.modalsModel.popModal(); + } + + @boundMethod + handleInlineChangeName(newVal: string): void { + if (this.session == null) { + return; + } + if (util.isStrEq(newVal, this.session.name.get())) { + return; + } + let prtn = GlobalCommandRunner.sessionSetSettings(this.sessionId, { name: newVal }, false); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + handleChangeArchived(val: boolean): void { + if (this.session == null) { + return; + } + if (this.session.archived.get() == val) { + return; + } + let prtn = GlobalCommandRunner.sessionArchive(this.sessionId, val); + commandRtnHandler(prtn, this.errorMessage); + } + + @boundMethod + handleDeleteSession(): void { + let message = SessionDeleteMessage; + let alertRtn = GlobalModel.showAlert({ message: message, confirm: true, markdown: true }); + alertRtn.then((result) => { + if (!result) { + return; + } + let prtn = GlobalCommandRunner.sessionDelete(this.sessionId); + commandRtnHandler(prtn, this.errorMessage, () => GlobalModel.modalsModel.popModal()); + }); + } + + @boundMethod + dismissError(): void { + mobx.action(() => { + this.errorMessage.set(null); + })(); + } + + render() { + if (this.session == null) { + return null; + } + return ( + + +
+
+
Name
+
+ +
+
+
+
+
Archived
+ + Archive will hide the workspace from the active menu. Commands and output will be + retained in history. + +
+
+ +
+
+
+
+
Actions
+ + Delete will remove the workspace, removing all commands and output from history. + +
+
+
+ Delete Workspace +
+
+
+ +
+ +
+ ); + } +} + +export { SessionSettingsModal }; diff --git a/src/app/common/modals/settings.less b/src/app/common/modals/settings.less deleted file mode 100644 index 7bd32d415..000000000 --- a/src/app/common/modals/settings.less +++ /dev/null @@ -1,1028 +0,0 @@ -@import "../../../app/common/themes/themes.less"; - -// modal css (also includes settings-field) - -.modal { - z-index: 105; - - .modal-background { - background-color: @base-background-transparent; - } -} - -.tos-modal { - a:hover { - text-decoration: underline; - } -} - -.disconnected-modal { - .wave-modal-content { - .wave-modal-body { - padding: 0; - - .modal-content { - footer { - .footer-text-link { - color: @term-white; - cursor: pointer; - } - } - } - - .inner-content { - .log { - height: 335px; - margin-bottom: 20px; - overflow: auto; - - &::-webkit-scrollbar-track, - &::-webkit-scrollbar-thumb, - &::-webkit-scrollbar-corner { - display: none; - } - - &:hover::-webkit-scrollbar-thumb { - display: block; - } - - pre { - color: @term-white; - background-color: @term-black; - } - } - } - } - - .wave-modal-footer { - button:first-child { - color: @term-green; - } - } - } -} - -.modal.prompt-modal.client-stop-modal { - footer { - justify-content: center; - } - - .inner-content { - display: flex; - flex-direction: column; - padding: 20px; - .progress-container { - margin-top: 20px; - } - .progress-text { - color: @term-white; - align-self: center; - } - } -} - -.modal.settings-modal { - footer { - justify-content: center; - - .button { - margin-left: 20px; - } - - .button:first-child { - margin-left: 0; - } - } - - .dropdown { - .dropdown-menu .dropdown-content { - .dropdown-item { - color: @term-white; - cursor: pointer; - &:hover { - background-color: #333; - } - } - } - - .dropdown-menu { - max-height: 120px; - overflow: auto; - } - } -} - -.modal.prompt-modal { - .modal-content { - border-radius: 5px; - background: @base-background; - box-shadow: 0px 2px 2px 0 rgba(255, 255, 255, 0.1), 0px 4px 5px 0 rgba(255, 255, 255, 0.2), - 0px 0px 5px 2.5px rgba(255, 255, 255, 0.5); - padding-bottom: 0.5em; - } - - header { - display: flex; - flex-direction: row; - padding: 10px; - position: relative; - background-color: @background-session-components-solid; - - .modal-title { - color: @base-color; - font-size: 1.5em; - .icon { - display: inline; - width: 1em; - height: 1em; - margin-right: 1em; - vertical-align: middle; - } - } - - .close-icon { - position: absolute; - right: 1em; - top: 0.8em; - cursor: pointer; - width: 1.5em; - height: 1.5em; - border-radius: 50%; - svg { - width: 1.5em; - height: 1.5em; - fill: @base-color; - } - } - } - - .inner-content { - padding: 1em; - margin: 0; - border-top: 1px solid fade(@base-border, 50%); - .icon { - display: inline; - width: 1em; - height: 1em; - } - &.is-hidden { - display: none; - } - .dropdown-trigger button { - justify-content: flex-start; - display: flex; - color: @base-color; - border: none !important; - &:hover { - box-shadow: none; - } - } - } - - footer { - display: flex; - flex-direction: row; - align-items: center; - padding: 15px 10px 10px 10px; - } -} - -.tos-modal { - width: 640px; - - .wave-modal-content .wave-modal-body { - padding: 32px 48px; - gap: 8px; - - .wave-modal-body-inner { - gap: 24px; - display: flex; - flex-direction: column; - - header.tos-header { - flex-direction: column; - gap: var(--sizing-sm, 12px); - border-bottom: none; - padding: 0; - - .modal-title { - text-align: center; - font-size: 20px; - font-weight: 300; - } - - .modal-subtitle { - color: @term-white; - text-align: center; - - font-style: normal; - font-weight: 300; - line-height: 20px; - } - } - - .content.tos-content { - display: flex; - flex-direction: column; - align-items: flex-start; - gap: 32px; - width: 100%; - margin-bottom: 0; - - .item { - display: flex; - width: 100%; - align-items: center; - gap: 16px; - - .item-inner { - display: flex; - flex-direction: column; - align-items: flex-start; - gap: 4px; - flex: 1 0 0; - - .item-title { - color: @term-bright-white; - font-style: normal; - line-height: 20px; - } - - .item-text { - color: @term-white; - font-style: normal; - line-height: 20px; - } - - .item-field { - display: flex; - align-items: center; - gap: 8px; - } - } - } - } - - footer { - .item-text { - text-align: center; - } - - .button-wrapper { - display: flex; - flex-direction: row; - align-items: center; - justify-content: center; - - button { - font-size: 12.5px !important; - margin-top: 16px; - } - - button.disabled-button { - cursor: default; - } - } - } - } - } -} - -.about-modal { - .wave-modal-content { - gap: 24px; - - .wave-modal-body { - margin-bottom: 0; - display: flex; - flex-direction: column; - align-items: flex-start; - gap: 24px; - - .about-section { - display: flex; - align-items: center; - gap: 16px; - align-self: stretch; - width: 100%; - - .logo-wrapper { - width: 72px; - height: 72px; - flex-shrink: 0; - - img { - border-radius: 10px; - } - } - - .text-wrapper { - display: flex; - align-items: flex-start; - flex-direction: column; - gap: 4px; - align-self: stretch; - font-style: normal; - line-height: 20px; - - div:first-child { - color: @term-bright-white; - font-size: 14.5px; - } - - div:last-child { - color: @term-white; - text-align: left; - } - } - - .status { - div { - display: flex; - align-items: center; - margin-bottom: 5px; - - i { - font-size: 16px; - margin-right: 10px; - } - } - - div:first-child + div { - color: @term-white; - } - } - - .status.updated { - div { - display: flex; - align-items: center; - margin-bottom: 5px; - - i { - color: @term-green; - } - } - } - - .status.outdated { - div { - i { - color: @term-yellow; - } - } - - button { - margin-top: 5px; - } - } - } - - .about-section:nth-child(3) { - display: flex; - align-items: flex-start; - gap: 10px; - - .wave-button-link { - display: flex; - align-items: center; - - i { - font-size: 16px; - } - } - } - - .about-section:last-child { - margin-bottom: 24px; - color: @term-white; - } - } - } -} - -.crconn-modal { - width: 452px; - min-height: 411px; - - .wave-modal-content { - gap: 24px; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - flex-direction: column; - align-items: flex-start; - gap: 12px; - align-self: stretch; - width: 100%; - - > div { - width: 100%; - } - } - } -} - -.screen-settings-modal { - width: 640px; - - .wave-modal-content { - gap: 24px; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - flex-direction: column; - align-items: flex-start; - gap: 4px; - align-self: stretch; - width: 100%; - - .screen-settings-dropdown { - width: 412px; - - .lefticon { - position: absolute; - top: 50%; - left: 16px; - transform: translateY(-50%); - - .globe-icon { - width: 16px; - height: 16px; - flex-shrink: 0; - } - - .status-icon { - position: absolute; - left: 7px; - top: 8px; - } - } - } - - .archived-label, - .actions-label { - div:first-child { - margin-right: 5px; - } - div:last-child i { - font-size: 13px; - } - } - } - } -} - -.tabswitcher-modal { - width: 452px; - min-height: 384px; - - .wave-modal-content { - .wave-modal-body { - display: flex; - padding: 0px; - flex-direction: column; - align-items: flex-start; - align-self: stretch; - width: 100%; - - .textfield-wrapper { - padding: 20px 20px 0px; - - .wave-input-decoration.start-position { - height: 100%; - - .tabswitcher-search-prefix { - opacity: 0.5; - font-size: 13px; - } - } - } - - .list-container { - overflow: hidden; - padding: 10px 0 20px; - width: 100%; - } - - .list-container-inner { - width: 100%; - max-height: 300px; - overflow-y: scroll; - padding: 0 16px 0 20px; - - &::-webkit-scrollbar-thumb { - display: none; - } - - &:hover::-webkit-scrollbar-thumb { - display: block; - } - - .options-list { - width: 100%; - - .search-option { - padding: 5px 5px 5px 8px; - display: flex; - align-items: center; - border: 1px solid transparent; - width: 100%; - overflow: hidden; - - div.tabname { - flex-grow: 1; - overflow: hidden; - text-overflow: ellipsis; - white-space: nowrap; - padding-right: 5px; - } - - div.icon { - flex-shrink: 0; - width: 20px; - margin-right: 6px; - } - } - - .focused-option { - border: 1px solid rgba(241, 246, 243, 0.15); - border-radius: 4px; - background: rgba(255, 255, 255, 0.06); - } - } - } - } - } -} - -.screen-settings-tooltip .wave-tooltip-icon { - i { - font-size: 13px; - } -} - -.session-settings-modal { - width: 640px; - - .wave-modal-content { - gap: 24px; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - flex-direction: column; - align-items: flex-start; - gap: 4px; - align-self: stretch; - width: 100%; - } - } -} - -.line-settings-modal { - width: 640px; - - .wave-modal-content { - gap: 24px; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - flex-direction: column; - align-items: flex-start; - gap: 4px; - align-self: stretch; - width: 100%; - - .settings-input .hotkey { - color: @text-secondary; - } - } - } -} - -.client-settings-modal { - width: 640px; - - .wave-modal-content { - gap: 24px; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - flex-direction: column; - align-items: flex-start; - gap: 4px; - align-self: stretch; - width: 100%; - } - } -} - -.erconn-modal { - width: 502px; - min-height: 411px; - - .wave-modal-content { - gap: 20px; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - flex-direction: column; - align-items: flex-start; - gap: 12px; - align-self: stretch; - width: 100%; - - > div { - width: 100%; - } - - .name-actions-section { - margin-bottom: 10px; - display: flex; - flex-direction: column; - align-items: flex-start; - gap: 12px; - - .name { - color: @term-bright-white; - font-size: 15px; - font-weight: 500; - line-height: 20px; - } - - .header-actions { - display: flex; - justify-content: flex-end; - align-items: flex-start; - - .wave-button { - padding: 4px 15px; - font-size: 11px; - margin-right: 8px; - } - } - } - } - } -} - -.alert-modal { - width: 510px; - - .wave-modal-content { - .wave-modal-body { - padding: 40px 20px; - - .dontshowagain-text { - margin-top: 15px; - } - } - } -} - -.rconndetail-modal { - width: 631px; - min-height: 565px; - - .wave-modal-content { - display: flex; - padding-bottom: 0px; - flex-direction: column; - align-items: center; - gap: 20px; - flex-shrink: 0; - - .wave-modal-body { - display: flex; - padding: 0px 20px; - align-items: flex-start; - width: 100%; - display: flex; - flex-direction: column; - gap: 16px; - align-self: stretch; - - .name-header-actions-wrapper { - display: flex; - flex-direction: column; - align-items: flex-start; - gap: 12px; - - .name-wrapper { - display: flex; - flex-direction: row; - } - - .rconndetail-name { - color: @term-bright-white; - font-size: 15px; - font-weight: 500; - line-height: 20px; - } - - .header-actions { - display: flex; - justify-content: flex-end; - align-items: flex-start; - - .wave-button { - padding: 4px 15px; - font-size: 11px; - margin-right: 8px; - } - } - } - - .remote-detail { - .settings-field { - display: flex; - flex-direction: row; - align-items: center; - - .settings-label { - font-weight: bold; - width: 12em; - display: flex; - flex-direction: row; - align-items: center; - } - - .settings-input { - display: flex; - flex-direction: row; - align-items: center; - color: @term-white; - } - } - - .settings-field:not(:first-child) { - margin-top: 4px; - } - - .status { - display: flex; - height: 30px; - padding: 3px 8px; - align-items: center; - gap: 8px; - align-self: stretch; - border-radius: 6px; - background: rgba(241, 246, 243, 0.08); - } - - .terminal-wrapper { - width: 100%; - margin-top: 5px; - - .terminal-connectelem { - height: 163px !important; // Needed to override plugin height - - .xterm-viewport { - display: flex; - padding: 6px 10px; - gap: 8px; - align-items: flex-start; - align-self: stretch; - border-radius: 6px; - border: 1px solid var(--element-separator, rgba(241, 246, 243, 0.15)); - background: #080a08; - height: 163px !important; // Needed to override plugin height - } - - .xterm-screen { - padding: 10px; - width: 541px !important; // Needed to override plugin width - } - } - } - } - } - } -} - -.wave-button.color-standard { - color: @term-white; - background: rgba(255, 255, 255, 0.12); - - &:hover { - color: @term-white; - } -} - -.wave-button-link { - display: flex; - padding: 6px 16px; - align-items: center; - gap: var(--sizing-2-xs, 4px); - border-radius: 6px; - background: var(--overlays-white-6, rgba(255, 255, 255, 0.12)); - cursor: pointer; -} - -.wave-modal-section { - display: flex; - align-items: center; - gap: 16px; - align-self: stretch; - width: 100%; -} - -.modal.welcome-modal { - footer { - .prev-button { - margin-left: 15px; - width: 100px; - } - - .prev-spacer { - margin-left: 15px; - width: 100px; - } - - .next-button { - margin-right: 15px; - width: 100px; - } - - .dots { - i { - margin-left: 5px; - } - } - } -} - -.client-settings-modal { - .settings-field { - .settings-label { - width: 157px; - } - .dropdown-menu { - min-width: 7.25em; - max-width: 7.25em; - } - } -} - -.settings-field { - display: flex; - flex-direction: row; - align-items: center; - - &.settings-field.sub-field { - .settings-label { - font-weight: normal; - - text-align: right; - padding-right: 20px; - } - } - - &.settings-error { - color: @term-red; - margin-top: 20px; - padding: 10px; - border-radius: 5px; - background-color: #200707; - border: 1px solid @term-red; - font-weight: bold; - - .error-dismiss { - padding: 2px; - cursor: pointer; - } - } - - .settings-label { - font-weight: bold; - width: 12em; - display: flex; - flex-direction: row; - align-items: center; - .info-message { - margin-left: 5px; - } - } - - .settings-input { - display: flex; - flex-direction: row; - align-items: center; - - &.settings-clickable { - cursor: pointer; - } - - &.inline-edit.edit-active { - input.input { - padding: 0; - height: 20px; - } - - .button { - height: 20px; - } - } - - input { - padding: 4px; - border-radius: 3px; - } - - .control { - .icon { - width: 1.5em; - height: 1.5em; - margin: 0; - } - } - - .tab-color-icon.color-default path { - fill: @tab-green; - } - .tab-color-icon.color-green path { - fill: @tab-green; - } - .tab-color-icon.color-orange path { - fill: @tab-orange; - } - .tab-color-icon.color-red path { - fill: @tab-red; - } - .tab-color-icon.color-yellow path { - fill: @tab-yellow; - } - .tab-color-icon.color-blue path { - fill: @tab-blue; - } - .tab-color-icon.color-mint path { - fill: @tab-mint; - } - .tab-color-icon.color-cyan path { - fill: @tab-cyan; - } - .tab-color-icon.color-white path { - fill: @tab-white; - } - .tab-color-icon.color-violet path { - fill: @tab-violet; - } - .tab-color-icon.color-pink path { - fill: @tab-pink; - } - - .tab-colors, - .tab-icons { - display: flex; - flex-direction: row; - align-items: center; - - .tab-color-sep, - .tab-icon-sep { - padding-left: 10px; - padding-right: 10px; - font-weight: bold; - } - - .tab-color-icon, - .tab-icon-icon { - width: 1.1em; - vertical-align: middle; - } - - .tab-color-name, - .tab-icon-name { - margin-left: 1em; - } - - .tab-color-select, - .tab-icon-select { - cursor: pointer; - margin: 5px; - &:hover { - outline: 2px solid white; - } - } - } - - .action-text { - margin-left: 20px; - - color: @term-red; - } - - .settings-share-link { - width: 160px; - } - } - - &:not(:first-child) { - margin-top: 10px; - } -} diff --git a/src/app/common/modals/settings.tsx b/src/app/common/modals/settings.tsx deleted file mode 100644 index 7f61e2ab0..000000000 --- a/src/app/common/modals/settings.tsx +++ /dev/null @@ -1,838 +0,0 @@ -// Copyright 2023, Command Line Inc. -// SPDX-License-Identifier: Apache-2.0 - -import * as React from "react"; -import * as mobxReact from "mobx-react"; -import * as mobx from "mobx"; -import { boundMethod } from "autobind-decorator"; -import { If, For } from "tsx-control-statements/components"; -import cn from "classnames"; -import { - GlobalModel, - GlobalCommandRunner, - TabColors, - MinFontSize, - MaxFontSize, - TabIcons, - Screen, - Session, -} from "../../../model/model"; -import { Toggle, InlineSettingsTextEdit, SettingsError, InfoMessage, Modal, Dropdown, Tooltip } from "../common"; -import { LineType, RendererPluginType, ClientDataType, CommandRtnType, RemoteType } from "../../../types/types"; -import { PluginModel } from "../../../plugins/plugins"; -import * as util from "../../../util/util"; -import { commandRtnHandler } from "../../../util/util"; -import { ReactComponent as SquareIcon } from "../../assets/icons/tab/square.svg"; -import { ReactComponent as GlobeIcon } from "../../assets/icons/globe.svg"; -import { ReactComponent as StatusCircleIcon } from "../../assets/icons/statuscircle.svg"; - -import "./settings.less"; - -type OV = mobx.IObservableValue; - -// @ts-ignore -const VERSION = __WAVETERM_VERSION__; -// @ts-ignore -const BUILD = __WAVETERM_BUILD__; - -const ScreenDeleteMessage = ` -Are you sure you want to delete this tab? - -All commands and output will be deleted. To hide the tab, and retain the commands and output, use 'archive'. -`.trim(); - -const SessionDeleteMessage = ` -Are you sure you want to delete this workspace? - -All commands and output will be deleted. To hide the workspace, and retain the commands and output, use 'archive'. -`.trim(); - -const WebShareConfirmMarkdown = ` -You are about to share a terminal tab on the web. Please make sure that you do -NOT share any private information, keys, passwords, or other sensitive information. -You are responsible for what you are sharing, be smart. -`.trim(); - -const WebStopShareConfirmMarkdown = ` -Are you sure you want to stop web-sharing this tab? -`.trim(); - -@mobxReact.observer -class ScreenSettingsModal extends React.Component<{}, {}> { - shareCopied: OV = mobx.observable.box(false, { name: "ScreenSettings-shareCopied" }); - errorMessage: OV = mobx.observable.box(null, { name: "ScreenSettings-errorMessage" }); - screen: Screen; - sessionId: string; - screenId: string; - remotes: RemoteType[]; - - constructor(props) { - super(props); - let screenSettingsModal = GlobalModel.screenSettingsModal.get(); - let { sessionId, screenId } = screenSettingsModal; - this.sessionId = sessionId; - this.screenId = screenId; - this.screen = GlobalModel.getScreenById(sessionId, screenId); - if (this.screen == null || sessionId == null || screenId == null) { - return; - } - this.remotes = GlobalModel.remotes; - } - - @boundMethod - getOptions(): { label: string; value: string }[] { - return this.remotes - .filter((r) => !r.archived) - .map((remote) => ({ - ...remote, - label: - remote.remotealias && !util.isBlank(remote.remotealias) - ? `${remote.remotecanonicalname}` - : remote.remotecanonicalname, - value: remote.remotecanonicalname, - })) - .sort((a, b) => { - let connValA = util.getRemoteConnVal(a); - let connValB = util.getRemoteConnVal(b); - if (connValA !== connValB) { - return connValA - connValB; - } - return a.remoteidx - b.remoteidx; - }); - } - - @boundMethod - closeModal(): void { - mobx.action(() => { - GlobalModel.screenSettingsModal.set(null); - })(); - GlobalModel.modalsModel.popModal(); - } - - @boundMethod - selectTabColor(color: string): void { - if (this.screen == null) { - return; - } - if (this.screen.getTabColor() == color) { - return; - } - let prtn = GlobalCommandRunner.screenSetSettings(this.screenId, { tabcolor: color }, false); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - selectTabIcon(icon: string): void { - if (this.screen.getTabIcon() == icon) { - return; - } - let prtn = GlobalCommandRunner.screenSetSettings(this.screen.screenId, { tabicon: icon }, false); - util.commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - handleChangeArchived(val: boolean): void { - if (this.screen == null) { - return; - } - if (this.screen.archived.get() == val) { - return; - } - let prtn = GlobalCommandRunner.screenArchive(this.screenId, val); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - handleChangeWebShare(val: boolean): void { - if (this.screen == null) { - return; - } - if (this.screen.isWebShared() == val) { - return; - } - let message = val ? WebShareConfirmMarkdown : WebStopShareConfirmMarkdown; - let alertRtn = GlobalModel.showAlert({ message: message, confirm: true, markdown: true }); - alertRtn.then((result) => { - if (!result) { - return; - } - let prtn = GlobalCommandRunner.screenWebShare(this.screen.screenId, val); - commandRtnHandler(prtn, this.errorMessage); - }); - } - - @boundMethod - copyShareLink(): void { - if (this.screen == null) { - return null; - } - let shareLink = this.screen.getWebShareUrl(); - if (shareLink == null) { - return; - } - navigator.clipboard.writeText(shareLink); - mobx.action(() => { - this.shareCopied.set(true); - })(); - setTimeout(() => { - mobx.action(() => { - this.shareCopied.set(false); - })(); - }, 600); - } - - @boundMethod - inlineUpdateName(val: string): void { - if (this.screen == null) { - return; - } - if (util.isStrEq(val, this.screen.name.get())) { - return; - } - let prtn = GlobalCommandRunner.screenSetSettings(this.screenId, { name: val }, false); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - inlineUpdateShareName(val: string): void { - if (this.screen == null) { - return; - } - if (util.isStrEq(val, this.screen.getShareName())) { - return; - } - let prtn = GlobalCommandRunner.screenSetSettings(this.screenId, { sharename: val }, false); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - dismissError(): void { - mobx.action(() => { - this.errorMessage.set(null); - })(); - } - - @boundMethod - handleDeleteScreen(): void { - if (this.screen == null) { - return; - } - if (this.screen.getScreenLines().lines.length == 0) { - GlobalCommandRunner.screenDelete(this.screenId, false); - GlobalModel.modalsModel.popModal(); - return; - } - let message = ScreenDeleteMessage; - let alertRtn = GlobalModel.showAlert({ message: message, confirm: true, markdown: true }); - alertRtn.then((result) => { - if (!result) { - return; - } - let prtn = GlobalCommandRunner.screenDelete(this.screenId, false); - commandRtnHandler(prtn, this.errorMessage); - GlobalModel.modalsModel.popModal(); - }); - } - - @boundMethod - selectRemote(cname: string): void { - let prtn = GlobalCommandRunner.screenSetRemote(cname, true, false); - util.commandRtnHandler(prtn, this.errorMessage); - } - - render() { - let screen = this.screen; - if (screen == null) { - return null; - } - let color: string = null; - let icon: string = null; - let index: number = 0; - let curRemote = GlobalModel.getRemote(GlobalModel.getActiveScreen().getCurRemoteInstance().remoteid); - - return ( - - -
-
-
Tab Id
-
{screen.screenId}
-
-
-
Name
-
- -
-
-
-
Connection
-
- - - -
- ), - }} - /> -
-
-
-
Tab Color
-
-
-
- - {screen.getTabColor()} -
-
|
- -
this.selectTabColor(color)} - > - -
-
-
-
-
-
-
Tab Icon
-
-
-
- - - - - - - {screen.getTabIcon()} -
-
|
- -
this.selectTabIcon(icon)} - > - -
-
-
-
-
-
-
-
Archived
- } - className="screen-settings-tooltip" - > - - -
-
- -
-
-
-
-
Actions
- } - className="screen-settings-tooltip" - > - - -
-
-
- Delete Tab -
-
-
- - - -
- ); - } -} - -@mobxReact.observer -class SessionSettingsModal extends React.Component<{}, {}> { - errorMessage: OV = mobx.observable.box(null, { name: "ScreenSettings-errorMessage" }); - session: Session; - sessionId: string; - - constructor(props: any) { - super(props); - this.sessionId = GlobalModel.sessionSettingsModal.get(); - this.session = GlobalModel.getSessionById(this.sessionId); - if (this.session == null) { - return; - } - } - - @boundMethod - closeModal(): void { - mobx.action(() => { - GlobalModel.sessionSettingsModal.set(null); - })(); - GlobalModel.modalsModel.popModal(); - } - - @boundMethod - handleInlineChangeName(newVal: string): void { - if (this.session == null) { - return; - } - if (util.isStrEq(newVal, this.session.name.get())) { - return; - } - let prtn = GlobalCommandRunner.sessionSetSettings(this.sessionId, { name: newVal }, false); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - handleChangeArchived(val: boolean): void { - if (this.session == null) { - return; - } - if (this.session.archived.get() == val) { - return; - } - let prtn = GlobalCommandRunner.sessionArchive(this.sessionId, val); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - handleDeleteSession(): void { - let message = SessionDeleteMessage; - let alertRtn = GlobalModel.showAlert({ message: message, confirm: true, markdown: true }); - alertRtn.then((result) => { - if (!result) { - return; - } - let prtn = GlobalCommandRunner.sessionDelete(this.sessionId); - commandRtnHandler(prtn, this.errorMessage, () => GlobalModel.modalsModel.popModal()); - }); - } - - @boundMethod - dismissError(): void { - mobx.action(() => { - this.errorMessage.set(null); - })(); - } - - render() { - if (this.session == null) { - return null; - } - return ( - - -
-
-
Name
-
- -
-
-
-
-
Archived
- - Archive will hide the workspace from the active menu. Commands and output will be - retained in history. - -
-
- -
-
-
-
-
Actions
- - Delete will remove the workspace, removing all commands and output from history. - -
-
-
- Delete Workspace -
-
-
- -
- -
- ); - } -} - -@mobxReact.observer -class LineSettingsModal extends React.Component<{}, {}> { - rendererDropdownActive: OV = mobx.observable.box(false, { name: "lineSettings-rendererDropdownActive" }); - errorMessage: OV = mobx.observable.box(null, { name: "ScreenSettings-errorMessage" }); - linenum: number; - - constructor(props: any) { - super(props); - this.linenum = GlobalModel.lineSettingsModal.get(); - if (this.linenum == null) { - return; - } - } - - @boundMethod - closeModal(): void { - mobx.action(() => { - GlobalModel.lineSettingsModal.set(null); - })(); - GlobalModel.modalsModel.popModal(); - } - - @boundMethod - handleChangeArchived(val: boolean): void { - let line = this.getLine(); - if (line == null) { - return; - } - let prtn = GlobalCommandRunner.lineArchive(line.lineid, val); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - toggleRendererDropdown(): void { - mobx.action(() => { - this.rendererDropdownActive.set(!this.rendererDropdownActive.get()); - })(); - } - - getLine(): LineType { - let screen = GlobalModel.getActiveScreen(); - if (screen == null) { - return; - } - return screen.getLineByNum(this.linenum); - } - - @boundMethod - clickSetRenderer(renderer: string): void { - let line = this.getLine(); - if (line == null) { - return; - } - let prtn = GlobalCommandRunner.lineSet(line.lineid, { renderer: renderer }); - commandRtnHandler(prtn, this.errorMessage); - mobx.action(() => { - this.rendererDropdownActive.set(false); - })(); - } - - getOptions(plugins: RendererPluginType[]) { - // Add label and value to each object in the array - const options = plugins.map((item) => ({ - ...item, - label: item.name, - value: item.name, - })); - - // Create an additional object with label "terminal" and value null - const terminalItem = { - label: "terminal", - value: null, - name: null, - rendererType: null, - heightType: null, - dataType: null, - collapseType: null, - globalCss: null, - mimeTypes: null, - }; - - // Create an additional object with label "none" and value none - const noneItem = { - label: "none", - value: "none", - name: null, - rendererType: null, - heightType: null, - dataType: null, - collapseType: null, - globalCss: null, - mimeTypes: null, - }; - - // Combine the options with the terminal item - return [terminalItem, ...options, noneItem]; - } - - render() { - let line = this.getLine(); - if (line == null) { - setTimeout(() => { - this.closeModal(); - }, 0); - return null; - } - let plugins = PluginModel.rendererPlugins; - let renderer = line.renderer ?? "terminal"; - - return ( - - -
-
-
Renderer
-
- -
-
- -
-
- - - ); - } -} - -@mobxReact.observer -class ClientSettingsModal extends React.Component<{}, {}> { - fontSizeDropdownActive: OV = mobx.observable.box(false, { name: "clientSettings-fontSizeDropdownActive" }); - errorMessage: OV = mobx.observable.box(null, { name: "ClientSettings-errorMessage" }); - - @boundMethod - closeModal(): void { - GlobalModel.modalsModel.popModal(); - } - - @boundMethod - dismissError(): void { - mobx.action(() => { - this.errorMessage.set(null); - })(); - } - - @boundMethod - handleChangeFontSize(fontSize: string): void { - let newFontSize = Number(fontSize); - this.fontSizeDropdownActive.set(false); - if (GlobalModel.termFontSize.get() == newFontSize) { - return; - } - let prtn = GlobalCommandRunner.setTermFontSize(newFontSize, false); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - togglefontSizeDropdown(): void { - mobx.action(() => { - this.fontSizeDropdownActive.set(!this.fontSizeDropdownActive.get()); - })(); - } - - @boundMethod - handleChangeTelemetry(val: boolean): void { - let prtn: Promise = null; - if (val) { - prtn = GlobalCommandRunner.telemetryOn(false); - } else { - prtn = GlobalCommandRunner.telemetryOff(false); - } - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - handleChangeReleaseCheck(val: boolean): void { - let prtn: Promise = null; - if (val) { - prtn = GlobalCommandRunner.releaseCheckAutoOn(false); - } else { - prtn = GlobalCommandRunner.releaseCheckAutoOff(false); - } - commandRtnHandler(prtn, this.errorMessage); - } - - getFontSizes(): any { - let availableFontSizes: { label: string; value: number }[] = []; - for (let s = MinFontSize; s <= MaxFontSize; s++) { - availableFontSizes.push({ label: s + "px", value: s }); - } - return availableFontSizes; - } - - @boundMethod - inlineUpdateOpenAIModel(newModel: string): void { - let prtn = GlobalCommandRunner.setClientOpenAISettings({ model: newModel }); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - inlineUpdateOpenAIToken(newToken: string): void { - let prtn = GlobalCommandRunner.setClientOpenAISettings({ apitoken: newToken }); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - inlineUpdateOpenAIMaxTokens(newMaxTokensStr: string): void { - let prtn = GlobalCommandRunner.setClientOpenAISettings({ maxtokens: newMaxTokensStr }); - commandRtnHandler(prtn, this.errorMessage); - } - - @boundMethod - setErrorMessage(msg: string): void { - mobx.action(() => { - this.errorMessage.set(msg); - })(); - } - - render() { - let cdata: ClientDataType = GlobalModel.clientData.get(); - let openAIOpts = cdata.openaiopts ?? {}; - let apiTokenStr = util.isBlank(openAIOpts.apitoken) ? "(not set)" : "********"; - let maxTokensStr = String( - openAIOpts.maxtokens == null || openAIOpts.maxtokens == 0 ? 1000 : openAIOpts.maxtokens - ); - let curFontSize = GlobalModel.termFontSize.get(); - return ( - - -
-
-
Term Font Size
-
- -
-
-
-
Client ID
-
{cdata.clientid}
-
-
-
Client Version
-
- {VERSION} {BUILD} -
-
-
-
DB Version
-
{cdata.dbversion}
-
-
-
Basic Telemetry
-
- -
-
-
-
Check for Updates
-
- -
-
-
-
OpenAI Token
-
- -
-
-
-
OpenAI Model
-
- -
-
-
-
OpenAI MaxTokens
-
- -
-
- -
- -
- ); - } -} - -export { - ScreenSettingsModal, - SessionSettingsModal, - LineSettingsModal, - ClientSettingsModal, - WebStopShareConfirmMarkdown, -}; From fc65e65f11946f8294f687b80a6e07f2f909d567 Mon Sep 17 00:00:00 2001 From: Red J Adaya Date: Sat, 3 Feb 2024 12:23:10 +0800 Subject: [PATCH 2/2] remove old connections (#273) --- .../connections_deprecated/connections.less | 408 ------ .../connections_deprecated/connections.tsx | 1300 ----------------- 2 files changed, 1708 deletions(-) delete mode 100644 src/app/connections_deprecated/connections.less delete mode 100644 src/app/connections_deprecated/connections.tsx diff --git a/src/app/connections_deprecated/connections.less b/src/app/connections_deprecated/connections.less deleted file mode 100644 index 369681b30..000000000 --- a/src/app/connections_deprecated/connections.less +++ /dev/null @@ -1,408 +0,0 @@ -@import "../../app/common/themes/themes.less"; - -.modal.prompt-modal.remotes-modal { - .modal-content { - min-width: 850px; - } - .icon { - width: 1em; - height: 1em; - fill: @base-color; - margin: 0; - } - .button { - svg { - float: right; - margin-top: 0.3em; - margin-right: 0; - } - } - .dropdown, - .button { - display: inline-flex; - } - .dropdown .button { - border: none !important; - } - .inner-content { - display: flex; - flex-direction: row; - align-items: stretch; - padding: 0; - min-height: 45em; - max-height: 45em; - - .remotes-menu { - flex: 0 0 200px; - border-right: 1px solid @disabled-color; - overflow-y: auto; - - .remote-menu-item { - border-top: 1px solid @disabled-color; - padding: 0.5em; - display: flex; - flex-direction: row; - cursor: pointer; - - &.add-remote { - padding: 10px 5px 10px 5px; - } - - &:hover { - background-color: #333; - } - - &.is-selected { - background-color: @active-menu-color; - - .remote-name .remote-name-secondary { - color: @term-white; - } - } - - &:first-child { - border-top: 0; - } - - .remote-status-light { - width: 2em; - margin-top: 0.7em; - margin-right: 0.7em; - font-size: 0.8em; - } - - .remote-name { - flex-grow: 1; - - .remote-name-primary { - font-weight: bold; - max-width: 10em; - margin-right: 1em; - } - - .remote-name-secondary { - color: @disabled-color; - max-width: 14em; - } - } - } - } - - .remote-detail { - padding: 10px; - flex-grow: 1; - - display: flex; - flex-direction: column; - - .settings-field { - margin-top: 0.75em; - } - - * { - flex-shrink: 0; - } - - .detail-subtitle { - margin-bottom: 10px; - margin-top: 10px; - } - - .title { - color: @term-white; - padding: 0.75em 0; - margin-bottom: 0; - border-bottom: 1px solid #777; - } - - .terminal-wrapper { - margin-left: 0; - margin-bottom: 0; - &.has-message { - margin-top: 0; - } - box-shadow: none; - border: 1px solid #777; - border-radius: 0 0 5px 5px; - .xterm-rows { - padding-top: 0.5em; - } - } - - .action-buttons { - display: flex; - flex-direction: row; - gap: 10px; - margin-top: 2px; - } - - .remote-message { - margin-top: 5px; - padding: 8px; - border-radius: 5px 5px 0 0; - background-color: #333; - border: 1px solid #777; - border-bottom: none; - - .message-row { - display: flex; - flex-direction: row; - align-items: center; - svg { - vertical-align: text-bottom; - } - } - - .remote-status { - position: relative; - top: -1px; - } - - .button { - height: 22px; - } - } - - .settings-field { - .update-auth-button { - visibility: hidden; - } - - &:hover { - .update-auth-button { - visibility: visible; - } - - .hide-hover { - display: none; - } - } - } - - &.auth-editing, - &.create-remote { - .settings-field.align-top { - align-items: flex-start; - - .settings-label { - margin-top: 8px; - } - - .settings-input { - align-items: flex-start; - } - } - - .settings-label { - display: flex; - flex-direction: row; - align-items: center; - width: 12em !important; - } - - .settings-field .settings-input .undo-icon { - cursor: pointer; - - margin-left: 5px; - } - - .editremote-dropdown .dropdown-trigger button { - width: 120px; - justify-content: flex-start; - color: @base-color; - border: none; - &:hover { - box-shadow: none; - } - } - - .settings-field .raw-input { - width: 120px; - } - - .settings-input input { - background: rgba(255, 255, 255, 0.8); - width: 250px; - outline: none; - } - - .dropdown .dropdown-item { - padding: 5px 5px 5px 12px; - } - - .dropdown .dropdown-content { - max-width: 10.6em; - } - - .settings-input { - .info-message { - margin-left: 22px; - } - } - - .settings-label { - .info-message { - margin-right: 15px; - } - } - } - } - } - - .terminal-wrapper { - position: relative; - padding: 2px 10px 5px 4px; - margin: 5px 5px 10px 5px; - box-shadow: 0 0 1px 1px rgba(255, 255, 255, 0.3); - &.focus { - box-shadow: 0 0 3px 3px rgba(255, 255, 255, 0.3); - } - - .term-tag { - position: absolute; - top: 0; - right: 0; - background-color: @term-red; - color: @term-white; - z-index: 110; - padding: 4px; - } - } -} - -.dropdown.conn-dropdown { - padding-left: 0; - border-radius: 8px; - background-color: rgba(241, 246, 243, 0.08); - - .conn-dd-trigger { - display: flex; - flex-direction: row; - width: 413px; - padding: 6px 8px 6px 12px; - align-items: center; - height: 42px; - - .lefticon { - margin-right: 8px; - margin-top: 4px; - position: relative; - - .status-icon { - width: 10px; - height: 10px; - stroke-width: 2px; - stroke: @status-outline; - position: absolute; - bottom: 3px; - right: -2px; - } - } - - .dd-control { - display: flex; - padding: 4px; - align-items: center; - - .icon { - height: 16px; - width: 16px; - } - } - - .globe-icon { - width: 16px; - height: 16px; - flex-shrink: 0; - } - - .conntext { - display: flex; - flex-direction: column; - justify-content: center; - align-items: flex-start; - flex: 1 0 0; - - .conntext-solo { - color: @text-primary; - text-overflow: ellipsis; - } - - .conntext-1 { - color: @text-primary; - text-overflow: ellipsis; - } - - .conntext-2 { - color: @text-secondary; - text-overflow: ellipsis; - } - } - } - - .conn-dd-menu { - display: flex; - width: 413px; - padding: 6px; - flex-direction: column; - align-items: flex-start; - border-radius: 8px; - background-color: @dropdown-menu; - - .dropdown-item { - display: flex; - padding: 5px 12px 5px 8px; - align-items: center; - gap: 8px; - align-self: stretch; - border-radius: 6px; - - .status-div { - display: flex; - align-items: center; - justify-content: center; - width: 16px; - height: 16px; - padding: 3px; - - svg.status-icon { - width: 10px; - height: 10px; - } - } - - .add-div { - display: flex; - align-items: center; - justify-content: center; - width: 16px; - height: 16px; - - svg.add-icon { - width: 16px; - height: 16px; - - path { - fill: @text-primary; - } - } - } - - .text-standard { - color: @text-secondary; - } - - .text-caption { - color: @text-caption; - } - - .ellipsis { - text-overflow: ellipsis; - } - - &:hover { - background-color: rgba(241, 246, 243, 0.08); - } - } - } -} diff --git a/src/app/connections_deprecated/connections.tsx b/src/app/connections_deprecated/connections.tsx deleted file mode 100644 index f57484390..000000000 --- a/src/app/connections_deprecated/connections.tsx +++ /dev/null @@ -1,1300 +0,0 @@ -// Copyright 2023, Command Line Inc. -// SPDX-License-Identifier: Apache-2.0 - -import * as React from "react"; -import * as mobxReact from "mobx-react"; -import * as mobx from "mobx"; -import { boundMethod } from "autobind-decorator"; -import { If, For } from "tsx-control-statements/components"; -import cn from "classnames"; -import { GlobalModel, GlobalCommandRunner, RemotesModalModel } from "../../model/model"; -import { Toggle, RemoteStatusLight, InfoMessage } from "../common/common"; -import * as T from "../../types/types"; -import * as util from "../../util/util"; -import * as textmeasure from "../../util/textmeasure"; - -import { ReactComponent as XmarkIcon } from "../assets/icons/line/xmark.svg"; -import { ReactComponent as AngleDownIcon } from "../assets/icons/history/angle-down.svg"; -import { ReactComponent as RotateLeftIcon } from "../assets/icons/rotate_left.svg"; -import { ReactComponent as AddIcon } from "../assets/icons/add.svg"; -import { ReactComponent as GlobeIcon } from "../assets/icons/globe.svg"; -import { ReactComponent as StatusCircleIcon } from "../assets/icons/statuscircle.svg"; -import { ReactComponent as ArrowsUpDownIcon } from "../assets/icons/arrowsupdown.svg"; -import { ReactComponent as CircleIcon } from "../assets/icons/circle.svg"; - -import "./connections.less"; - -type OV = mobx.IObservableValue; -type OArr = mobx.IObservableArray; -type OMap = mobx.ObservableMap; - -const RemotePtyRows = 8; -const RemotePtyCols = 80; -const PasswordUnchangedSentinel = "--unchanged--"; - -function getRemoteCNWithPort(remote: T.RemoteType) { - if (util.isBlank(remote.remotevars.port) || remote.remotevars.port == "22") { - return remote.remotecanonicalname; - } - return remote.remotecanonicalname + ":" + remote.remotevars.port; -} - -function getRemoteTitle(remote: T.RemoteType) { - if (!util.isBlank(remote.remotealias)) { - return remote.remotealias + " (" + remote.remotecanonicalname + ")"; - } - return remote.remotecanonicalname; -} - -@mobxReact.observer -class AuthModeDropdown extends React.Component<{ tempVal: OV }, {}> { - active: OV = mobx.observable.box(false, { name: "AuthModeDropdown-active" }); - - @boundMethod - toggleActive(): void { - mobx.action(() => { - this.active.set(!this.active.get()); - })(); - } - - @boundMethod - updateValue(val: string): void { - mobx.action(() => { - this.props.tempVal.set(val); - this.active.set(false); - })(); - } - - render() { - return ( -
-
- -
-
-
-
this.updateValue("none")} className="dropdown-item"> - none -
-
this.updateValue("key")} className="dropdown-item"> - key -
-
this.updateValue("password")} className="dropdown-item"> - password -
-
this.updateValue("key+password")} - className="dropdown-item" - > - key+password -
-
-
-
- ); - } -} - -@mobxReact.observer -class ConnectModeDropdown extends React.Component<{ tempVal: OV }, {}> { - active: OV = mobx.observable.box(false, { name: "ConnectModeDropdown-active" }); - - @boundMethod - toggleActive(): void { - mobx.action(() => { - this.active.set(!this.active.get()); - })(); - } - - @boundMethod - updateValue(val: string): void { - mobx.action(() => { - this.props.tempVal.set(val); - this.active.set(false); - })(); - } - - render() { - return ( -
-
- -
-
-
-
this.updateValue("startup")} className="dropdown-item"> - startup -
-
this.updateValue("auto")} className="dropdown-item"> - auto -
-
this.updateValue("manual")} className="dropdown-item"> - manual -
-
-
-
- ); - } -} - -@mobxReact.observer -class CreateRemote extends React.Component<{ model: RemotesModalModel; remoteEdit: T.RemoteEditType }, {}> { - tempAlias: OV; - tempHostName: OV; - tempPort: OV; - tempAuthMode: OV; - tempConnectMode: OV; - tempManualMode: OV; - tempPassword: OV; - tempKeyFile: OV; - errorStr: OV; - - constructor(props: any) { - super(props); - let { remoteEdit } = this.props; - this.tempAlias = mobx.observable.box("", { name: "CreateRemote-alias" }); - this.tempHostName = mobx.observable.box("", { name: "CreateRemote-hostName" }); - this.tempPort = mobx.observable.box("", { name: "CreateRemote-port" }); - this.tempAuthMode = mobx.observable.box("none", { name: "CreateRemote-authMode" }); - this.tempConnectMode = mobx.observable.box("auto", { name: "CreateRemote-connectMode" }); - this.tempKeyFile = mobx.observable.box("", { name: "CreateRemote-keystr" }); - this.tempPassword = mobx.observable.box("", { name: "CreateRemote-password" }); - this.errorStr = mobx.observable.box(remoteEdit.errorstr, { name: "CreateRemote-errorStr" }); - } - - remoteCName(): string { - let hostName = this.tempHostName.get(); - if (hostName == "") { - return "[no host]"; - } - if (hostName.indexOf("@") == -1) { - hostName = "[no user]@" + hostName; - } - return hostName; - } - - getErrorStr(): string { - if (this.errorStr.get() != null) { - return this.errorStr.get(); - } - return this.props.remoteEdit.errorstr; - } - - @boundMethod - submitRemote(): void { - mobx.action(() => { - this.errorStr.set(null); - })(); - let authMode = this.tempAuthMode.get(); - let cname = this.tempHostName.get(); - if (cname == "") { - this.errorStr.set("You must specify a 'user@host' value to create a new connection"); - return; - } - let kwargs: Record = {}; - kwargs["alias"] = this.tempAlias.get(); - if (this.tempPort.get() != "" && this.tempPort.get() != "22") { - kwargs["port"] = this.tempPort.get(); - } - if (authMode == "key" || authMode == "key+password") { - if (this.tempKeyFile.get() == "") { - this.errorStr.set("When AuthMode is set to 'key', you must supply a valid key file name."); - return; - } - kwargs["key"] = this.tempKeyFile.get(); - } else { - kwargs["key"] = ""; - } - if (authMode == "password" || authMode == "key+password") { - if (this.tempPassword.get() == "") { - this.errorStr.set("When AuthMode is set to 'password', you must supply a password."); - return; - } - kwargs["password"] = this.tempPassword.get(); - } else { - kwargs["password"] = ""; - } - kwargs["connectmode"] = this.tempConnectMode.get(); - kwargs["visual"] = "1"; - kwargs["submit"] = "1"; - let model = this.props.model; - let shouldCr = model.onlyAddNewRemote.get(); - let prtn = GlobalCommandRunner.createRemote(cname, kwargs, false); - prtn.then((crtn) => { - if (crtn.success) { - if (shouldCr) { - let crRtn = GlobalCommandRunner.screenSetRemote(cname, true, false); - crRtn.then((crcrtn) => { - if (crcrtn.success) { - model.closeModal(); - return; - } - mobx.action(() => { - this.errorStr.set(crcrtn.error); - })(); - }); - } - return; - } - mobx.action(() => { - this.errorStr.set(crtn.error); - })(); - }); - } - - @boundMethod - handleChangeKeyFile(e: any): void { - mobx.action(() => { - this.tempKeyFile.set(e.target.value); - })(); - } - - @boundMethod - handleChangePassword(e: any): void { - mobx.action(() => { - this.tempPassword.set(e.target.value); - })(); - } - - @boundMethod - handleChangeAlias(e: any): void { - mobx.action(() => { - this.tempAlias.set(e.target.value); - })(); - } - - @boundMethod - handleChangePort(e: any): void { - mobx.action(() => { - this.tempPort.set(e.target.value); - })(); - } - - @boundMethod - handleChangeHostName(e: any): void { - mobx.action(() => { - this.tempHostName.set(e.target.value); - })(); - } - - render() { - let { model, remoteEdit } = this.props; - let authMode = this.tempAuthMode.get(); - return ( -
-
Create New Connection
-
-
-
user@host
-
- - (Required) The user and host that you want to connect with. This is in the same format as - you would pass to ssh, e.g. "ubuntu@test.mydomain.com". - -
-
- -
-
-
-
-
Alias
-
- - (Optional) A short alias to use when selecting or displaying this connection. - -
-
- -
-
-
-
-
Port
-
- - (Optional) Defaults to 22. Set if the server you are connecting to listens to a non-standard - SSH port. - -
-
- -
-
-
-
-
Auth Mode
-
- -
    -
  • - none - no authentication, or authentication is already configured in your ssh - config. -
  • -
  • - key - use a private key. -
  • -
  • - password - use a password. -
  • -
  • - key+password - use a key with a passphrase. -
  • -
-
-
-
-
- -
-
-
- -
-
SSH Keyfile
-
- -
-
-
- -
-
- {authMode == "password" ? "SSH Password" : "Key Passphrase"} -
-
- -
-
-
-
-
-
Connect Mode
-
- -
    -
  • - startup - Connect when Wave Terminal starts. -
  • -
  • - auto - Connect when you first run a command using this connection. -
  • -
  • - manual - Connect manually. Note, if your connection requires manual input, - like an OPT code, you must use this setting. -
  • -
-
-
-
-
-
- -
-
-
-
- -
Error: {this.getErrorStr()}
-
-
-
-
-
- Cancel -
-
- Create Remote -
-
-
- ); - } -} - -@mobxReact.observer -class EditRemoteSettings extends React.Component< - { model: RemotesModalModel; remote: T.RemoteType; remoteEdit: T.RemoteEditType }, - {} -> { - tempAlias: OV; - tempAuthMode: OV; - tempConnectMode: OV; - tempManualMode: OV; - tempPassword: OV; - tempKeyFile: OV; - - constructor(props: any) { - super(props); - let { remote, remoteEdit } = this.props; - this.tempAlias = mobx.observable.box(remote.remotealias ?? "", { name: "EditRemoteSettings-alias" }); - this.tempAuthMode = mobx.observable.box(remote.authtype, { name: "EditRemoteSettings-authMode" }); - this.tempConnectMode = mobx.observable.box(remote.connectmode, { name: "EditRemoteSettings-connectMode" }); - this.tempKeyFile = mobx.observable.box(remoteEdit.keystr ?? "", { name: "EditRemoteSettings-keystr" }); - this.tempPassword = mobx.observable.box(remoteEdit.haspassword ? PasswordUnchangedSentinel : "", { - name: "EditRemoteSettings-password", - }); - } - - componentDidUpdate() { - let { remote } = this.props; - if (remote == null || remote.archived) { - this.props.model.deSelectRemote(); - } - } - - @boundMethod - clickArchive(): void { - let { remote } = this.props; - if (remote.status == "connected") { - GlobalModel.showAlert({ message: "Cannot archived a connected remote. Disconnect and try again." }); - return; - } - let prtn = GlobalModel.showAlert({ - message: "Are you sure you want to archive this connection?", - confirm: true, - }); - prtn.then((confirm) => { - if (!confirm) { - return; - } - GlobalCommandRunner.archiveRemote(remote.remoteid); - }); - } - - @boundMethod - clickForceInstall(): void { - let { remote } = this.props; - GlobalCommandRunner.installRemote(remote.remoteid); - } - - @boundMethod - handleChangeKeyFile(e: any): void { - mobx.action(() => { - this.tempKeyFile.set(e.target.value); - })(); - } - - @boundMethod - handleChangePassword(e: any): void { - mobx.action(() => { - this.tempPassword.set(e.target.value); - })(); - } - - @boundMethod - handleChangeAlias(e: any): void { - mobx.action(() => { - this.tempAlias.set(e.target.value); - })(); - } - - @boundMethod - canResetPw(): boolean { - let { remoteEdit } = this.props; - if (remoteEdit == null) { - return false; - } - return remoteEdit.haspassword && this.tempPassword.get() != PasswordUnchangedSentinel; - } - - @boundMethod - resetPw(): void { - mobx.action(() => { - this.tempPassword.set(PasswordUnchangedSentinel); - })(); - } - - @boundMethod - onFocusPassword(e: any) { - if (this.tempPassword.get() == PasswordUnchangedSentinel) { - e.target.select(); - } - } - - @boundMethod - submitRemote(): void { - let { remote, remoteEdit } = this.props; - let authMode = this.tempAuthMode.get(); - let kwargs: Record = {}; - if (!util.isStrEq(this.tempKeyFile.get(), remoteEdit.keystr)) { - if (authMode == "key" || authMode == "key+password") { - kwargs["key"] = this.tempKeyFile.get(); - } else { - kwargs["key"] = ""; - } - } - if (authMode == "password" || authMode == "key+password") { - if (this.tempPassword.get() != PasswordUnchangedSentinel) { - kwargs["password"] = this.tempPassword.get(); - } - } else { - if (remoteEdit.haspassword) { - kwargs["password"] = ""; - } - } - if (!util.isStrEq(this.tempAlias.get(), remote.remotealias)) { - kwargs["alias"] = this.tempAlias.get(); - } - if (!util.isStrEq(this.tempConnectMode.get(), remote.connectmode)) { - kwargs["connectmode"] = this.tempConnectMode.get(); - } - if (Object.keys(kwargs).length == 0) { - return; - } - kwargs["visual"] = "1"; - kwargs["submit"] = "1"; - GlobalCommandRunner.editRemote(remote.remoteid, kwargs); - } - - renderAuthModeMessage(): any { - let authMode = this.tempAuthMode.get(); - if (authMode == "none") { - return ( - - This connection requires no authentication. -
- Or authentication is already configured in ssh_config. -
- ); - } - if (authMode == "key") { - return Use a public/private keypair.; - } - if (authMode == "password") { - return Use a password.; - } - if (authMode == "key+password") { - return Use a public/private keypair with a passphrase.; - } - return null; - } - - render() { - let { model, remote, remoteEdit } = this.props; - let authMode = this.tempAuthMode.get(); - return ( -
-
{getRemoteTitle(remote)}
-
Editing Connection Settings
-
-
-
Alias
-
- - (Optional) A short alias to use when selecting or displaying this connection. - -
-
- -
-
-
-
-
Auth Mode
-
- -
    -
  • - none - no authentication, or authentication is already configured in your ssh - config. -
  • -
  • - key - use a private key. -
  • -
  • - password - use a password. -
  • -
  • - key+password - use a key with a passphrase. -
  • -
-
-
-
-
- -
-
-
- -
-
SSH Keyfile
-
- -
-
-
- -
-
- {authMode == "password" ? "SSH Password" : "Key Passphrase"} -
-
- - -
- -
-
-
-
-
-
-
-
Connect Mode
-
- -
    -
  • - startup - Connect when Wave Terminal starts. -
  • -
  • - auto - Connect when you first run a command using this connection. -
  • -
  • - manual - Connect manually. Note, if your connection requires manual input, - like an OPT code, you must use this setting. -
  • -
-
-
-
-
-
- -
-
-
-
-
-
Actions
-
-
- Archive Connection -
-
- Force Install -
-
-
- -
Error: {remoteEdit.errorstr ?? "An error occured"}
-
-
-
-
-
- Cancel -
-
- Submit -
-
-
- ); - } -} - -@mobxReact.observer -class RemoteDetailView extends React.Component<{ model: RemotesModalModel; remote: T.RemoteType }, {}> { - termRef: React.RefObject = React.createRef(); - - componentDidMount() { - let elem = this.termRef.current; - if (elem == null) { - console.log("ERROR null term-remote element"); - return; - } - this.props.model.createTermWrap(elem); - } - - componentDidUpdate() { - let { remote } = this.props; - if (remote == null || remote.archived) { - this.props.model.deSelectRemote(); - } - } - - componentWillUnmount() { - this.props.model.disposeTerm(); - } - - @boundMethod - clickTermBlock(): void { - if (this.props.model.remoteTermWrap != null) { - this.props.model.remoteTermWrap.giveFocus(); - } - } - - getRemoteTypeStr(remote: T.RemoteType): string { - if (!util.isBlank(remote.uname)) { - let unameStr = remote.uname; - unameStr = unameStr.replace("|", ", "); - return remote.remotetype + " (" + unameStr + ")"; - } - return remote.remotetype; - } - - @boundMethod - connectRemote(remoteId: string) { - GlobalCommandRunner.connectRemote(remoteId); - } - - @boundMethod - disconnectRemote(remoteId: string) { - GlobalCommandRunner.disconnectRemote(remoteId); - } - - @boundMethod - installRemote(remoteId: string) { - GlobalCommandRunner.installRemote(remoteId); - } - - @boundMethod - cancelInstall(remoteId: string) { - GlobalCommandRunner.installCancelRemote(remoteId); - } - - @boundMethod - editAuthSettings(): void { - this.props.model.startEditAuth(); - } - - renderInstallStatus(remote: T.RemoteType): any { - let statusStr: string = null; - if (remote.installstatus == "disconnected") { - if (remote.needsmshellupgrade) { - statusStr = "mshell " + remote.mshellversion + " - needs upgrade"; - } else if (util.isBlank(remote.mshellversion)) { - statusStr = "mshell unknown"; - } else { - statusStr = "mshell " + remote.mshellversion + " - current"; - } - } else { - statusStr = remote.installstatus; - } - if (statusStr == null) { - return null; - } - return ( -
-
Install Status
-
{statusStr}
-
- ); - } - - renderRemoteMessage(remote: T.RemoteType): any { - let message: string = ""; - let buttons: any[] = []; - // connect, disconnect, editauth, tryreconnect, install - - let disconnectButton = ( -
this.disconnectRemote(remote.remoteid)} - className="button is-prompt-danger is-outlined is-small" - > - Disconnect Now -
- ); - let connectButton = ( -
this.connectRemote(remote.remoteid)} - className="button is-wave-green is-outlined is-small" - > - Connect Now -
- ); - let tryReconnectButton = ( -
this.connectRemote(remote.remoteid)} - className="button is-wave-green is-outlined is-small" - > - Try Reconnect -
- ); - let updateAuthButton = ( -
this.editAuthSettings()} - className="button is-plain is-outlined is-small" - > - Update Auth Settings -
- ); - let cancelInstallButton = ( -
this.cancelInstall(remote.remoteid)} - className="button is-prompt-danger is-outlined is-small" - > - Cancel Install -
- ); - let installNowButton = ( -
this.installRemote(remote.remoteid)} - className="button is-wave-green is-outlined is-small" - > - Install Now -
- ); - if (remote.local) { - installNowButton = null; - updateAuthButton = null; - cancelInstallButton = null; - } - if (remote.status == "connected") { - message = "Connected and ready to run commands."; - buttons = [disconnectButton]; - } else if (remote.status == "connecting") { - message = remote.waitingforpassword ? "Connecting, waiting for user-input..." : "Connecting..."; - let connectTimeout = remote.connecttimeout ?? 0; - message = message + " (" + connectTimeout + "s)"; - buttons = [disconnectButton]; - } else if (remote.status == "disconnected") { - message = "Disconnected"; - buttons = [connectButton]; - } else if (remote.status == "error") { - if (remote.noinitpk) { - message = "Error, could not connect."; - buttons = [tryReconnectButton, updateAuthButton]; - } else if (remote.needsmshellupgrade) { - if (remote.installstatus == "connecting") { - message = "Installing..."; - buttons = [cancelInstallButton]; - } else { - message = "Error, needs install."; - buttons = [installNowButton, updateAuthButton]; - } - } else { - message = "Error"; - buttons = [tryReconnectButton, updateAuthButton]; - } - } - let button: any = null; - return ( -
-
-
- {message} -
-
- - {button} - -
-
- ); - } - - render() { - let { model, remote } = this.props; - let isTermFocused = model.remoteTermWrapFocus.get(); - let termFontSize = GlobalModel.termFontSize.get(); - let remoteMessage = this.renderRemoteMessage(remote); - let termWidth = textmeasure.termWidthFromCols(RemotePtyCols, termFontSize); - let remoteAliasText = util.isBlank(remote.remotealias) ? "(none)" : remote.remotealias; - return ( -
-
{getRemoteTitle(remote)}
-
-
Conn Id
-
{remote.remoteid}
-
-
-
Type
-
{this.getRemoteTypeStr(remote)}
-
-
-
Canonical Name
-
- {remote.remotecanonicalname} - - (port {remote.remotevars.port}) - -
-
-
-
Alias
-
{remoteAliasText}
-
-
-
Auth Type
-
- {remote.authtype} - local -
-
-
-
Connect Mode
-
{remote.connectmode}
-
- {this.renderInstallStatus(remote)} -
-
Actions
-
-
this.editAuthSettings()} - className="button is-wave-green is-outlined is-small is-inline-height" - > - Edit Connection Settings -
-
-
-
-
{remoteMessage}
-
- -
-
- -
- input is only allowed while status is 'connecting' -
-
-
-
-
- ); - } -} - -@mobxReact.observer -class RemotesModal extends React.Component<{ model: RemotesModalModel }, {}> { - @boundMethod - closeModal(): void { - this.props.model.closeModal(); - } - - @boundMethod - selectRemote(remoteId: string): void { - let model = this.props.model; - model.selectRemote(remoteId); - } - - @boundMethod - clickAddRemote(): void { - GlobalCommandRunner.openCreateRemote(); - } - - renderRemoteMenuItem(remote: T.RemoteType, selectedId: string): any { - return ( -
this.selectRemote(remote.remoteid)} - className={cn("remote-menu-item", { "is-selected": remote.remoteid == selectedId })} - > -
- -
- -
-
{remote.remotecanonicalname}
-
-
- -
-
{remote.remotealias}
-
{remote.remotecanonicalname}
-
-
-
- ); - } - - renderAddRemoteMenuItem(): any { - return ( -
-
- Add SSH Connection -
-
- ); - } - - renderEmptyDetail(): any { - return ( -
-
No Connection Selected
-
- ); - } - - render() { - let model = this.props.model; - let selectedRemoteId = model.selectedRemoteId.get(); - let allRemotes = util.sortAndFilterRemotes(GlobalModel.remotes.slice()); - let remote: T.RemoteType = null; - let isAuthEditMode = model.isAuthEditMode(); - let selectedRemote = GlobalModel.getRemote(selectedRemoteId); - let remoteEdit = model.remoteEdit.get(); - let onlyAddNewRemote = model.onlyAddNewRemote.get(); - - // @TODO: this is a hack to determine which create modal to show - if (remoteEdit && !remoteEdit.old) { - return null; - } - - return ( -
-
-
-
-
Connections
-
- -
-
-
- -
- {this.renderAddRemoteMenuItem()} - - {this.renderRemoteMenuItem(remote, selectedRemoteId)} - -
{" "} -
- - - - - {this.renderEmptyDetail()} - - - - - - - - - -
-
-
- ); - } -} - -@mobxReact.observer -class ConnectionDropdown extends React.Component< - { - curRemote: T.RemoteType; - onSelectRemote?: (cname: string) => void; - allowNewConn: boolean; - onNewConn?: () => void; - }, - {} -> { - connDropdownActive: OV = mobx.observable.box(false, { name: "connDropdownActive" }); - - @boundMethod - toggleConnDropdown(): void { - mobx.action(() => { - this.connDropdownActive.set(!this.connDropdownActive.get()); - })(); - } - - @boundMethod - selectRemote(cname: string): void { - mobx.action(() => { - this.connDropdownActive.set(false); - })(); - if (this.props.onSelectRemote) { - this.props.onSelectRemote(cname); - } - } - - @boundMethod - clickNewConnection(): void { - mobx.action(() => { - this.connDropdownActive.set(false); - })(); - if (this.props.onNewConn) { - this.props.onNewConn(); - } - } - - render() { - let { curRemote } = this.props; - let remote: T.RemoteType = null; - let allRemotes = util.sortAndFilterRemotes(GlobalModel.remotes.slice()); - return ( -
-
-
- -
- - -
-
- -
{curRemote.remotecanonicalname}
-
- -
{curRemote.remotealias}
-
{curRemote.remotecanonicalname}
-
-
-
- -
-
- -
- -
-
-
(no connection)
-
-
- -
-
-
-
-
-
- -
this.selectRemote(remote.remotecanonicalname)} - > -
- -
- -
{remote.remotecanonicalname}
-
- -
{remote.remotealias}
-
{remote.remotecanonicalname}
-
-
-
- -
-
- -
-
New Connection
-
-
-
-
-
- ); - } -} - -export { RemotesModal, ConnectionDropdown };