mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
748 lines
19 KiB
TypeScript
748 lines
19 KiB
TypeScript
|
import {FocusLayer} from 'app/client/lib/FocusLayer';
|
||
|
import {makeT} from 'app/client/lib/localization';
|
||
|
import {AppModel} from 'app/client/models/AppModel';
|
||
|
import {logError} from 'app/client/models/errors';
|
||
|
import {getMainOrgUrl, urlState} from 'app/client/models/gristUrlState';
|
||
|
import {getUserPrefObs} from 'app/client/models/UserPrefs';
|
||
|
import {textInput} from 'app/client/ui/inputs';
|
||
|
import {PlayerState, YouTubePlayer} from 'app/client/ui/YouTubePlayer';
|
||
|
import {bigBasicButton, bigPrimaryButton, bigPrimaryButtonLink} from 'app/client/ui2018/buttons';
|
||
|
import {colors, mediaMedium, mediaXSmall, theme} from 'app/client/ui2018/cssVars';
|
||
|
import {icon} from 'app/client/ui2018/icons';
|
||
|
import {IconName} from 'app/client/ui2018/IconList';
|
||
|
import {modal} from 'app/client/ui2018/modals';
|
||
|
import {BaseAPI} from 'app/common/BaseAPI';
|
||
|
import {getPageTitleSuffix, ONBOARDING_VIDEO_YOUTUBE_EMBED_ID} from 'app/common/gristUrls';
|
||
|
import {UserPrefs} from 'app/common/Prefs';
|
||
|
import {getGristConfig} from 'app/common/urlUtils';
|
||
|
import {
|
||
|
Computed,
|
||
|
Disposable,
|
||
|
dom,
|
||
|
DomContents,
|
||
|
IDisposableOwner,
|
||
|
input,
|
||
|
makeTestId,
|
||
|
Observable,
|
||
|
styled,
|
||
|
subscribeElem,
|
||
|
} from 'grainjs';
|
||
|
|
||
|
const t = makeT('OnboardingPage');
|
||
|
|
||
|
const testId = makeTestId('test-onboarding-');
|
||
|
|
||
|
const choices: Array<{icon: IconName, color: string, textKey: string}> = [
|
||
|
{icon: 'UseProduct', color: `${colors.lightGreen}`, textKey: 'Product Development' },
|
||
|
{icon: 'UseFinance', color: '#0075A2', textKey: 'Finance & Accounting'},
|
||
|
{icon: 'UseMedia', color: '#F7B32B', textKey: 'Media Production' },
|
||
|
{icon: 'UseMonitor', color: '#F2545B', textKey: 'IT & Technology' },
|
||
|
{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: 'HR & Management' },
|
||
|
{icon: 'UseOther', color: '#929299', textKey: 'Other' },
|
||
|
];
|
||
|
|
||
|
export function shouldShowOnboardingPage(userPrefsObs: Observable<UserPrefs>): boolean {
|
||
|
return Boolean(getGristConfig().survey && userPrefsObs.get()?.showNewUserQuestions);
|
||
|
}
|
||
|
|
||
|
type IncrementStep = (delta?: 1 | -1) => void;
|
||
|
|
||
|
interface Step {
|
||
|
state?: QuestionsState | VideoState;
|
||
|
buildDom(): DomContents;
|
||
|
onNavigateAway?(): void;
|
||
|
}
|
||
|
|
||
|
interface QuestionsState {
|
||
|
organization: Observable<string>;
|
||
|
role: Observable<string>;
|
||
|
useCases: Array<Observable<boolean>>;
|
||
|
useOther: Observable<string>;
|
||
|
}
|
||
|
|
||
|
interface VideoState {
|
||
|
watched: Observable<boolean>;
|
||
|
}
|
||
|
|
||
|
export class OnboardingPage extends Disposable {
|
||
|
private _steps: Array<Step>;
|
||
|
private _stepIndex: Observable<number> = Observable.create(this, 0);
|
||
|
|
||
|
constructor(private _appModel: AppModel) {
|
||
|
super();
|
||
|
|
||
|
this.autoDispose(this._stepIndex.addListener((_, prevIndex) => {
|
||
|
this._steps[prevIndex].onNavigateAway?.();
|
||
|
}));
|
||
|
|
||
|
const incrementStep: IncrementStep = (delta: -1 | 1 = 1) => {
|
||
|
this._stepIndex.set(this._stepIndex.get() + delta);
|
||
|
};
|
||
|
|
||
|
this._steps = [
|
||
|
{
|
||
|
state: {
|
||
|
organization: Observable.create(this, ''),
|
||
|
role: Observable.create(this, ''),
|
||
|
useCases: choices.map(() => Observable.create(this, false)),
|
||
|
useOther: Observable.create(this, ''),
|
||
|
},
|
||
|
buildDom() { return dom.create(buildQuestions, incrementStep, this.state as QuestionsState); },
|
||
|
onNavigateAway() { saveQuestions(this.state as QuestionsState); },
|
||
|
},
|
||
|
{
|
||
|
state: {
|
||
|
watched: Observable.create(this, false),
|
||
|
},
|
||
|
buildDom() { return dom.create(buildVideo, incrementStep, this.state as VideoState); },
|
||
|
},
|
||
|
{
|
||
|
buildDom() { return dom.create(buildTutorial, incrementStep); },
|
||
|
},
|
||
|
];
|
||
|
|
||
|
document.title = `Welcome${getPageTitleSuffix(getGristConfig())}`;
|
||
|
|
||
|
getUserPrefObs(this._appModel.userPrefsObs, 'showNewUserQuestions').set(undefined);
|
||
|
}
|
||
|
|
||
|
public buildDom() {
|
||
|
return cssPageContainer(
|
||
|
cssOnboardingPage(
|
||
|
cssSidebar(
|
||
|
cssSidebarContent(
|
||
|
cssSidebarHeading1(t('Welcome')),
|
||
|
cssSidebarHeading2(this._appModel.currentUser!.name + '!'),
|
||
|
testId('sidebar'),
|
||
|
),
|
||
|
cssGetStarted(
|
||
|
cssGetStartedImg({src: 'img/get-started.png'}),
|
||
|
),
|
||
|
),
|
||
|
cssMainPanel(
|
||
|
buildStepper(this._steps, this._stepIndex),
|
||
|
dom.domComputed(this._stepIndex, index => {
|
||
|
return this._steps[index].buildDom();
|
||
|
}),
|
||
|
),
|
||
|
testId('page'),
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function buildStepper(steps: Step[], stepIndex: Observable<number>) {
|
||
|
return cssStepper(
|
||
|
steps.map((_, i) =>
|
||
|
cssStep(
|
||
|
cssStepCircle(
|
||
|
cssStepCircle.cls('-done', use => (i < use(stepIndex))),
|
||
|
dom.domComputed(use => i < use(stepIndex), (done) => done ? icon('Tick') : String(i + 1)),
|
||
|
cssStepCircle.cls('-current', use => (i === use(stepIndex))),
|
||
|
dom.on('click', () => { stepIndex.set(i); }),
|
||
|
testId(`step-${i + 1}`)
|
||
|
)
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
function saveQuestions(state: QuestionsState) {
|
||
|
const {organization, role, useCases, useOther} = state;
|
||
|
if (!organization.get() && !role.get() && !useCases.map(useCase => useCase.get()).includes(true)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const org_name = organization.get();
|
||
|
const org_role = role.get();
|
||
|
const use_cases = choices.filter((c, i) => useCases[i].get()).map(c => c.textKey);
|
||
|
const use_other = use_cases.includes('Other') ? useOther.get() : '';
|
||
|
const submitUrl = new URL(window.location.href);
|
||
|
submitUrl.pathname = '/welcome/info';
|
||
|
BaseAPI.request(submitUrl.href, {
|
||
|
method: 'POST',
|
||
|
body: JSON.stringify({org_name, org_role, use_cases, use_other})
|
||
|
}).catch((e) => logError(e));
|
||
|
}
|
||
|
|
||
|
function buildQuestions(owner: IDisposableOwner, incrementStep: IncrementStep, state: QuestionsState) {
|
||
|
const {organization, role, useCases, useOther} = state;
|
||
|
const isFilled = Computed.create(owner, (use) => {
|
||
|
return Boolean(use(organization) || use(role) || useCases.map(useCase => use(useCase)).includes(true));
|
||
|
});
|
||
|
|
||
|
return cssQuestions(
|
||
|
cssHeading(t("Tell us who you are")),
|
||
|
cssQuestion(
|
||
|
cssFieldHeading(t('What organization are you with?')),
|
||
|
cssInput(
|
||
|
organization,
|
||
|
{type: 'text', placeholder: t('Your organization')},
|
||
|
testId('questions-organization'),
|
||
|
),
|
||
|
),
|
||
|
cssQuestion(
|
||
|
cssFieldHeading(t('What is your role?')),
|
||
|
cssInput(
|
||
|
role,
|
||
|
{type: 'text', placeholder: t('Your role')},
|
||
|
testId('questions-role'),
|
||
|
),
|
||
|
),
|
||
|
cssQuestion(
|
||
|
cssFieldHeading(t("What brings you to Grist (you can select multiple)?")),
|
||
|
cssUseCases(
|
||
|
choices.map((item, i) => cssUseCase(
|
||
|
cssUseCaseIcon(icon(item.icon)),
|
||
|
cssUseCase.cls('-selected', useCases[i]),
|
||
|
dom.on('click', () => useCases[i].set(!useCases[i].get())),
|
||
|
(item.icon !== 'UseOther' ?
|
||
|
t(item.textKey) :
|
||
|
[
|
||
|
cssOtherLabel(t(item.textKey)),
|
||
|
cssOtherInput(useOther, {}, {type: 'text', placeholder: t("Type here")},
|
||
|
// The following subscribes to changes to selection observable, and focuses the input when
|
||
|
// this item is selected.
|
||
|
(elem) => subscribeElem(elem, useCases[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('questions-use-case'),
|
||
|
)),
|
||
|
),
|
||
|
),
|
||
|
cssContinue(
|
||
|
bigPrimaryButton(
|
||
|
t('Next step'),
|
||
|
dom.show(isFilled),
|
||
|
dom.on('click', () => incrementStep()),
|
||
|
testId('next-step'),
|
||
|
),
|
||
|
bigBasicButton(
|
||
|
t('Skip step'),
|
||
|
dom.hide(isFilled),
|
||
|
dom.on('click', () => incrementStep()),
|
||
|
testId('skip-step'),
|
||
|
),
|
||
|
),
|
||
|
testId('questions'),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
function buildVideo(_owner: IDisposableOwner, incrementStep: IncrementStep, state: VideoState) {
|
||
|
const {watched} = state;
|
||
|
|
||
|
function onPlay() {
|
||
|
watched.set(true);
|
||
|
|
||
|
return modal((ctl, modalOwner) => {
|
||
|
const youtubePlayer = YouTubePlayer.create(modalOwner,
|
||
|
ONBOARDING_VIDEO_YOUTUBE_EMBED_ID,
|
||
|
{
|
||
|
onPlayerReady: (player) => player.playVideo(),
|
||
|
onPlayerStateChange(_player, {data}) {
|
||
|
if (data !== PlayerState.Ended) { return; }
|
||
|
|
||
|
ctl.close();
|
||
|
},
|
||
|
height: '100%',
|
||
|
width: '100%',
|
||
|
origin: getMainOrgUrl(),
|
||
|
},
|
||
|
cssYouTubePlayer.cls(''),
|
||
|
);
|
||
|
|
||
|
return [
|
||
|
dom.on('click', () => ctl.close()),
|
||
|
elem => { FocusLayer.create(modalOwner, {defaultFocusElem: elem, pauseMousetrap: true}); },
|
||
|
dom.onKeyDown({
|
||
|
Escape: () => ctl.close(),
|
||
|
' ': () => youtubePlayer.playPause(),
|
||
|
}),
|
||
|
cssModalHeader(
|
||
|
cssModalCloseButton(
|
||
|
cssCloseIcon('CrossBig'),
|
||
|
),
|
||
|
),
|
||
|
cssModalBody(
|
||
|
cssVideoPlayer(
|
||
|
dom.on('click', (ev) => ev.stopPropagation()),
|
||
|
youtubePlayer.buildDom(),
|
||
|
testId('video-player'),
|
||
|
),
|
||
|
cssModalButtons(
|
||
|
bigPrimaryButton(
|
||
|
t('Next step'),
|
||
|
dom.on('click', (ev) => {
|
||
|
ev.stopPropagation();
|
||
|
ctl.close();
|
||
|
incrementStep();
|
||
|
}),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
cssVideoPlayerModal.cls(''),
|
||
|
];
|
||
|
});
|
||
|
}
|
||
|
|
||
|
return dom('div',
|
||
|
cssHeading(t('Discover Grist in 3 minutes')),
|
||
|
cssScreenshot(
|
||
|
dom.on('click', onPlay),
|
||
|
dom('div',
|
||
|
cssScreenshotImg({src: 'img/youtube-screenshot.png'}),
|
||
|
cssActionOverlay(
|
||
|
cssAction(
|
||
|
cssRoundButton(cssVideoPlayIcon('VideoPlay')),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
testId('video-thumbnail'),
|
||
|
),
|
||
|
cssContinue(
|
||
|
cssBackButton(
|
||
|
t('Back'),
|
||
|
dom.on('click', () => incrementStep(-1)),
|
||
|
testId('back'),
|
||
|
),
|
||
|
bigPrimaryButton(
|
||
|
t('Next step'),
|
||
|
dom.show(watched),
|
||
|
dom.on('click', () => incrementStep()),
|
||
|
testId('next-step'),
|
||
|
),
|
||
|
bigBasicButton(
|
||
|
t('Skip step'),
|
||
|
dom.hide(watched),
|
||
|
dom.on('click', () => incrementStep()),
|
||
|
testId('skip-step'),
|
||
|
),
|
||
|
),
|
||
|
testId('video'),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
function buildTutorial(_owner: IDisposableOwner, incrementStep: IncrementStep) {
|
||
|
const {templateOrg, onboardingTutorialDocId} = getGristConfig();
|
||
|
return dom('div',
|
||
|
cssHeading(
|
||
|
t('Go hands-on with the Grist Basics tutorial'),
|
||
|
cssSubHeading(
|
||
|
t("Grist may look like a spreadsheet, but it doesn't always "
|
||
|
+ "act like one. Discover what makes Grist different."
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
cssTutorial(
|
||
|
cssScreenshot(
|
||
|
dom.on('click', () => urlState().pushUrl({org: templateOrg!, doc: onboardingTutorialDocId})),
|
||
|
cssTutorialScreenshotImg({src: 'img/tutorial-screenshot.png'}),
|
||
|
cssTutorialOverlay(
|
||
|
cssAction(
|
||
|
cssTutorialButton(t('Go to the tutorial!')),
|
||
|
),
|
||
|
),
|
||
|
testId('tutorial-thumbnail'),
|
||
|
),
|
||
|
),
|
||
|
cssContinue(
|
||
|
cssBackButton(
|
||
|
t('Back'),
|
||
|
dom.on('click', () => incrementStep(-1)),
|
||
|
testId('back'),
|
||
|
),
|
||
|
bigBasicButton(
|
||
|
t('Skip tutorial'),
|
||
|
dom.on('click', () => window.location.href = urlState().makeUrl(urlState().state.get())),
|
||
|
testId('skip-tutorial'),
|
||
|
),
|
||
|
),
|
||
|
testId('tutorial'),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
const cssPageContainer = styled('div', `
|
||
|
overflow: auto;
|
||
|
height: 100%;
|
||
|
background-color: ${theme.mainPanelBg};
|
||
|
`);
|
||
|
|
||
|
const cssOnboardingPage = styled('div', `
|
||
|
display: flex;
|
||
|
min-height: 100%;
|
||
|
`);
|
||
|
|
||
|
const cssSidebar = styled('div', `
|
||
|
width: 460px;
|
||
|
background-color: ${colors.lightGreen};
|
||
|
color: ${colors.light};
|
||
|
background-image:
|
||
|
linear-gradient(to bottom, rgb(41, 185, 131) 32px, transparent 32px),
|
||
|
linear-gradient(to right, rgb(41, 185, 131) 32px, transparent 32px);
|
||
|
background-size: 240px 120px;
|
||
|
background-position: 0 0, 40%;
|
||
|
display: flex;
|
||
|
flex-direction: column;
|
||
|
|
||
|
@media ${mediaMedium} {
|
||
|
& {
|
||
|
display: none;
|
||
|
}
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssGetStarted = styled('div', `
|
||
|
width: 500px;
|
||
|
height: 350px;
|
||
|
margin: auto -77px 0 37px;
|
||
|
overflow: hidden;
|
||
|
`);
|
||
|
|
||
|
const cssGetStartedImg = styled('img', `
|
||
|
display: block;
|
||
|
width: 500px;
|
||
|
height: auto;
|
||
|
`);
|
||
|
|
||
|
const cssSidebarContent = styled('div', `
|
||
|
line-height: 32px;
|
||
|
margin: 112px 16px 64px 16px;
|
||
|
font-size: 24px;
|
||
|
line-height: 48px;
|
||
|
font-weight: 500;
|
||
|
`);
|
||
|
|
||
|
const cssSidebarHeading1 = styled('div', `
|
||
|
font-size: 32px;
|
||
|
text-align: center;
|
||
|
`);
|
||
|
|
||
|
const cssSidebarHeading2 = styled('div', `
|
||
|
font-size: 28px;
|
||
|
text-align: center;
|
||
|
`);
|
||
|
|
||
|
const cssMainPanel = styled('div', `
|
||
|
margin: 56px auto;
|
||
|
padding: 0px 96px;
|
||
|
text-align: center;
|
||
|
|
||
|
@media ${mediaMedium} {
|
||
|
& {
|
||
|
padding: 0px 32px;
|
||
|
}
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssHeading = styled('div', `
|
||
|
color: ${theme.text};
|
||
|
font-size: 24px;
|
||
|
font-weight: 500;
|
||
|
margin: 32px 0px;
|
||
|
`);
|
||
|
|
||
|
const cssSubHeading = styled(cssHeading, `
|
||
|
font-size: 15px;
|
||
|
font-weight: 400;
|
||
|
margin-top: 16px;
|
||
|
`);
|
||
|
|
||
|
const cssStep = styled('div', `
|
||
|
display: flex;
|
||
|
align-items: center;
|
||
|
cursor: default;
|
||
|
|
||
|
&:not(:last-child)::after {
|
||
|
content: "";
|
||
|
width: 50px;
|
||
|
height: 2px;
|
||
|
background-color: var(--grist-color-light-green);
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssStepCircle = styled('div', `
|
||
|
--icon-color: ${theme.controlPrimaryFg};
|
||
|
--step-color: ${theme.controlPrimaryBg};
|
||
|
display: inline-block;
|
||
|
width: 24px;
|
||
|
height: 24px;
|
||
|
border-radius: 30px;
|
||
|
border: 1px solid var(--step-color);
|
||
|
color: var(--step-color);
|
||
|
margin: 4px;
|
||
|
position: relative;
|
||
|
cursor: pointer;
|
||
|
|
||
|
&:hover {
|
||
|
--step-color: ${theme.controlPrimaryHoverBg};
|
||
|
}
|
||
|
&-current {
|
||
|
background-color: var(--step-color);
|
||
|
color: ${theme.controlPrimaryFg};
|
||
|
outline: 3px solid ${theme.cursorInactive};
|
||
|
}
|
||
|
&-done {
|
||
|
background-color: var(--step-color);
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssQuestions = styled('div', `
|
||
|
max-width: 500px;
|
||
|
`);
|
||
|
|
||
|
const cssQuestion = styled('div', `
|
||
|
margin: 16px 0 8px 0;
|
||
|
text-align: left;
|
||
|
`);
|
||
|
|
||
|
const cssFieldHeading = styled('div', `
|
||
|
color: ${theme.text};
|
||
|
font-size: 13px;
|
||
|
font-weight: 700;
|
||
|
margin-bottom: 12px;
|
||
|
`);
|
||
|
|
||
|
const cssContinue = styled('div', `
|
||
|
display: flex;
|
||
|
justify-content: center;
|
||
|
margin-top: 40px;
|
||
|
gap: 16px;
|
||
|
`);
|
||
|
|
||
|
const cssUseCases = styled('div', `
|
||
|
display: flex;
|
||
|
flex-wrap: wrap;
|
||
|
align-items: center;
|
||
|
margin: -8px -4px;
|
||
|
`);
|
||
|
|
||
|
const cssUseCase = styled('div', `
|
||
|
flex: 1 0 40%;
|
||
|
min-width: 200px;
|
||
|
margin: 8px 4px 0 4px;
|
||
|
height: 40px;
|
||
|
border: 1px solid ${theme.inputBorder};
|
||
|
border-radius: 3px;
|
||
|
display: flex;
|
||
|
align-items: center;
|
||
|
text-align: left;
|
||
|
cursor: pointer;
|
||
|
color: ${theme.text};
|
||
|
--icon-color: ${theme.accentIcon};
|
||
|
|
||
|
&:hover {
|
||
|
background-color: ${theme.hover};
|
||
|
}
|
||
|
&-selected {
|
||
|
border: 2px solid ${theme.controlFg};
|
||
|
}
|
||
|
&-selected:hover {
|
||
|
border: 2px solid ${theme.controlHoverFg};
|
||
|
}
|
||
|
&-selected:focus-within {
|
||
|
box-shadow: 0 0 2px 0px ${theme.controlFg};
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssUseCaseIcon = styled('div', `
|
||
|
margin: 0 16px;
|
||
|
--icon-color: ${theme.accentIcon};
|
||
|
`);
|
||
|
|
||
|
const cssOtherLabel = styled('div', `
|
||
|
display: block;
|
||
|
|
||
|
.${cssUseCase.className}-selected & {
|
||
|
display: none;
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssInput = styled(textInput, `
|
||
|
height: 40px;
|
||
|
`);
|
||
|
|
||
|
const cssOtherInput = styled(input, `
|
||
|
color: ${theme.inputFg};
|
||
|
display: none;
|
||
|
border: none;
|
||
|
background: none;
|
||
|
outline: none;
|
||
|
padding: 0px;
|
||
|
|
||
|
&::placeholder {
|
||
|
color: ${theme.inputPlaceholderFg};
|
||
|
}
|
||
|
.${cssUseCase.className}-selected & {
|
||
|
display: block;
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssTutorial = styled('div', `
|
||
|
display: flex;
|
||
|
justify-content: center;
|
||
|
`);
|
||
|
|
||
|
const cssScreenshot = styled('div', `
|
||
|
max-width: 720px;
|
||
|
display: flex;
|
||
|
position: relative;
|
||
|
border-radius: 3px;
|
||
|
border: 3px solid ${colors.lightGreen};
|
||
|
overflow: hidden;
|
||
|
cursor: pointer;
|
||
|
`);
|
||
|
|
||
|
const cssActionOverlay = styled('div', `
|
||
|
position: absolute;
|
||
|
z-index: 1;
|
||
|
top: 0;
|
||
|
left: 0;
|
||
|
width: 100%;
|
||
|
height: 100%;
|
||
|
background-color: rgba(0, 0, 0, 0.20);
|
||
|
`);
|
||
|
|
||
|
const cssTutorialOverlay = styled(cssActionOverlay, `
|
||
|
background-color: transparent;
|
||
|
`);
|
||
|
|
||
|
const cssAction = styled('div', `
|
||
|
display: flex;
|
||
|
flex-direction: column;
|
||
|
margin: auto;
|
||
|
align-items: center;
|
||
|
justify-content: center;
|
||
|
height: 100%;
|
||
|
`);
|
||
|
|
||
|
const cssVideoPlayIcon = styled(icon, `
|
||
|
--icon-color: ${colors.light};
|
||
|
width: 38px;
|
||
|
height: 33.25px;
|
||
|
`);
|
||
|
|
||
|
const cssCloseIcon = styled(icon, `
|
||
|
--icon-color: ${colors.light};
|
||
|
width: 22px;
|
||
|
height: 22px;
|
||
|
`);
|
||
|
|
||
|
const cssYouTubePlayer = styled('iframe', `
|
||
|
border-radius: 4px;
|
||
|
`);
|
||
|
|
||
|
const cssModalHeader = styled('div', `
|
||
|
display: flex;
|
||
|
flex-shrink: 0;
|
||
|
justify-content: flex-end;
|
||
|
`);
|
||
|
|
||
|
const cssModalBody = styled('div', `
|
||
|
display: flex;
|
||
|
flex-grow: 1;
|
||
|
flex-direction: column;
|
||
|
justify-content: center;
|
||
|
align-items: center;
|
||
|
`);
|
||
|
|
||
|
const cssBackButton = styled(bigBasicButton, `
|
||
|
border: none;
|
||
|
`);
|
||
|
|
||
|
const cssModalButtons = styled('div', `
|
||
|
display: flex;
|
||
|
justify-content: center;
|
||
|
margin-top: 24px;
|
||
|
`);
|
||
|
|
||
|
const cssVideoPlayer = styled('div', `
|
||
|
width: 100%;
|
||
|
max-width: 1280px;
|
||
|
height: 100%;
|
||
|
max-height: 720px;
|
||
|
|
||
|
@media ${mediaXSmall} {
|
||
|
& {
|
||
|
max-height: 240px;
|
||
|
}
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssVideoPlayerModal = styled('div', `
|
||
|
display: flex;
|
||
|
flex-direction: column;
|
||
|
width: 100%;
|
||
|
height: 100%;
|
||
|
padding: 8px;
|
||
|
background-color: transparent;
|
||
|
box-shadow: none;
|
||
|
`);
|
||
|
|
||
|
const cssModalCloseButton = styled('div', `
|
||
|
margin-bottom: 8px;
|
||
|
padding: 4px;
|
||
|
border-radius: 4px;
|
||
|
cursor: pointer;
|
||
|
|
||
|
&:hover {
|
||
|
background-color: ${theme.hover};
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssScreenshotImg = styled('img', `
|
||
|
transform: scale(1.2);
|
||
|
width: 100%;
|
||
|
`);
|
||
|
|
||
|
const cssTutorialScreenshotImg = styled('img', `
|
||
|
width: 100%;
|
||
|
opacity: 0.4;
|
||
|
`);
|
||
|
|
||
|
const cssRoundButton = styled('div', `
|
||
|
width: 75px;
|
||
|
height: 75px;
|
||
|
flex-shrink: 0;
|
||
|
border-radius: 100px;
|
||
|
background: ${colors.lightGreen};
|
||
|
display: flex;
|
||
|
align-items: center;
|
||
|
justify-content: center;
|
||
|
--icon-color: var(--light, #FFF);
|
||
|
|
||
|
.${cssScreenshot.className}:hover & {
|
||
|
background: ${colors.darkGreen};
|
||
|
}
|
||
|
`);
|
||
|
|
||
|
const cssStepper = styled('div', `
|
||
|
display: flex;
|
||
|
justify-content: center;
|
||
|
text-align: center;
|
||
|
font-size: 14px;
|
||
|
font-style: normal;
|
||
|
font-weight: 700;
|
||
|
line-height: 20px;
|
||
|
text-transform: uppercase;
|
||
|
`);
|
||
|
|
||
|
const cssTutorialButton = styled(bigPrimaryButtonLink, `
|
||
|
.${cssScreenshot.className}:hover & {
|
||
|
background-color: ${theme.controlPrimaryHoverBg};
|
||
|
border-color: ${theme.controlPrimaryHoverBg};
|
||
|
}
|
||
|
`);
|