About Modal (#39)

* menu item and send event to renderer

* wire up everything

* update content

* modal content

* add mising text

* minor fix

* style fixes

* figma design init

* about header, content, and general styles

* css refactor

* more on refactor

* reuse styles from app.less in about modal

* dynamic status

* external linkn

* finish about modal

* update links

* update license link
This commit is contained in:
Red J Adaya 2023-11-01 13:24:36 +08:00 committed by GitHub
parent 8211a0bb26
commit 15a4719b17
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 415 additions and 55 deletions

View File

@ -24,7 +24,7 @@ import {
import { RemotesModal } from "./connections/connections";
import { TosModal } from "./common/modals/modals";
import { MainSideBar } from "./sidebar/MainSideBar";
import { DisconnectedModal, ClientStopModal, AlertModal, WelcomeModal } from "./common/modals/modals";
import { DisconnectedModal, ClientStopModal, AlertModal, WelcomeModal, AboutModal } from "./common/modals/modals";
import { ErrorBoundary } from "./common/error/errorboundary";
import "./app.less";
@ -127,6 +127,9 @@ class App extends React.Component<{}, {}> {
<If condition={GlobalModel.welcomeModalOpen.get()}>
<WelcomeModal />
</If>
<If condition={GlobalModel.aboutModalOpen.get()}>
<AboutModal />
</If>
<If condition={screenSettingsModal != null}>
<ScreenSettingsModal
key={screenSettingsModal.sessionId + ":" + screenSettingsModal.screenId}

View File

@ -0,0 +1,5 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="cancel">
<path id="Vector" d="M13.0261 2.96991C13.319 3.26281 13.319 3.73768 13.0261 4.03057L9.05864 7.998L13.0261 11.9654C13.3189 12.2583 13.3189 12.7332 13.0261 13.0261C12.7332 13.319 12.2583 13.319 11.9654 13.0261L7.99798 9.05866L4.03059 13.0261C3.73769 13.3189 3.26282 13.3189 2.96993 13.0261C2.67703 12.7332 2.67703 12.2583 2.96993 11.9654L6.93732 7.998L2.96991 4.03059C2.67702 3.7377 2.67702 3.26282 2.96991 2.96993C3.26281 2.67704 3.73768 2.67704 4.03057 2.96993L7.99798 6.93734L11.9654 2.96991C12.2583 2.67702 12.7332 2.67702 13.0261 2.96991Z" fill="#C3C8C2"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 684 B

View File

@ -0,0 +1,29 @@
<svg width="74" height="73" viewBox="0 0 74 73" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="Icon Center Image Artwork">
<g id="Guide - Hide">
<rect x="1" width="44.7379" height="44.7379" fill="#EA33EC" fill-opacity="0.1"/>
<rect x="3.18447" y="2.18447" width="40.3689" height="40.3689" stroke="#EA33EC" stroke-opacity="0.09" stroke-width="4.36893"/>
</g>
<g id="wave-logo_appicon 1">
<path id="Vector" d="M73.8156 -0.808594H0.191406V72.8156H73.8156V-0.808594Z" fill="black"/>
<g id="Group">
<g id="Group_2">
<path id="Vector_2" d="M27.2969 34.6218C24.7516 34.6218 23.3523 36.2741 22.5859 39.9726L11.125 38.3132C12.5242 26.2194 17.2352 20.3624 26.5305 20.3624C33.407 20.3624 40.157 25.5866 43.4688 25.5866C46.0141 25.5866 47.4133 23.8007 48.1797 20.2358L59.6406 21.8882C58.368 33.9819 53.5305 39.846 44.2352 39.846C37.225 39.846 30.7352 34.6218 27.2969 34.6218Z" fill="url(#paint0_linear_1373_32879)"/>
</g>
<g id="Group_3">
<path id="Vector_3" d="M30.5312 46.4133C27.9859 46.4133 26.5867 48.0656 25.8203 51.7641L14.3594 50.1047C15.7586 38.0109 20.4695 32.1539 29.7648 32.1539C36.6414 32.1539 43.3914 37.3781 46.7031 37.3781C49.2484 37.3781 50.6477 35.5922 51.4141 32.0273L62.875 33.6797C61.6023 45.7734 56.7648 51.6375 47.4695 51.6375C40.4664 51.6375 33.9695 46.4133 30.5312 46.4133Z" fill="url(#paint1_linear_1373_32879)"/>
</g>
</g>
</g>
</g>
<defs>
<linearGradient id="paint0_linear_1373_32879" x1="11.1241" y1="30.1035" x2="59.6372" y2="30.1035" gradientUnits="userSpaceOnUse">
<stop offset="0.1418" stop-color="#1F4D22"/>
<stop offset="0.8656" stop-color="#418D31"/>
</linearGradient>
<linearGradient id="paint1_linear_1373_32879" x1="14.3628" y1="41.8964" x2="62.8759" y2="41.8964" gradientUnits="userSpaceOnUse">
<stop offset="0.2223" stop-color="#418D31"/>
<stop offset="0.7733" stop-color="#58C142"/>
</linearGradient>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 1.8 KiB

View File

@ -258,14 +258,12 @@
}
.button.is-wave-green {
margin-top: 16px;
display: flex;
padding: 6px 16px !important;
color: @term-bright-white !important;
align-items: center;
gap: 4px;
border-radius: 6px !important;
font-size: 14px !important;
height: auto !important;
background: @term-green !important;
box-shadow: 0px 1px 3px 0px rgba(0, 0, 0, 0.4), 0px 0px 0.5px 0px rgba(0, 0, 0, 0.5),

View File

@ -184,10 +184,9 @@
.modal.wave-modal {
.modal-content {
display: flex;
padding: 32px 48px;
flex-direction: column;
align-items: flex-start;
gap: 8px;
gap: 16px;
border-radius: 10px;
background: var(--olive-dark-1, #151715);
box-shadow: 0px 3px 5px 0px rgba(0, 0, 0, 0.35), 0px 10px 24px 0px rgba(0, 0, 0, 0.45),
@ -201,28 +200,26 @@
width: 100%;
header {
display: flex;
flex-direction: column;
align-items: center;
gap: var(--sizing-sm, 12px);
width: 100%;
display: flex;
align-items: center;
padding: 12px 20px;
justify-content: space-between;
line-height: 20px;
border-bottom: 1px solid rgba(250, 250, 250, 0.1);
.modal-title {
color: @term-bright-white;
text-align: center;
font-size: 20px;
font-style: normal;
font-weight: 300;
line-height: 20px;
font-size: 13px;
}
.modal-subtitle {
color: @term-white;
text-align: center;
font-style: normal;
font-weight: 300;
line-height: 20px;
.close-icon-wrapper {
display: flex;
padding: 4px;
align-items: center;
gap: 8px;
}
}
@ -230,40 +227,77 @@
display: flex;
flex-direction: column;
align-items: flex-start;
gap: 32px;
gap: 24px;
width: 87%;
}
}
}
}
.modal.tos-modal {
.modal-content.tos-modal-content {
padding: 32px 48px;
gap: 8px;
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%;
margin-bottom: 0;
align-items: center;
gap: 16px;
.item {
.item-inner {
display: flex;
width: 100%;
align-items: center;
gap: 16px;
flex-direction: column;
align-items: flex-start;
gap: 4px;
flex: 1 0 0;
.item-inner {
.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;
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;
}
align-items: center;
gap: 8px;
}
}
}
@ -276,6 +310,11 @@
align-items: center;
justify-content: center;
button {
font-size: 12.5px !important;
margin-top: 16px;
}
button.disabled-button {
cursor: default;
}
@ -284,6 +323,156 @@
}
}
.modal.about-modal {
.modal-content.about-modal-content {
width: 401px;
.about-content {
margin-bottom: 0;
section {
.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;
}
}
}
section:nth-child(3) {
display: flex;
align-items: flex-start;
gap: 10px;
.button-link {
display: flex;
align-items: center;
i {
font-size: 16px;
}
}
}
section:last-child {
margin-bottom: 24px;
color: @term-white;
}
}
}
}
.button {
display: flex;
padding: 6px 16px;
align-items: center;
gap: var(--sizing-2-xs, 4px);
border-radius: 6px;
height: auto;
}
.button.color-green {
color: @term-bright-white;
background: @term-green !important; // !important is needed to override the default button color
box-shadow: 0px 1px 3px 0px rgba(0, 0, 0, 0.4), 0px 0px 0.5px 0px rgba(0, 0, 0, 0.5),
0px 0px 0.5px 0px rgba(255, 255, 255, 0.8) inset, 0px 0.5px 0px 0px rgba(255, 255, 255, 0.6) inset;
&:hover {
color: @term-bright-white;
}
}
.button.color-standard {
color: @term-white;
background: var(--overlays-white-6, rgba(255, 255, 255, 0.12));
&:hover {
color: @term-white;
}
}
.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;
}
section {
display: flex;
align-items: center;
gap: 16px;
align-self: stretch;
width: 100%;
}
.modal.welcome-modal {
footer {
.prev-button {

View File

@ -15,14 +15,18 @@ import * as util from "../../../util/util";
import { Toggle, Checkbox } from "../common";
import { ClientDataType } from "../../../types/types";
import { ReactComponent as XmarkIcon } from "../../assets/icons/line/xmark.svg";
import close from "../../assets/icons/close.svg";
import { ReactComponent as WarningIcon } from "../../assets/icons/line/triangle-exclamation.svg";
import shield from "../../assets/icons/shield_check.svg";
import help from "../../assets/icons/help_filled.svg";
import github from "../../assets/icons/github.svg";
import logo from "../../assets/waveterm-logo-with-bg.svg";
dayjs.extend(localizedFormat);
// @ts-ignore
const VERSION = __PROMPT_VERSION__;
type OV<V> = mobx.IObservableValue<V>;
@mobxReact.observer
@ -269,7 +273,7 @@ class WelcomeModal extends React.Component<{}, {}> {
<div className="modal-background" onClick={this.closeModal} />
<div className="modal-content">
<header>
<div className="modal-title">welcome to [prompt]</div>
<div className="modal-title">About</div>
<div className="close-icon hoverEffect" title="Close (Escape)" onClick={this.closeModal}>
<XmarkIcon />
</div>
@ -346,13 +350,13 @@ class TosModal extends React.Component<{}, {}> {
return (
<div className={cn("modal tos-modal wave-modal is-active")}>
<div className="modal-background" />
<div className="modal-content">
<div className="modal-content tos-modal-content">
<div className="modal-content-wrapper">
<header>
<header className="tos-header">
<div className="modal-title">Welcome to Wave Terminal!</div>
<div className="modal-subtitle">Lets set everything for you</div>
</header>
<div className="content">
<div className="content tos-content">
<div className="item">
<img src={shield} alt="Privacy" />
<div className="item-inner">
@ -429,4 +433,112 @@ class TosModal extends React.Component<{}, {}> {
}
}
export { WelcomeModal, LoadingSpinner, ClientStopModal, AlertModal, DisconnectedModal, TosModal };
@mobxReact.observer
class AboutModal extends React.Component<{}, {}> {
@boundMethod
closeModal(): void {
mobx.action(() => {
GlobalModel.aboutModalOpen.set(false);
})();
}
@boundMethod
isUpToDate(): boolean {
return true;
}
@boundMethod
updateApp(): void {
// GlobalCommandRunner.updateApp();
}
@boundMethod
getStatus(isUpToDate: boolean): JSX.Element {
if (isUpToDate) {
return (
<div className="status updated">
<div>
<i className="fa-sharp fa-solid fa-circle-check" />
<span>Up to Date</span>
</div>
<div>Client Version v0.4.0 20231016-110014</div>
</div>
);
}
return (
<div className="status outdated">
<div>
<i className="fa-sharp fa-solid fa-triangle-exclamation" />
<span>Outdated Version</span>
</div>
<div>Client Version v0.4.0 20231016-110014</div>
<div>
<button onClick={this.updateApp} className="button color-green text-secondary">
Update
</button>
</div>
</div>
);
}
render() {
return (
<div className={cn("modal about-modal wave-modal is-active")}>
<div className="modal-background" />
<div className="modal-content about-modal-content">
<div className="modal-content-wrapper">
<header className="common-header">
<div className="modal-title">About</div>
<div className="close-icon-wrapper" onClick={this.closeModal}>
<img src={close} alt="Close (Escape)" />
</div>
</header>
<div className="content about-content">
<section>
<div className="logo-wrapper">
<img src={logo} alt="logo" />
</div>
<div className="text-wrapper">
<div>Wave Terminal</div>
<div className="text-standard">Modern Terminal for Seamless Workflow</div>
</div>
</section>
<section className="text-standard">{this.getStatus(this.isUpToDate())}</section>
<section>
<a
className="button button-link color-standard"
href={util.makeExternLink("https://github.com/wavetermdev/waveterm")}
target="_blank"
>
<i className="fa-brands fa-github"></i>
Github
</a>
<a
className="button button-link color-standard"
href={util.makeExternLink("https://www.commandline.dev/")}
target="_blank"
>
<i className="fa-sharp fa-light fa-globe"></i>
Website
</a>
<a
className="button button-link color-standard"
href={util.makeExternLink(
"https://github.com/wavetermdev/waveterm/blob/main/LICENSE"
)}
target="_blank"
>
<i className="fa-sharp fa-light fa-book-blank"></i>
License
</a>
</section>
<section className="text-standard">Copyright © 2023 Command Line Inc.</section>
</div>
</div>
</div>
</div>
);
}
}
export { WelcomeModal, LoadingSpinner, ClientStopModal, AlertModal, DisconnectedModal, TosModal, AboutModal };

View File

@ -146,6 +146,14 @@ function readAuthKey() {
let menuTemplate = [
{
role: "appMenu",
submenu: [
{
label: 'About Wave Terminal',
click: () => {
MainWindow?.webContents.send('menu-item-about');
}
}
],
},
{
label: "File",
@ -165,7 +173,7 @@ let menuTemplate = [
let menu = electron.Menu.buildFromTemplate(menuTemplate);
electron.Menu.setApplicationMenu(menu);
let MainWindow = null;
let MainWindow: Electron.BrowserWindow | null = null;
function getMods(input: any) {
return { meta: input.meta, shift: input.shift, ctrl: input.control, alt: input.alt };

View File

@ -19,6 +19,7 @@ contextBridge.exposeInMainWorld("api", {
onMetaPageDown: (callback) => ipcRenderer.on("meta-pagedown", callback),
onBracketCmd: (callback) => ipcRenderer.on("bracket-cmd", callback),
onDigitCmd: (callback) => ipcRenderer.on("digit-cmd", callback),
onMenuItemAbout: (callback) => ipcRenderer.on("menu-item-about", callback),
contextScreen: (screenOpts, position) => ipcRenderer.send("context-screen", screenOpts, position),
contextEditMenu: (position, opts) => ipcRenderer.send("context-editmenu", position, opts),
onWaveSrvStatusChange: (callback) => ipcRenderer.on("wavesrv-status-change", callback),

View File

@ -180,6 +180,7 @@ type ElectronApi = {
onICmd: (callback: (mods: KeyModsType) => void) => void;
onLCmd: (callback: (mods: KeyModsType) => void) => void;
onHCmd: (callback: (mods: KeyModsType) => void) => void;
onMenuItemAbout: (callback: () => void) => void;
onMetaArrowUp: (callback: () => void) => void;
onMetaArrowDown: (callback: () => void) => void;
onMetaPageUp: (callback: () => void) => void;
@ -2699,6 +2700,9 @@ class Model {
welcomeModalOpen: OV<boolean> = mobx.observable.box(false, {
name: "welcomeModalOpen",
});
aboutModalOpen: OV<boolean> = mobx.observable.box(false, {
name: "aboutModalOpen",
});
screenSettingsModal: OV<{ sessionId: string; screenId: string }> = mobx.observable.box(null, {
name: "screenSettingsModal",
});
@ -2759,6 +2763,7 @@ class Model {
getApi().onICmd(this.onICmd.bind(this));
getApi().onLCmd(this.onLCmd.bind(this));
getApi().onHCmd(this.onHCmd.bind(this));
getApi().onMenuItemAbout(this.onMenuItemAbout.bind(this));
getApi().onMetaArrowUp(this.onMetaArrowUp.bind(this));
getApi().onMetaArrowDown(this.onMetaArrowDown.bind(this));
getApi().onMetaPageUp(this.onMetaPageUp.bind(this));
@ -2977,6 +2982,10 @@ class Model {
GlobalModel.welcomeModalOpen.set(false);
didSomething = true;
}
if (GlobalModel.welcomeModalOpen.get()) {
GlobalModel.welcomeModalOpen.set(false);
didSomething = true;
}
})();
return didSomething;
}
@ -3106,6 +3115,12 @@ class Model {
}
}
onMenuItemAbout(): void {
mobx.action(() => {
this.aboutModalOpen.set(true);
})();
}
onMetaPageUp(): void {
GlobalCommandRunner.screenSelectLine("-1");
}