2022-08-22 19:46:25 +00:00
|
|
|
import { Disposable, dom, domComputed, DomContents, MultiHolder, Observable, styled } from "grainjs";
|
2020-10-02 15:10:00 +00:00
|
|
|
|
2022-01-19 19:41:06 +00:00
|
|
|
import { handleSubmit, submitForm } from "app/client/lib/formUtils";
|
2023-06-13 23:32:29 +00:00
|
|
|
import { AppModel } from "app/client/models/AppModel";
|
2021-06-25 14:00:01 +00:00
|
|
|
import { getLoginUrl, getSignupUrl, urlState } from "app/client/models/gristUrlState";
|
2020-10-02 15:10:00 +00:00
|
|
|
import { AccountWidget } from "app/client/ui/AccountWidget";
|
2021-08-18 17:49:34 +00:00
|
|
|
import { AppHeader } from 'app/client/ui/AppHeader';
|
2022-08-22 19:46:25 +00:00
|
|
|
import { textInput } from 'app/client/ui/inputs';
|
2020-10-02 15:10:00 +00:00
|
|
|
import { pagePanels } from "app/client/ui/PagePanels";
|
2021-10-01 14:24:23 +00:00
|
|
|
import { createUserImage } from 'app/client/ui/UserImage';
|
|
|
|
import { cssMemberImage, cssMemberListItem, cssMemberPrimary,
|
|
|
|
cssMemberSecondary, cssMemberText } from 'app/client/ui/UserItem';
|
2023-06-13 23:32:29 +00:00
|
|
|
import { buildWelcomeSitePicker } from 'app/client/ui/WelcomeSitePicker';
|
|
|
|
import { basicButtonLink, bigBasicButtonLink, bigPrimaryButton } from "app/client/ui2018/buttons";
|
2023-03-20 14:25:17 +00:00
|
|
|
import { mediaSmall, testId, theme, vars } from "app/client/ui2018/cssVars";
|
|
|
|
import { cssLink } from "app/client/ui2018/links";
|
2023-06-13 23:32:29 +00:00
|
|
|
import { WelcomePage as WelcomePageEnum } from 'app/common/gristUrls';
|
2020-10-02 15:10:00 +00:00
|
|
|
|
2021-06-25 14:00:01 +00:00
|
|
|
// Redirect from ..../welcome/thing to .../welcome/${name}
|
|
|
|
function _redirectToSiblingPage(name: string) {
|
|
|
|
const url = new URL(location.href);
|
|
|
|
const parts = url.pathname.split('/');
|
|
|
|
parts.pop();
|
|
|
|
parts.push(name);
|
|
|
|
url.pathname = parts.join('/');
|
|
|
|
window.location.assign(url.href);
|
|
|
|
}
|
|
|
|
|
2022-01-19 19:41:06 +00:00
|
|
|
function handleSubmitForm(
|
|
|
|
pending: Observable<boolean>,
|
2022-02-24 05:50:26 +00:00
|
|
|
onSuccess: (v: any) => void,
|
2022-01-19 19:41:06 +00:00
|
|
|
onError?: (e: unknown) => void
|
|
|
|
): (elem: HTMLFormElement) => void {
|
|
|
|
return handleSubmit(pending, submitForm, onSuccess, onError);
|
2020-10-15 21:51:30 +00:00
|
|
|
}
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
export class WelcomePage extends Disposable {
|
|
|
|
|
|
|
|
constructor(private _appModel: AppModel) {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
public buildDom() {
|
2023-06-13 23:32:29 +00:00
|
|
|
return domComputed(urlState().state, state => this._buildDomInPagePanels(state.welcome));
|
|
|
|
}
|
|
|
|
|
|
|
|
private _buildDomInPagePanels(page?: WelcomePageEnum) {
|
2020-10-02 15:10:00 +00:00
|
|
|
return pagePanels({
|
|
|
|
leftPanel: {
|
|
|
|
panelWidth: Observable.create(this, 240),
|
|
|
|
panelOpen: Observable.create(this, false),
|
|
|
|
hideOpener: true,
|
2021-08-18 17:49:34 +00:00
|
|
|
header: dom.create(AppHeader, '', this._appModel),
|
2020-10-02 15:10:00 +00:00
|
|
|
content: null,
|
|
|
|
},
|
|
|
|
headerMain: [cssFlexSpace(), dom.create(AccountWidget, this._appModel)],
|
2023-06-13 23:32:29 +00:00
|
|
|
contentMain: (
|
|
|
|
page === 'teams' ? dom.create(buildWelcomeSitePicker, this._appModel) :
|
|
|
|
this._buildPageContent(page)
|
|
|
|
),
|
2020-10-02 15:10:00 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-06-13 23:32:29 +00:00
|
|
|
private _buildPageContent(page?: WelcomePageEnum): Element {
|
2020-10-02 15:10:00 +00:00
|
|
|
return cssScrollContainer(cssContainer(
|
|
|
|
cssTitle('Welcome to Grist'),
|
|
|
|
testId('welcome-page'),
|
2023-06-13 23:32:29 +00:00
|
|
|
page === 'signup' ? dom.create(this._buildSignupForm.bind(this)) :
|
|
|
|
page === 'verify' ? dom.create(this._buildVerifyForm.bind(this)) :
|
|
|
|
page === 'select-account' ? dom.create(this._buildAccountPicker.bind(this)) :
|
|
|
|
null
|
2020-10-02 15:10:00 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2021-06-25 14:00:01 +00:00
|
|
|
private _buildSignupForm(owner: MultiHolder) {
|
|
|
|
let inputEl: HTMLInputElement;
|
|
|
|
const pending = Observable.create(owner, false);
|
|
|
|
|
|
|
|
// delayed focus
|
|
|
|
setTimeout(() => inputEl.focus(), 10);
|
|
|
|
|
|
|
|
// We expect to have an email query parameter on welcome/signup.
|
|
|
|
// TODO: make form work without email parameter - except the real todo is:
|
|
|
|
// TODO: replace this form with Amplify.
|
|
|
|
const url = new URL(location.href);
|
|
|
|
const email = Observable.create(owner, url.searchParams.get('email') || '');
|
|
|
|
const password = Observable.create(owner, '');
|
|
|
|
|
|
|
|
const action = new URL(window.location.href);
|
|
|
|
action.pathname = '/signup/register';
|
|
|
|
|
|
|
|
return dom(
|
|
|
|
'form',
|
|
|
|
{ method: "post", action: action.href },
|
2022-01-19 19:41:06 +00:00
|
|
|
handleSubmitForm(pending, () => _redirectToSiblingPage('verify')),
|
2023-03-20 14:25:17 +00:00
|
|
|
cssParagraph(
|
2021-06-25 14:00:01 +00:00
|
|
|
`Welcome Sumo-ling! ` + // This flow currently only used with AppSumo.
|
|
|
|
`Your Grist site is almost ready. Let's get your account set up and verified. ` +
|
|
|
|
`If you already have a Grist account as `,
|
|
|
|
dom('b', email.get()),
|
|
|
|
` you can just `,
|
2023-03-20 14:25:17 +00:00
|
|
|
cssLink({href: getLoginUrl('')}, 'log in'),
|
2021-06-25 14:00:01 +00:00
|
|
|
` now. Otherwise, please pick a password.`
|
|
|
|
),
|
|
|
|
cssSeparatedLabel('The email address you activated Grist with:'),
|
|
|
|
cssInput(
|
|
|
|
email, { onInput: true, },
|
|
|
|
{ name: "emailShow" },
|
|
|
|
dom.boolAttr('disabled', true),
|
|
|
|
dom.attr('type', 'email'),
|
|
|
|
),
|
|
|
|
// Duplicate email as a hidden form since disabled input won't get submitted
|
|
|
|
// for some reason.
|
|
|
|
cssInput(
|
|
|
|
email, { onInput: true, },
|
2023-03-20 14:25:17 +00:00
|
|
|
{ name: "email", style: 'visibility: hidden;' },
|
2021-06-25 14:00:01 +00:00
|
|
|
dom.boolAttr('hidden', true),
|
|
|
|
dom.attr('type', 'email'),
|
|
|
|
),
|
|
|
|
cssSeparatedLabel('A password to use with Grist:'),
|
|
|
|
inputEl = cssInput(
|
|
|
|
password, { onInput: true, },
|
|
|
|
{ name: "password" },
|
|
|
|
dom.attr('type', 'password'),
|
|
|
|
),
|
|
|
|
cssButtonGroup(
|
|
|
|
bigPrimaryButton(
|
|
|
|
'Continue',
|
|
|
|
testId('continue-button')
|
|
|
|
),
|
|
|
|
bigBasicButtonLink('Did this already', dom.on('click', () => {
|
|
|
|
_redirectToSiblingPage('verify');
|
|
|
|
}))
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _buildVerifyForm(owner: MultiHolder) {
|
|
|
|
let inputEl: HTMLInputElement;
|
|
|
|
const pending = Observable.create(owner, false);
|
|
|
|
|
|
|
|
// delayed focus
|
|
|
|
setTimeout(() => inputEl.focus(), 10);
|
|
|
|
|
|
|
|
const action = new URL(window.location.href);
|
|
|
|
action.pathname = '/signup/verify';
|
|
|
|
|
|
|
|
const url = new URL(location.href);
|
|
|
|
const email = Observable.create(owner, url.searchParams.get('email') || '');
|
|
|
|
const code = Observable.create(owner, url.searchParams.get('code') || '');
|
|
|
|
return dom(
|
|
|
|
'form',
|
|
|
|
{ method: "post", action: action.href },
|
2022-01-19 19:41:06 +00:00
|
|
|
handleSubmitForm(pending, (result) => {
|
2022-02-11 06:03:30 +00:00
|
|
|
if (result.status === 'confirmed') {
|
2021-06-25 14:00:01 +00:00
|
|
|
const verified = new URL(window.location.href);
|
|
|
|
verified.pathname = '/verified';
|
|
|
|
window.location.assign(verified.href);
|
2022-02-11 06:03:30 +00:00
|
|
|
} else if (result.status === 'resent') {
|
2021-06-25 14:00:01 +00:00
|
|
|
// just to give a sense that something happened...
|
|
|
|
window.location.reload();
|
|
|
|
}
|
|
|
|
}),
|
2023-03-20 14:25:17 +00:00
|
|
|
cssParagraph(
|
2021-06-25 14:00:01 +00:00
|
|
|
`Please check your email for a 6-digit verification code, and enter it here.`),
|
2023-03-20 14:25:17 +00:00
|
|
|
cssParagraph(
|
2021-06-25 14:00:01 +00:00
|
|
|
`If you've any trouble, try our full set of sign-up options. Do take care to use ` +
|
|
|
|
`the email address you activated with: `,
|
|
|
|
dom('b', email.get())),
|
|
|
|
cssSeparatedLabel('Confirmation code'),
|
|
|
|
inputEl = cssInput(
|
|
|
|
code, { onInput: true, },
|
|
|
|
{ name: "code" },
|
|
|
|
dom.attr('type', 'number'),
|
|
|
|
),
|
|
|
|
cssInput(
|
|
|
|
email, { onInput: true, },
|
|
|
|
{ name: "email" },
|
|
|
|
dom.boolAttr('hidden', true),
|
|
|
|
),
|
|
|
|
cssButtonGroup(
|
|
|
|
bigPrimaryButton(
|
|
|
|
dom.domComputed(code, c => c ?
|
|
|
|
'Apply verification code' : 'Resend verification email')
|
|
|
|
),
|
|
|
|
bigBasicButtonLink('More sign-up options',
|
2023-03-20 14:25:17 +00:00
|
|
|
{href: getSignupUrl('')})
|
2021-06-25 14:00:01 +00:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-10-01 14:24:23 +00:00
|
|
|
private _buildAccountPicker(): DomContents {
|
|
|
|
function addUserToLink(email: string): string {
|
|
|
|
const next = new URLSearchParams(location.search).get('next') || '';
|
|
|
|
const url = new URL(next, location.href);
|
|
|
|
url.searchParams.set('user', email);
|
|
|
|
return url.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
return [
|
|
|
|
cssParagraph(
|
|
|
|
"Select an account to continue with.",
|
|
|
|
),
|
|
|
|
dom.maybe(this._appModel.topAppModel.users, users =>
|
|
|
|
users.map(user => basicButtonLink(
|
|
|
|
cssUserItem.cls(''),
|
|
|
|
cssMemberListItem(
|
|
|
|
cssMemberImage(
|
|
|
|
createUserImage(user, 'large')
|
|
|
|
),
|
|
|
|
cssMemberText(
|
|
|
|
cssMemberPrimary(user.name || dom('span', user.email, testId('select-email'))),
|
|
|
|
user.name ? cssMemberSecondary(user.email, testId('select-email')) : null
|
|
|
|
),
|
|
|
|
),
|
|
|
|
{href: addUserToLink(user.email)},
|
|
|
|
testId('select-user'),
|
|
|
|
)),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
}
|
2020-10-02 15:10:00 +00:00
|
|
|
}
|
|
|
|
|
2021-10-01 14:24:23 +00:00
|
|
|
const cssUserItem = styled('div', `
|
|
|
|
margin: 0 0 8px;
|
|
|
|
align-items: center;
|
|
|
|
&:first-of-type {
|
|
|
|
margin-top: 16px;
|
|
|
|
}
|
|
|
|
&:hover {
|
2023-03-20 14:25:17 +00:00
|
|
|
background-color: ${theme.lightHover};
|
2021-10-01 14:24:23 +00:00
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
const cssScrollContainer = styled('div', `
|
|
|
|
display: flex;
|
|
|
|
overflow-y: auto;
|
|
|
|
flex-direction: column;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssContainer = styled('div', `
|
2021-02-08 19:00:02 +00:00
|
|
|
max-width: 450px;
|
2020-10-02 15:10:00 +00:00
|
|
|
align-self: center;
|
|
|
|
margin: 60px;
|
|
|
|
display: flex;
|
|
|
|
flex-direction: column;
|
|
|
|
&:after {
|
|
|
|
content: "";
|
|
|
|
height: 8px;
|
|
|
|
}
|
2021-02-08 19:00:02 +00:00
|
|
|
@media ${mediaSmall} {
|
|
|
|
& {
|
|
|
|
margin: 24px;
|
|
|
|
}
|
|
|
|
}
|
2020-10-02 15:10:00 +00:00
|
|
|
`);
|
|
|
|
|
|
|
|
const cssFlexSpace = styled('div', `
|
|
|
|
flex: 1 1 0px;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssTitle = styled('div', `
|
|
|
|
height: 32px;
|
|
|
|
line-height: 32px;
|
|
|
|
margin: 0 0 28px 0;
|
2023-03-20 14:25:17 +00:00
|
|
|
color: ${theme.text};
|
2020-10-02 15:10:00 +00:00
|
|
|
font-size: ${vars.xxxlargeFontSize};
|
|
|
|
font-weight: ${vars.headerControlTextWeight};
|
|
|
|
`);
|
|
|
|
|
|
|
|
const textStyle = `
|
|
|
|
font-weight: normal;
|
|
|
|
font-size: ${vars.mediumFontSize};
|
2023-03-20 14:25:17 +00:00
|
|
|
color: ${theme.text};
|
2020-10-02 15:10:00 +00:00
|
|
|
`;
|
|
|
|
|
2021-06-25 14:00:01 +00:00
|
|
|
// TODO: there's probably a much better way to style labels with a bit of
|
|
|
|
// space between them and things they are not the label for?
|
|
|
|
const cssSeparatedLabel = styled('label', textStyle + ' margin-top: 20px;');
|
2020-10-02 15:10:00 +00:00
|
|
|
const cssParagraph = styled('p', textStyle);
|
|
|
|
|
|
|
|
const cssButtonGroup = styled('div', `
|
|
|
|
margin-top: 24px;
|
|
|
|
display: flex;
|
2021-06-25 14:00:01 +00:00
|
|
|
justify-content: space-evenly;
|
2020-10-15 21:51:30 +00:00
|
|
|
&-right {
|
|
|
|
justify-content: flex-end;
|
|
|
|
}
|
2020-10-02 15:10:00 +00:00
|
|
|
`);
|
|
|
|
|
2022-08-22 19:46:25 +00:00
|
|
|
const cssInput = styled(textInput, `
|
|
|
|
display: inline;
|
|
|
|
height: 42px;
|
|
|
|
line-height: 16px;
|
|
|
|
padding: 13px;
|
|
|
|
border-radius: 3px;
|
|
|
|
`);
|