2022-10-28 16:11:08 +00:00
|
|
|
import {makeT} from 'app/client/lib/localization';
|
2022-06-17 03:43:26 +00:00
|
|
|
import * as commands from 'app/client/components/commands';
|
2022-01-14 02:55:55 +00:00
|
|
|
import {getUserPrefObs} from 'app/client/models/UserPrefs';
|
|
|
|
import {colors, testId} from 'app/client/ui2018/cssVars';
|
|
|
|
import {icon} from 'app/client/ui2018/icons';
|
|
|
|
import {IconName} from 'app/client/ui2018/IconList';
|
|
|
|
import {ISaveModalOptions, saveModal} from 'app/client/ui2018/modals';
|
|
|
|
import {BaseAPI} from 'app/common/BaseAPI';
|
|
|
|
import {UserPrefs} from 'app/common/Prefs';
|
2022-01-19 22:16:16 +00:00
|
|
|
import {getGristConfig} from 'app/common/urlUtils';
|
2022-01-14 02:55:55 +00:00
|
|
|
import {dom, input, Observable, styled, subscribeElem} from 'grainjs';
|
|
|
|
|
2022-10-28 16:11:08 +00:00
|
|
|
const t = makeT('WelcomeQuestions');
|
|
|
|
|
2022-01-14 02:55:55 +00:00
|
|
|
export function showWelcomeQuestions(userPrefsObs: Observable<UserPrefs>) {
|
2022-01-19 22:16:16 +00:00
|
|
|
if (!(getGristConfig().survey && userPrefsObs.get()?.showNewUserQuestions)) {
|
2022-01-14 02:55:55 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return saveModal((ctl, owner): ISaveModalOptions => {
|
|
|
|
const selection = choices.map(c => Observable.create(owner, false));
|
|
|
|
const otherText = Observable.create(owner, '');
|
|
|
|
const showQuestions = getUserPrefObs(userPrefsObs, 'showNewUserQuestions');
|
|
|
|
|
|
|
|
async function onConfirm() {
|
2022-10-28 16:11:08 +00:00
|
|
|
const selected = choices.filter((c, i) => selection[i].get()).map(c => t(c.textKey));
|
2022-01-14 02:55:55 +00:00
|
|
|
const use_cases = ['L', ...selected]; // Format to populate a ChoiceList column
|
2022-10-28 16:11:08 +00:00
|
|
|
const use_other = selected.includes(t('Other')) ? otherText.get() : '';
|
2022-01-14 02:55:55 +00:00
|
|
|
|
|
|
|
const submitUrl = new URL(window.location.href);
|
|
|
|
submitUrl.pathname = '/welcome/info';
|
2022-02-24 05:50:26 +00:00
|
|
|
return BaseAPI.request(submitUrl.href,
|
2022-01-14 02:55:55 +00:00
|
|
|
{method: 'POST', body: JSON.stringify({use_cases, use_other})});
|
|
|
|
}
|
|
|
|
|
2022-06-17 03:43:26 +00:00
|
|
|
|
|
|
|
owner.onDispose(async () => {
|
|
|
|
// Whichever way the modal is closed, don't show the questions again. (We set the value to
|
|
|
|
// undefined to remove it from the JSON prefs object entirely; it's never used again.)
|
|
|
|
showQuestions.set(undefined);
|
|
|
|
|
|
|
|
// Show the Grist video tour when the modal is closed.
|
|
|
|
await commands.allCommands.leftPanelOpen.run();
|
|
|
|
commands.allCommands.videoTourToolsOpen.run();
|
|
|
|
});
|
2022-01-14 02:55:55 +00:00
|
|
|
|
|
|
|
return {
|
2022-10-28 16:11:08 +00:00
|
|
|
title: [cssLogo(), dom('div', t('WelcomeToGrist'))],
|
2022-01-14 02:55:55 +00:00
|
|
|
body: buildInfoForm(selection, otherText),
|
|
|
|
saveLabel: 'Start using Grist',
|
|
|
|
saveFunc: onConfirm,
|
|
|
|
hideCancel: true,
|
|
|
|
width: 'fixed-wide',
|
|
|
|
modalArgs: cssModalCentered.cls(''),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-10-28 16:11:08 +00:00
|
|
|
const choices: Array<{icon: IconName, color: string, textKey: string}> = [
|
|
|
|
{icon: 'UseProduct', color: `${colors.lightGreen}`, textKey: 'ProductDevelopment' },
|
|
|
|
{icon: 'UseFinance', color: '#0075A2', textKey: 'FinanceAccounting' },
|
|
|
|
{icon: 'UseMedia', color: '#F7B32B', textKey: 'MediaProduction' },
|
|
|
|
{icon: 'UseMonitor', color: '#F2545B', textKey: 'ITTechnology' },
|
|
|
|
{icon: 'UseChart', color: '#7141F9', textKey: 'Marketing' },
|
|
|
|
{icon: 'UseScience', color: '#231942', textKey: 'Research' },
|
|
|
|
{icon: 'UseSales', color: '#885A5A', textKey: 'Sales' },
|
|
|
|
{icon: 'UseEducate', color: '#4A5899', textKey: 'Education' },
|
|
|
|
{icon: 'UseHr', color: '#688047', textKey: 'HRManagement' },
|
|
|
|
{icon: 'UseOther', color: '#929299', textKey: 'Other' },
|
2022-01-14 02:55:55 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
function buildInfoForm(selection: Observable<boolean>[], otherText: Observable<string>) {
|
|
|
|
return [
|
2022-10-28 16:11:08 +00:00
|
|
|
dom('span', t('WhatBringsYouToGrist')),
|
2022-01-14 02:55:55 +00:00
|
|
|
cssChoices(
|
|
|
|
choices.map((item, i) => cssChoice(
|
|
|
|
cssIcon(icon(item.icon), {style: `--icon-color: ${item.color}`}),
|
|
|
|
cssChoice.cls('-selected', selection[i]),
|
|
|
|
dom.on('click', () => selection[i].set(!selection[i].get())),
|
|
|
|
(item.icon !== 'UseOther' ?
|
2022-10-28 16:11:08 +00:00
|
|
|
t(item.textKey) :
|
2022-01-14 02:55:55 +00:00
|
|
|
[
|
2022-10-28 16:11:08 +00:00
|
|
|
cssOtherLabel(t(item.textKey)),
|
|
|
|
cssOtherInput(otherText, {}, {type: 'text', placeholder: t('TypeHere')},
|
2022-01-14 02:55:55 +00:00
|
|
|
// The following subscribes to changes to selection observable, and focuses the input when
|
|
|
|
// this item is selected.
|
|
|
|
(elem) => subscribeElem(elem, selection[i], val => val && setTimeout(() => elem.focus(), 0)),
|
|
|
|
// It's annoying if clicking into the input toggles selection; better to turn that
|
|
|
|
// off (user can click icon to deselect).
|
|
|
|
dom.on('click', ev => ev.stopPropagation()),
|
|
|
|
// Similarly, ignore Enter/Escape in "Other" textbox, so that they don't submit/close the form.
|
|
|
|
dom.onKeyDown({
|
|
|
|
Enter: (ev, elem) => elem.blur(),
|
|
|
|
Escape: (ev, elem) => elem.blur(),
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
]
|
|
|
|
)
|
|
|
|
)),
|
|
|
|
testId('welcome-questions'),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
const cssModalCentered = styled('div', `
|
|
|
|
text-align: center;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssLogo = styled('div', `
|
|
|
|
display: inline-block;
|
|
|
|
height: 48px;
|
|
|
|
width: 48px;
|
|
|
|
background-image: var(--icon-GristLogo);
|
|
|
|
background-size: 32px 32px;
|
|
|
|
background-repeat: no-repeat;
|
|
|
|
background-position: center;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssChoices = styled('div', `
|
|
|
|
display: flex;
|
|
|
|
flex-wrap: wrap;
|
|
|
|
align-items: center;
|
|
|
|
margin-top: 24px;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssChoice = styled('div', `
|
|
|
|
flex: 1 0 40%;
|
|
|
|
min-width: 0px;
|
|
|
|
margin: 8px 4px 0 4px;
|
|
|
|
height: 40px;
|
|
|
|
border: 1px solid ${colors.darkGrey};
|
|
|
|
border-radius: 3px;
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
text-align: left;
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
border-color: ${colors.lightGreen};
|
|
|
|
}
|
|
|
|
&-selected {
|
|
|
|
background-color: ${colors.mediumGrey};
|
|
|
|
}
|
|
|
|
&-selected:hover {
|
|
|
|
border-color: ${colors.darkGreen};
|
|
|
|
}
|
|
|
|
&-selected:focus-within {
|
|
|
|
box-shadow: 0 0 2px 0px var(--grist-color-cursor);
|
|
|
|
border-color: ${colors.lightGreen};
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssIcon = styled('div', `
|
|
|
|
margin: 0 16px;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssOtherLabel = styled('div', `
|
|
|
|
display: block;
|
|
|
|
.${cssChoice.className}-selected & {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssOtherInput = styled(input, `
|
|
|
|
display: none;
|
|
|
|
border: none;
|
|
|
|
background: none;
|
|
|
|
outline: none;
|
|
|
|
padding: 0px;
|
|
|
|
.${cssChoice.className}-selected & {
|
|
|
|
display: block;
|
|
|
|
}
|
|
|
|
`);
|