diff --git a/src/frontend/src/components/authenticateBox.ts b/src/frontend/src/components/authenticateBox.ts index f1e87685cd..a9328bb082 100644 --- a/src/frontend/src/components/authenticateBox.ts +++ b/src/frontend/src/components/authenticateBox.ts @@ -48,6 +48,7 @@ import { promptUserNumber } from "./promptUserNumber"; import { DerEncodedPublicKey } from "@dfinity/agent"; import copyJson from "./authenticateBox.json"; +import { landingPage } from "./landingPage"; /** Template used for rendering specific authentication screens. See `authnScreens` below * for meaning of "firstTime", "useExisting" and "pick". */ @@ -562,12 +563,22 @@ export const authnPages = (i18n: I18n, props: AuthnTemplates) => { const templates = authnTemplates(i18n, props); return { - firstTime: (opts: Parameters[0]) => - page(templates.firstTime(opts)), + firstTime: ( + opts: Parameters[0], + // TODO: remove this parameter once the landing page is ready. + landingPageTemplate: boolean = false + ) => + // TODO: Use the landing page template always when ready. + page(templates.firstTime(opts), landingPageTemplate), useExisting: (opts: Parameters[0]) => page(templates.useExisting(opts)), - pick: (opts: Parameters[0]) => - page(templates.pick(opts)), + pick: ( + opts: Parameters[0], + // TODO: remove this parameter once the landing page is ready. + landingPageTemplate: boolean = false + ) => + // TODO: Use the landing page template always when ready. + page(templates.pick(opts), landingPageTemplate), }; }; @@ -615,12 +626,16 @@ export const authnScreens = (i18n: I18n, props: AuthnTemplates) => { }; // Wrap the template with header & footer and render the page -const page = (slot: TemplateResult) => { - const template = mainWindow({ - slot: html` -

Internet Identity

- ${slot}`, - }); +const page = (slot: TemplateResult, landingPageTemplate: boolean = false) => { + const template = landingPageTemplate + ? landingPage({ + slot, + }) + : mainWindow({ + slot: html` +

Internet Identity

+ ${slot}`, + }); const container = document.getElementById("pageContent") as HTMLElement; render(template, container); }; diff --git a/src/frontend/src/components/landingPage.json b/src/frontend/src/components/landingPage.json new file mode 100644 index 0000000000..4dc859738c --- /dev/null +++ b/src/frontend/src/components/landingPage.json @@ -0,0 +1,6 @@ +{ + "en": { + "title": "Secure, seamless & privacy-preserving digital identity", + "subtitle": "Internet Identity is a decentralized identity solution running end-to-end on the Internet Computer blockchain." + } +} diff --git a/src/frontend/src/components/landingPage.ts b/src/frontend/src/components/landingPage.ts new file mode 100644 index 0000000000..aafaa03d94 --- /dev/null +++ b/src/frontend/src/components/landingPage.ts @@ -0,0 +1,58 @@ +import { I18n } from "$src/i18n"; +import { html, TemplateResult } from "lit-html"; +import { navigationLink } from "./footer"; +import { githubIcon, icLogo, questionIcon } from "./icons"; +import copyJson from "./landingPage.json"; + +/** + * Landing page template + * + * It is a component with split panes left and right in desktop view. + * To the left there is some static content, and to the right there is a slot. + * + * In movile view, the static content is below the slot. + * + */ +export const landingPage = ({ + slot, +}: { + slot: TemplateResult; +}): TemplateResult => { + const i18n = new I18n(); + const copy = i18n.i18n(copyJson); + + return html`
+ +
+
+
${slot}
+
+
+
+
+

${copy.title}

+

{${copy.subtitle}}

+
+ +
+
`; +}; diff --git a/src/frontend/src/styles/main.css b/src/frontend/src/styles/main.css index fc9f11048a..07d6bac4ec 100644 --- a/src/frontend/src/styles/main.css +++ b/src/frontend/src/styles/main.css @@ -131,6 +131,11 @@ --vc-brand-alt: rgba(215, 205, 235, 1); --vc-grey: grey; --vc-light-grey: #d4d8e5; + --vc-brand-gradient: linear-gradient( + 90deg, + var(--vc-brand-purple), + var(--vc-brand-purple-light) + ); /** * value tokens: sizes (--vs => valueSize) @@ -150,6 +155,8 @@ --vs-border-radius: 0.8rem; + --vs-screen-width: 40rem; + /* * reference tokens * @@ -203,6 +210,9 @@ --rc-button--disabled: var(--vc-shadow); + --rc-background--highlight: var(--vc-brand-gradient); + --rc-background--highlight-text: var(--vc-snow); + /* color used to elevate something from its background secondary button, card, icon button @@ -254,6 +264,8 @@ --rg-brand-bruised: var(--vc-brand-blue) 50%, var(--vc-brand-purple) 90%; /* reference tokens: sizes */ + --rs-main-content-width: var(--vs-screen-width); + --rs-inline-grid-gap: var(--vs-inline); --rs-inline-icon-gap: var(--vs-inline); @@ -313,6 +325,15 @@ --rs-line--thick: calc(var(--rs-line) * 4); --rs-footer-height: 4rem; + + --rs-split-page-padding--mobile: calc(var(--vs-stack) * 3); + --rs-split-page-left-pane-padding: calc(var(--vs-gutter) * 8); + --rs-split-page-right-pane-padding--desktop: 10vmax; + --rs-split-page-right-pane-padding--mobile: calc(var(--vs-stack) * 2); + --rs-split-page-stack: calc(var(--vs-stack) * 4); + + /* This is not used in CSS, but it's used to compensate for the logo size. */ + --rs-logo-height: 3rem; } /* II) Optional Theme Settings: DarkMode, LightMode, Product Specific stuff */ @@ -755,8 +776,8 @@ a:hover, .l-container { position: relative; font-size: 1.6rem; - min-width: 40rem; - max-width: 40rem; + min-width: var(--rs-main-content-width); + max-width: var(--rs-main-content-width); /* centers the container and adds a bit of space to make sure the footer does not stick to it */ margin: 0 auto 2rem; } @@ -949,6 +970,159 @@ a:hover, color: rgba(0, 0, 0, 0.5); } +/** + * LandingPage component + */ + +.c-landingPage { + display: grid; + grid-template-columns: 1fr minmax(var(--rs-main-content-width), 1fr); + grid-template-areas: "left right"; + + height: 100dvh; +} + +.c-landingPage__logo { + position: absolute; + left: var(--rs-split-page-left-pane-padding); + top: var(--rs-split-page-left-pane-padding); + + color: var(--rc-background--highlight-text); +} + +.c-landingPage__logo .c-logo { + padding: 0; +} + +.c-landingPage__left { + background: var(--rc-background--highlight); + color: var(--rc-background--highlight-text); + + grid-area: left; + + display: flex; + flex-direction: column; + align-items: flex-start; + + padding: var(--rs-split-page-left-pane-padding); +} + +.c-landingPage__left__content { + flex: 1; + + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + + padding: 0 20%; + + margin-top: var(--rs-logo-height); +} + +.c-landingPage__logo--mobile { + display: none; +} + +.c-landingPage__logo--desktop { + display: block; +} + +.c-landingPage__right { + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + + grid-area: right; + + overflow-y: scroll; +} + +.c-landingPage__right__content { + min-width: var(--rs-main-content-width); + max-width: var(--rs-main-content-width); + /* This is necessary in case the content is bigger than the parent. */ + /* It keeps the start of the box at the top of the container. */ + max-height: 90%; + + display: flex; + flex-direction: column; + align-items: stretch; +} + +.c-landingPage__right__content--full-width { + width: 100%; +} + +.c-landingPage .c-footer__link { + padding: 0; +} + +.c-landingPage__left__footer { + display: flex; + gap: var(--rs-inline-grid-gap); +} + +@media (max-width: 512px) { + .c-landingPage { + display: block; + + padding: 0 var(--rs-split-page-padding--mobile); + + background: var(--rc-background--highlight); + + height: auto; + min-height: 100vh; + } + + .c-landingPage__logo { + display: flex; + position: static; + top: unset; + left: unset; + justify-content: center; + + order: 1; + color: var(--rc-background--highlight-text); + + padding: var(--rs-split-page-stack) 0; + } + + .c-landingPage__left { + background: none; + order: 2; + + padding: 0; + margin-top: var(--rs-split-page-stack); + padding-bottom: var(--rs-split-page-stack); + + gap: var(--rs-split-page-stack); + } + + .c-landingPage__left__content { + padding: 0; + /* We don't need to account for the logo with absolute position */ + margin-top: 0; + } + + .c-landingPage__right { + order: 1; + + background: var(--rc-light); + border-radius: var(--rs-card-border-radius); + padding: var(--rs-split-page-right-pane-padding--mobile); + + overflow: auto; + } + + .c-landingPage__right__content { + min-width: auto; + max-width: none; + width: 100%; + } +} + /** * Card components */ diff --git a/src/showcase/src/pages/manageNewLanding.astro b/src/showcase/src/pages/manageNewLanding.astro new file mode 100644 index 0000000000..fc339c48da --- /dev/null +++ b/src/showcase/src/pages/manageNewLanding.astro @@ -0,0 +1,15 @@ +--- +import Screen from "../layouts/Screen.astro"; +--- + + + + diff --git a/src/showcase/src/pages/managePickManyNewLanding.astro b/src/showcase/src/pages/managePickManyNewLanding.astro new file mode 100644 index 0000000000..dd3c0995e2 --- /dev/null +++ b/src/showcase/src/pages/managePickManyNewLanding.astro @@ -0,0 +1,17 @@ +--- +import Screen from "../layouts/Screen.astro"; +--- + + + + diff --git a/src/showcase/src/pages/managePickNewLanding.astro b/src/showcase/src/pages/managePickNewLanding.astro new file mode 100644 index 0000000000..9b8218c838 --- /dev/null +++ b/src/showcase/src/pages/managePickNewLanding.astro @@ -0,0 +1,17 @@ +--- +import Screen from "../layouts/Screen.astro"; +--- + + + +