2022-01-07 18:11:52 +00:00
|
|
|
import {AppModel, reportError} from 'app/client/models/AppModel';
|
2022-03-17 02:32:17 +00:00
|
|
|
import {urlState} from 'app/client/models/gristUrlState';
|
2022-01-07 18:11:52 +00:00
|
|
|
import {ApiKey} from 'app/client/ui/ApiKey';
|
|
|
|
import {AppHeader} from 'app/client/ui/AppHeader';
|
2022-03-17 02:32:17 +00:00
|
|
|
import {buildChangePasswordDialog} from 'app/client/ui/ChangePasswordDialog';
|
2022-01-07 18:11:52 +00:00
|
|
|
import {leftPanelBasic} from 'app/client/ui/LeftPanelCommon';
|
2022-01-19 19:41:06 +00:00
|
|
|
import {MFAConfig} from 'app/client/ui/MFAConfig';
|
2022-01-07 18:11:52 +00:00
|
|
|
import {pagePanels} from 'app/client/ui/PagePanels';
|
|
|
|
import {createTopBarHome} from 'app/client/ui/TopBar';
|
|
|
|
import {transientInput} from 'app/client/ui/transientInput';
|
|
|
|
import {cssBreadcrumbs, cssBreadcrumbsLink, separator} from 'app/client/ui2018/breadcrumbs';
|
2022-02-14 21:26:21 +00:00
|
|
|
import {labeledSquareCheckbox} from 'app/client/ui2018/checkbox';
|
2022-01-07 18:11:52 +00:00
|
|
|
import {icon} from 'app/client/ui2018/icons';
|
|
|
|
import {colors, vars} from 'app/client/ui2018/cssVars';
|
2022-03-17 02:32:17 +00:00
|
|
|
import {FullUser} from 'app/common/UserAPI';
|
2022-01-07 18:11:52 +00:00
|
|
|
import {Computed, Disposable, dom, domComputed, makeTestId, Observable, styled} from 'grainjs';
|
|
|
|
|
|
|
|
const testId = makeTestId('test-account-page-');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates the account page where a user can manage their profile settings.
|
|
|
|
*/
|
|
|
|
export class AccountPage extends Disposable {
|
|
|
|
private _apiKey = Observable.create<string>(this, '');
|
|
|
|
private _userObs = Observable.create<FullUser|null>(this, null);
|
|
|
|
private _isEditingName = Observable.create(this, false);
|
|
|
|
private _nameEdit = Observable.create<string>(this, '');
|
|
|
|
private _isNameValid = Computed.create(this, this._nameEdit, (_use, val) => checkName(val));
|
2022-02-14 21:26:21 +00:00
|
|
|
private _allowGoogleLogin = Computed.create(this, (use) => use(this._userObs)?.allowGoogleLogin ?? false)
|
|
|
|
.onWrite((val) => this._updateAllowGooglelogin(val));
|
2022-01-07 18:11:52 +00:00
|
|
|
|
|
|
|
constructor(private _appModel: AppModel) {
|
|
|
|
super();
|
|
|
|
this._fetchAll().catch(reportError);
|
|
|
|
}
|
|
|
|
|
|
|
|
public buildDom() {
|
|
|
|
const panelOpen = Observable.create(this, false);
|
|
|
|
return pagePanels({
|
|
|
|
leftPanel: {
|
|
|
|
panelWidth: Observable.create(this, 240),
|
|
|
|
panelOpen,
|
|
|
|
hideOpener: true,
|
|
|
|
header: dom.create(AppHeader, this._appModel.currentOrgName, this._appModel),
|
|
|
|
content: leftPanelBasic(this._appModel, panelOpen),
|
|
|
|
},
|
|
|
|
headerMain: this._buildHeaderMain(),
|
|
|
|
contentMain: this._buildContentMain(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private _buildContentMain() {
|
|
|
|
return domComputed(this._userObs, (user) => user && (
|
|
|
|
cssContainer(cssAccountPage(
|
|
|
|
cssHeader('Account settings'),
|
2022-01-24 20:38:32 +00:00
|
|
|
cssDataRow(
|
|
|
|
cssSubHeader('Email'),
|
|
|
|
cssEmail(user.email),
|
|
|
|
),
|
2022-01-07 18:11:52 +00:00
|
|
|
cssDataRow(
|
|
|
|
cssSubHeader('Name'),
|
|
|
|
domComputed(this._isEditingName, (isEditing) => (
|
|
|
|
isEditing ? [
|
|
|
|
transientInput(
|
|
|
|
{
|
|
|
|
initialValue: user.name,
|
|
|
|
save: (val) => this._isNameValid.get() && this._updateUserName(val),
|
|
|
|
close: () => { this._isEditingName.set(false); this._nameEdit.set(''); },
|
|
|
|
},
|
2022-01-24 20:38:32 +00:00
|
|
|
{ size: '5' }, // Lower size so that input can shrink below ~152px.
|
2022-01-07 18:11:52 +00:00
|
|
|
dom.on('input', (_ev, el) => this._nameEdit.set(el.value)),
|
2022-01-24 20:38:32 +00:00
|
|
|
cssFlexGrow.cls(''),
|
2022-01-07 18:11:52 +00:00
|
|
|
),
|
|
|
|
cssTextBtn(
|
|
|
|
cssIcon('Settings'), 'Save',
|
|
|
|
// No need to save on 'click'. The transient input already does it on close.
|
|
|
|
),
|
|
|
|
] : [
|
2022-01-24 20:38:32 +00:00
|
|
|
cssName(user.name),
|
2022-01-07 18:11:52 +00:00
|
|
|
cssTextBtn(
|
|
|
|
cssIcon('Settings'), 'Edit',
|
|
|
|
dom.on('click', () => this._isEditingName.set(true)),
|
|
|
|
),
|
|
|
|
]
|
|
|
|
)),
|
|
|
|
testId('username'),
|
|
|
|
),
|
|
|
|
// show warning for invalid name but not for the empty string
|
|
|
|
dom.maybe(use => use(this._nameEdit) && !use(this._isNameValid), cssWarnings),
|
|
|
|
cssHeader('Password & Security'),
|
|
|
|
cssDataRow(
|
|
|
|
cssSubHeader('Login Method'),
|
2022-01-24 20:38:32 +00:00
|
|
|
cssLoginMethod(user.loginMethod),
|
2022-03-17 02:32:17 +00:00
|
|
|
user.loginMethod === 'Email + Password' ? cssTextBtn('Change Password',
|
|
|
|
dom.on('click', () => this._showChangePasswordDialog()),
|
2022-01-07 18:11:52 +00:00
|
|
|
) : null,
|
|
|
|
testId('login-method'),
|
|
|
|
),
|
2022-01-19 19:41:06 +00:00
|
|
|
user.loginMethod !== 'Email + Password' ? null : dom.frag(
|
2022-02-14 21:26:21 +00:00
|
|
|
cssDataRow(
|
|
|
|
labeledSquareCheckbox(
|
|
|
|
this._allowGoogleLogin,
|
|
|
|
'Allow signing in to this account with Google',
|
|
|
|
testId('allow-google-login-checkbox'),
|
|
|
|
),
|
|
|
|
testId('allow-google-login'),
|
|
|
|
),
|
2022-01-19 19:41:06 +00:00
|
|
|
cssSubHeaderFullWidth('Two-factor authentication'),
|
|
|
|
cssDescription(
|
|
|
|
"Two-factor authentication is an extra layer of security for your Grist account designed " +
|
|
|
|
"to ensure that you're the only person who can access your account, even if someone " +
|
|
|
|
"knows your password."
|
|
|
|
),
|
2022-03-17 02:32:17 +00:00
|
|
|
dom.create(MFAConfig, user),
|
2022-01-19 19:41:06 +00:00
|
|
|
),
|
2022-01-07 18:11:52 +00:00
|
|
|
cssHeader('API'),
|
|
|
|
cssDataRow(cssSubHeader('API Key'), cssContent(
|
|
|
|
dom.create(ApiKey, {
|
|
|
|
apiKey: this._apiKey,
|
|
|
|
onCreate: () => this._createApiKey(),
|
|
|
|
onDelete: () => this._deleteApiKey(),
|
|
|
|
anonymous: false,
|
2022-01-24 20:38:32 +00:00
|
|
|
inputArgs: [{ size: '5' }], // Lower size so that input can shrink below ~152px.
|
2022-01-07 18:11:52 +00:00
|
|
|
})
|
|
|
|
)),
|
|
|
|
),
|
|
|
|
testId('body'),
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
|
|
|
|
private _buildHeaderMain() {
|
|
|
|
return dom.frag(
|
|
|
|
cssBreadcrumbs({ style: 'margin-left: 16px;' },
|
|
|
|
cssBreadcrumbsLink(
|
|
|
|
urlState().setLinkUrl({}),
|
|
|
|
'Home',
|
|
|
|
testId('home'),
|
|
|
|
),
|
|
|
|
separator(' / '),
|
|
|
|
dom('span', 'Account'),
|
|
|
|
),
|
|
|
|
createTopBarHome(this._appModel),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _fetchApiKey() {
|
|
|
|
this._apiKey.set(await this._appModel.api.fetchApiKey());
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _createApiKey() {
|
|
|
|
this._apiKey.set(await this._appModel.api.createApiKey());
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _deleteApiKey() {
|
|
|
|
await this._appModel.api.deleteApiKey();
|
|
|
|
this._apiKey.set('');
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _fetchUserProfile() {
|
|
|
|
this._userObs.set(await this._appModel.api.getUserProfile());
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _fetchAll() {
|
|
|
|
await Promise.all([
|
|
|
|
this._fetchApiKey(),
|
|
|
|
this._fetchUserProfile(),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _updateUserName(val: string) {
|
|
|
|
const user = this._userObs.get();
|
|
|
|
if (user && val && val === user.name) { return; }
|
|
|
|
|
|
|
|
await this._appModel.api.updateUserName(val);
|
|
|
|
await this._fetchAll();
|
|
|
|
}
|
2022-02-14 21:26:21 +00:00
|
|
|
|
|
|
|
private async _updateAllowGooglelogin(allowGoogleLogin: boolean) {
|
|
|
|
await this._appModel.api.updateAllowGoogleLogin(allowGoogleLogin);
|
|
|
|
await this._fetchUserProfile();
|
|
|
|
}
|
2022-01-07 18:11:52 +00:00
|
|
|
|
2022-03-17 02:32:17 +00:00
|
|
|
private _showChangePasswordDialog() {
|
|
|
|
return buildChangePasswordDialog();
|
|
|
|
}
|
2022-01-07 18:11:52 +00:00
|
|
|
}
|
|
|
|
|
2022-02-24 05:50:26 +00:00
|
|
|
/**
|
|
|
|
* We allow alphanumeric characters and certain common whitelisted characters (except at the start),
|
|
|
|
* plus everything non-ASCII (for non-English alphabets, which we want to allow but it's hard to be
|
|
|
|
* more precise about what exactly to allow).
|
|
|
|
*/
|
|
|
|
// eslint-disable-next-line no-control-regex
|
|
|
|
const VALID_NAME_REGEXP = /^(\w|[^\u0000-\u007F])(\w|[- ./'"()]|[^\u0000-\u007F])*$/;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test name against various rules to check if it is a valid username.
|
|
|
|
*/
|
|
|
|
export function checkName(name: string): boolean {
|
|
|
|
return VALID_NAME_REGEXP.test(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Builds dom to show marning messages to the user.
|
|
|
|
*/
|
|
|
|
function buildNameWarningsDom() {
|
|
|
|
return cssWarning(
|
|
|
|
"Names only allow letters, numbers and certain special characters",
|
|
|
|
testId('username-warning'),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-07 18:11:52 +00:00
|
|
|
const cssContainer = styled('div', `
|
|
|
|
display: flex;
|
|
|
|
justify-content: center;
|
|
|
|
overflow: auto;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssHeader = styled('div', `
|
|
|
|
height: 32px;
|
|
|
|
line-height: 32px;
|
|
|
|
margin: 28px 0 16px 0;
|
|
|
|
color: ${colors.dark};
|
|
|
|
font-size: ${vars.xxxlargeFontSize};
|
|
|
|
font-weight: ${vars.headerControlTextWeight};
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssAccountPage = styled('div', `
|
|
|
|
max-width: 600px;
|
2022-01-24 20:38:32 +00:00
|
|
|
padding: 16px;
|
2022-01-07 18:11:52 +00:00
|
|
|
`);
|
|
|
|
|
|
|
|
const cssDataRow = styled('div', `
|
|
|
|
margin: 8px 0px;
|
|
|
|
display: flex;
|
|
|
|
align-items: baseline;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssSubHeaderFullWidth = styled('div', `
|
|
|
|
padding: 8px 0;
|
|
|
|
display: inline-block;
|
|
|
|
vertical-align: top;
|
|
|
|
font-weight: bold;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssSubHeader = styled(cssSubHeaderFullWidth, `
|
2022-01-24 20:38:32 +00:00
|
|
|
min-width: 110px;
|
2022-01-07 18:11:52 +00:00
|
|
|
`);
|
|
|
|
|
|
|
|
const cssContent = styled('div', `
|
|
|
|
flex: 1 1 300px;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssTextBtn = styled('button', `
|
|
|
|
font-size: ${vars.mediumFontSize};
|
|
|
|
color: ${colors.lightGreen};
|
|
|
|
cursor: pointer;
|
2022-01-24 20:38:32 +00:00
|
|
|
margin-left: 16px;
|
2022-01-07 18:11:52 +00:00
|
|
|
background-color: transparent;
|
|
|
|
border: none;
|
|
|
|
padding: 0;
|
|
|
|
text-align: left;
|
2022-03-17 02:32:17 +00:00
|
|
|
min-width: 110px;
|
2022-01-07 18:11:52 +00:00
|
|
|
|
|
|
|
&:hover {
|
|
|
|
color: ${colors.darkGreen};
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssIcon = styled(icon, `
|
|
|
|
background-color: ${colors.lightGreen};
|
|
|
|
margin: 0 4px 2px 0;
|
|
|
|
|
|
|
|
.${cssTextBtn.className}:hover > & {
|
|
|
|
background-color: ${colors.darkGreen};
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssWarnings = styled(buildNameWarningsDom, `
|
|
|
|
margin: -8px 0 0 110px;
|
|
|
|
`);
|
|
|
|
|
2022-01-19 19:41:06 +00:00
|
|
|
const cssDescription = styled('div', `
|
|
|
|
color: #8a8a8a;
|
|
|
|
font-size: 13px;
|
|
|
|
`);
|
2022-01-24 20:38:32 +00:00
|
|
|
|
|
|
|
const cssFlexGrow = styled('div', `
|
|
|
|
flex-grow: 1;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssName = styled(cssFlexGrow, `
|
|
|
|
word-break: break-word;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssEmail = styled('div', `
|
|
|
|
word-break: break-word;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssLoginMethod = styled(cssFlexGrow, `
|
|
|
|
word-break: break-word;
|
|
|
|
`);
|
2022-02-24 05:50:26 +00:00
|
|
|
|
|
|
|
const cssWarning = styled('div', `
|
|
|
|
color: red;
|
|
|
|
`);
|