Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(editor-ui): onboarding welcome modal and hook #4363

Open
wants to merge 8 commits into
base: staging
Choose a base branch
from
Open
2 changes: 2 additions & 0 deletions packages/editor/src/core/editor.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import {
debouncedStoreToLocalStorage,
getStateFromLocalStorage,
} from '@editor/editor-ui/save/local-storage-notice'
import { WelcomeModal } from '@editor/editor-ui/welcome-modal/welcome-modal'
import { getEditorVersion } from '@editor/package/editor-version'
import { cn } from '@editor/utils/cn'
import { useState, useMemo } from 'react'
Expand Down Expand Up @@ -52,6 +53,7 @@ export function Editor(props: EditorProps) {
{/* For non serlo environments, we need to render the toaster
(already gets rendered in the web project) */}
{!isSerlo ? <Toaster /> : null}
<WelcomeModal />
<div
className={cn(
'editor-core mb-24 text-lg leading-cozy',
Expand Down
36 changes: 36 additions & 0 deletions packages/editor/src/editor-ui/welcome-modal/use-welcome-modal.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { fold } from 'fp-ts/lib/Either'
import { pipe } from 'fp-ts/lib/function'
import * as t from 'io-ts'
import { useEffect, useState } from 'react'

const localStorageKey = 'hasUserSeenWelcomeModal'
elbotho marked this conversation as resolved.
Show resolved Hide resolved

export const useWelcomeModal = () => {
const [isOpen, setIsOpen] = useState(false)

useEffect(() => {
const localStorageValue = localStorage.getItem(localStorageKey)
if (localStorageValue === null) return setIsOpen(true)
const hasSeenModal = decodeWelcomeModalData(localStorageValue)
Comment on lines +12 to +14
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you don't reuse the preferences:
Could we simplify this by just checking if the key exists?
That way we wouldn't need any type checks and parsing.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We could, but I like the idea of decoding any value coming from localStorage, think it's a good practice.

Also, if we later want to expand on this in any way, for example using logged-in user ID to check if a specific user has seen the modal, in case of shared school devices, we can easily expand the codec.

Don't get me wrong, if you implemented it by just checking if the key exists, I wouldn't mind it. But since we already have this code, I wouldn't remove it.

if (!hasSeenModal) setIsOpen(true)
}, [])

const handleClose = () => {
setIsOpen(false)
localStorage.setItem(localStorageKey, 'true')
}

return { isOpen, onClose: handleClose }
}

const WelcomeModalDataCodec = t.boolean

function decodeWelcomeModalData(input: string) {
return pipe(
WelcomeModalDataCodec.decode(JSON.parse(input)),
fold(
() => false,
(decoded) => decoded
)
)
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { cn } from '@editor/utils/cn'

interface WelcomeModalButtonProps {
isActive: boolean
onClick: () => void
}

export function WelcomeModalButton(props: WelcomeModalButtonProps) {
const { isActive, onClick } = props

return (
<button
className={cn(
'h-4 w-4 rounded-full transition-all',
isActive ? 'w-8 bg-brand-600' : 'border-2 border-brand-600'
)}
onClick={onClick}
/>
)
}
105 changes: 105 additions & 0 deletions packages/editor/src/editor-ui/welcome-modal/welcome-modal.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
import { EditorModal } from '@editor/editor-ui/editor-modal'
import { faCircleQuestion } from '@fortawesome/free-regular-svg-icons'
import { faArrowCircleRight } from '@fortawesome/free-solid-svg-icons'
import { useState } from 'react'

import { FaIcon } from '../fa-icon'
import { useWelcomeModal } from './use-welcome-modal'
import { WelcomeModalButton } from './welcome-modal-button'

export function WelcomeModal() {
const { isOpen, onClose } = useWelcomeModal()
const [currentStep, setCurrentStep] = useState(1)

function handleNextButtonClick() {
setCurrentStep((previousValue) => {
if (previousValue < steps.length) return previousValue + 1
return onClose() ?? 1
})
}

return (
<EditorModal
// width should match slideWidth const
className="top-1/2 flex w-[600px] max-w-[95%] flex-col gap-16 px-0 pt-20"
extraTitleClassName="sr-only"
title="Herzlich Willkommen beim Serlo Editor"
isOpen={isOpen}
setIsOpen={(isOpen) => !isOpen && onClose()}
>
<div className="flex h-[400px] overflow-hidden">
<div
className="flex items-center transition-transform"
style={{
transform: `translateX(${translateXValuesMap[currentStep]})`,
}}
>
<div {...slideProps}>
<h1 className="serlo-h1">
Herzlich Willkommen! <br />
<span className="text-brand-600">beim Serlo Editor</span>
</h1>
<p className="serlo-p">
Der Serlo Editor hilft Dir,{' '}
<b>Texte, Bilder und interaktive Aufgaben</b>
direkt zu bearbeiten und sofort sehen, wie sie später aussehen.
hejtful marked this conversation as resolved.
Show resolved Hide resolved
</p>
<p className="serlo-p">
So kannst Du Lernmaterialien <b>einfach</b> erstellen,{' '}
<b>ohne technische Vorkenntnisse</b> zu benötigen.
</p>
</div>

<div {...slideProps}>
<div className="px-4">
<video controls>
<source
src="https://storage.googleapis.com/assets.serlo.org/wikimedia/Der_Menstruationszyklus.webm"
type="video/webm"
/>
</video>
</div>
<p className="serlo-p mb-0 mt-4">
Weitere <b>Erklärungen und Videos</b> findest du jeweils in der
Toolbar der einzelnen Plugins. Einfach auf das{' '}
<FaIcon icon={faCircleQuestion} /> klicken.
hejtful marked this conversation as resolved.
Show resolved Hide resolved
</p>
</div>
</div>
</div>

<div className="flex items-center justify-between px-8">
<div className="flex items-center gap-1">
{steps.map((step) => (
<WelcomeModalButton
key={step}
isActive={step === currentStep}
onClick={() => setCurrentStep(step)}
/>
))}
</div>
<button
className="serlo-button-learner-primary"
onClick={handleNextButtonClick}
>
{currentStep === steps.length ? "Los geht's!" : 'Weiter'}{' '}
<FaIcon icon={faArrowCircleRight} />
</button>
</div>
</EditorModal>
)
}

const steps = [1, 2]

const slideWidth = 600

const slideProps = {
style: { width: slideWidth },
className: 'px-4',
}

const translateXValuesMap: Record<number, string> = {
hejtful marked this conversation as resolved.
Show resolved Hide resolved
1: '0',
2: `-${slideWidth}px`,
}
Loading