webapp/src/components/Views/PublishSettings/PublishSettings.tsx

306 lines
11 KiB
TypeScript
Raw Normal View History

2024-09-15 16:41:02 +00:00
import { useNavigate } from '@solidjs/router'
import { clsx } from 'clsx'
2024-02-04 11:25:21 +00:00
import { Show, createEffect, createMemo, createSignal, lazy, onMount } from 'solid-js'
import { createStore } from 'solid-js/store'
import { Button } from '~/components/_shared/Button'
import { Icon } from '~/components/_shared/Icon'
import { Image } from '~/components/_shared/Image'
import { ShoutForm, useEditorContext } from '~/context/editor'
import { useLocalize } from '~/context/localize'
import { useSession } from '~/context/session'
import { useTopics } from '~/context/topics'
2024-06-24 17:50:27 +00:00
import { useSnackbar, useUI } from '~/context/ui'
import { Topic } from '~/graphql/schema/core.gen'
2024-09-15 16:41:02 +00:00
import { UploadedFile } from '~/types/upload'
2024-02-04 11:25:21 +00:00
import { TopicSelect, UploadModalContent } from '../../Editor'
2024-07-21 02:17:42 +00:00
import { Modal } from '../../_shared/Modal'
import stylesBeside from '../../Feed/Beside.module.scss'
2024-02-04 11:25:21 +00:00
import styles from './PublishSettings.module.scss'
const SimplifiedEditor = lazy(() => import('../../Editor/SimplifiedEditor'))
const GrowingTextarea = lazy(() => import('~/components/_shared/GrowingTextarea/GrowingTextarea'))
const DESCRIPTION_MAX_LENGTH = 400
type Props = {
shoutId: number
form: ShoutForm
}
const shorten = (str: string, maxLen: number) => {
if (str.length <= maxLen) return str
const result = str.slice(0, Math.max(0, str.lastIndexOf(' ', maxLen))).trim()
return `${result}...`
}
2024-01-31 12:34:15 +00:00
const EMPTY_TOPIC: Topic = {
id: -1,
2024-06-26 08:22:05 +00:00
slug: ''
2024-01-31 12:34:15 +00:00
}
2024-05-24 14:59:15 +00:00
interface FormConfig {
coverImageUrl?: string
mainTopic?: Topic
slug?: string
title?: string
subtitle?: string
description?: string
selectedTopics?: Topic[]
}
const emptyConfig: FormConfig = {
2024-01-31 12:34:15 +00:00
coverImageUrl: '',
mainTopic: EMPTY_TOPIC,
slug: '',
title: '',
subtitle: '',
description: '',
2024-06-26 08:22:05 +00:00
selectedTopics: []
2024-01-31 12:34:15 +00:00
}
2023-12-03 08:44:11 +00:00
export const PublishSettings = (props: Props) => {
const { t } = useLocalize()
2024-06-24 17:50:27 +00:00
const { showModal, hideModal } = useUI()
const navigate = useNavigate()
const { session } = useSession()
2024-05-06 23:44:25 +00:00
const { sortedTopics } = useTopics()
2024-02-05 19:05:49 +00:00
const { showSnackbar } = useSnackbar()
2024-01-23 19:01:41 +00:00
const [topics, setTopics] = createSignal<Topic[]>(sortedTopics())
const composeDescription = () => {
if (!props.form.description) {
2024-05-10 12:44:36 +00:00
const cleanFootnotes = props.form.body.replaceAll(/<footnote data-value=".*?">(.*?)<\/footnote>/g, '')
const leadText = cleanFootnotes.replaceAll(/<\/?[^>]+(>|$)/gi, ' ')
return shorten(leadText, DESCRIPTION_MAX_LENGTH).trim()
}
return props.form.description
}
2024-01-31 12:34:15 +00:00
const initialData = createMemo(() => {
return {
coverImageUrl: props.form?.coverImageUrl,
mainTopic: props.form?.mainTopic || EMPTY_TOPIC,
2024-02-17 15:03:01 +00:00
slug: props.form?.slug || '',
title: props.form?.title || '',
subtitle: props.form?.subtitle || '',
description: composeDescription() || '',
2024-06-26 08:22:05 +00:00
selectedTopics: []
2024-01-31 12:34:15 +00:00
}
})
2024-05-24 14:59:15 +00:00
const [settingsForm, setSettingsForm] = createStore<FormConfig>(emptyConfig)
2024-01-31 12:34:15 +00:00
onMount(() => {
setSettingsForm(initialData())
})
createEffect(() => setTopics(sortedTopics()))
2024-02-04 17:40:15 +00:00
const { formErrors, setForm, setFormErrors, saveShout, publishShout } = useEditorContext()
2024-06-24 17:50:27 +00:00
const handleUploadModalContentCloseSetCover = (image: UploadedFile | undefined) => {
hideModal()
2024-06-24 17:50:27 +00:00
setSettingsForm('coverImageUrl', image?.url)
}
const handleDeleteCoverImage = () => {
setSettingsForm('coverImageUrl', '')
}
2024-05-24 14:59:15 +00:00
const handleTopicSelectChange = (newSelectedTopics: Topic[]) => {
if (
props.form.selectedTopics.length === 0 ||
2024-05-24 14:59:15 +00:00
newSelectedTopics.every((topic: Topic) => topic.id !== props.form.mainTopic?.id)
) {
2024-06-24 17:50:27 +00:00
setSettingsForm((prev) => {
return {
...prev,
2024-06-26 08:22:05 +00:00
mainTopic: newSelectedTopics[0]
}
})
}
if (newSelectedTopics.length > 0) {
setFormErrors('selectedTopics', '')
}
setForm('selectedTopics', newSelectedTopics)
}
const handleBackClick = () => {
2024-06-24 17:50:27 +00:00
navigate(`/edit/${props.shoutId}`)
}
const handleCancelClick = () => {
2024-01-31 12:34:15 +00:00
setSettingsForm(initialData())
handleBackClick()
}
const handlePublishSubmit = () => {
2024-02-05 19:05:49 +00:00
const shoutData = { ...props.form, ...settingsForm }
2024-05-05 16:13:48 +00:00
if (shoutData?.mainTopic) {
2024-02-05 19:05:49 +00:00
publishShout(shoutData)
2024-05-05 16:13:48 +00:00
} else {
showSnackbar({ body: t('Please, set the main topic first') })
2024-02-05 19:05:49 +00:00
}
}
const handleSaveDraft = () => {
saveShout({ ...props.form, ...settingsForm })
}
2024-07-13 11:42:53 +00:00
const removeSpecial = (ev: InputEvent) => {
const input = ev.target as HTMLInputElement
const value = input.value
const newValue = value.startsWith('@') || value.startsWith('!') ? value.substring(1) : value
input.value = newValue
}
return (
2023-08-15 10:24:08 +00:00
<form class={clsx(styles.PublishSettings, 'inputs-wrapper')}>
<div class="wide-container">
<div class="row">
<div class="col-md-19 col-lg-18 col-xl-16 offset-md-5">
<div>
<button type="button" class={styles.goBack} onClick={handleBackClick}>
<Icon name="arrow-left" class={stylesBeside.icon} />
{t('Back to editor')}
</button>
</div>
2023-08-15 10:24:08 +00:00
<h1>{t('Publish Settings')}</h1>
<h4>{t('Material card')}</h4>
<div class={styles.articlePreview}>
<div class={styles.actions}>
<Button
variant="primary"
onClick={() => showModal('uploadCoverImage')}
value={settingsForm.coverImageUrl ? t('Add another image') : t('Add image')}
/>
<Show when={settingsForm.coverImageUrl}>
<Button variant="secondary" onClick={handleDeleteCoverImage} value={t('Delete cover')} />
</Show>
</div>
<div
class={clsx(styles.shoutCardCoverContainer, {
2024-06-26 08:22:05 +00:00
[styles.hasImage]: settingsForm.coverImageUrl
2023-08-15 10:24:08 +00:00
})}
>
2024-01-31 12:34:15 +00:00
<Show when={settingsForm.coverImageUrl ?? initialData().coverImageUrl}>
2023-08-15 10:24:08 +00:00
<div class={styles.shoutCardCover}>
2024-01-31 12:34:15 +00:00
<Image src={settingsForm.coverImageUrl} alt={initialData().title} width={800} />
2023-08-15 10:24:08 +00:00
</div>
</Show>
<div class={styles.text}>
<Show when={settingsForm.mainTopic}>
2024-06-24 17:50:27 +00:00
<div class={styles.mainTopic}>{settingsForm.mainTopic?.title || ''}</div>
2023-08-15 10:24:08 +00:00
</Show>
<div class={styles.shoutCardTitle}>{settingsForm.title}</div>
2024-02-17 15:13:54 +00:00
<div class={styles.shoutCardSubtitle}>{settingsForm.subtitle || ''}</div>
2024-06-24 17:50:27 +00:00
<div class={styles.shoutAuthor}>
{session()?.user?.app_data?.profile?.name || t('Anonymous')}
</div>
2023-08-15 10:24:08 +00:00
</div>
</div>
</div>
<p class="description">
{t(
2024-06-26 08:22:05 +00:00
'Choose a title image for the article. You can immediately see how the publication card will look like.'
2023-08-15 10:24:08 +00:00
)}
</p>
2023-08-15 10:24:08 +00:00
<div class={styles.commonSettings}>
<GrowingTextarea
class={styles.settingInput}
variant="bordered"
fieldName={t('Header')}
2023-08-15 10:24:08 +00:00
placeholder={t('Come up with a title for your story')}
initialValue={settingsForm.title}
2024-05-24 14:59:15 +00:00
// biome-ignore lint/suspicious/noExplicitAny: <explanation>
value={(value: any) => setSettingsForm('title', value)}
2023-08-15 10:24:08 +00:00
allowEnterKey={false}
maxLength={100}
/>
<GrowingTextarea
class={styles.settingInput}
variant="bordered"
fieldName={t('Subheader')}
2023-08-15 10:24:08 +00:00
placeholder={t('Come up with a subtitle for your story')}
2024-02-17 15:13:54 +00:00
initialValue={settingsForm.subtitle || ''}
2024-05-24 14:59:15 +00:00
// biome-ignore lint/suspicious/noExplicitAny: <explanation>
value={(value: any) => setSettingsForm('subtitle', value)}
2023-08-15 10:24:08 +00:00
allowEnterKey={false}
maxLength={100}
/>
<SimplifiedEditor
2023-08-15 10:24:08 +00:00
variant="bordered"
onlyBubbleControls={true}
smallHeight={true}
2023-08-15 10:24:08 +00:00
placeholder={t('Write a short introduction')}
label={t('Description')}
initialContent={composeDescription()}
2024-05-24 14:59:15 +00:00
// biome-ignore lint/suspicious/noExplicitAny: <explanation>
onChange={(value: any) => setForm('description', value)}
maxLength={DESCRIPTION_MAX_LENGTH}
2023-08-15 10:24:08 +00:00
/>
</div>
2023-08-15 10:24:08 +00:00
<h4>{t('Slug')}</h4>
<div class="pretty-form__item">
2024-07-13 11:42:53 +00:00
<input type="text" name="slug" id="slug" value={settingsForm.slug} onInput={removeSpecial} />
2023-08-15 10:24:08 +00:00
<label for="slug">{t('Slug')}</label>
</div>
<h4>{t('Topics')}</h4>
<p class="description">
{t(
2024-06-26 08:22:05 +00:00
'Add a few topics so that the reader knows what your content is about and can find it on pages of topics that interest them. Topics can be swapped, the first topic becomes the title'
2023-08-15 10:24:08 +00:00
)}
</p>
<div class={styles.inputContainer}>
<div class={clsx('pretty-form__item', styles.topicSelectContainer)}>
2024-01-23 19:01:41 +00:00
<Show when={topics().length > 0}>
2023-08-15 10:24:08 +00:00
<TopicSelect
2024-01-23 19:01:41 +00:00
topics={topics()}
2023-08-15 10:24:08 +00:00
onChange={handleTopicSelectChange}
selectedTopics={props.form.selectedTopics}
onMainTopicChange={(mainTopic) => setForm('mainTopic', mainTopic)}
mainTopic={props.form.mainTopic}
/>
</Show>
</div>
<Show when={formErrors.selectedTopics}>
<div class={styles.validationError}>{formErrors.selectedTopics}</div>
</Show>
</div>
<h4>{t('Collaborators')}</h4>
<Button
variant="primary"
onClick={() => showModal('inviteMembers')}
value={t('Invite collaborators')}
/>
2023-08-15 10:24:08 +00:00
</div>
</div>
</div>
<div class={styles.formActions}>
2023-08-15 10:24:08 +00:00
<div class="wide-container">
<div class="row">
<div class="col-md-19 col-lg-18 col-xl-16 offset-md-5">
<div class={styles.content}>
<Button
variant="light"
value={t('Cancel changes')}
class={styles.cancel}
onClick={handleCancelClick}
/>
<Button variant="secondary" onClick={handleSaveDraft} value={t('Save draft')} />
<Button onClick={handlePublishSubmit} variant="primary" value={t('Publish')} />
</div>
</div>
</div>
</div>
</div>
<Modal variant="narrow" name="uploadCoverImage">
2024-06-24 17:50:27 +00:00
<UploadModalContent
onClose={(value: UploadedFile | undefined) =>
handleUploadModalContentCloseSetCover(value as UploadedFile)
}
/>
</Modal>
2023-08-15 10:24:08 +00:00
</form>
)
}