elk/components/publish/PublishWidget.vue

401 lines
13 KiB
Vue
Raw Normal View History

2022-11-21 06:55:31 +00:00
<script setup lang="ts">
import type { Attachment, CreateStatusParams, Status, StatusVisibility } from 'masto'
import { fileOpen } from 'browser-fs-access'
import { useDropZone } from '@vueuse/core'
import { EditorContent } from '@tiptap/vue-3'
import ISO6391 from 'iso-639-1'
2022-12-13 14:03:30 +00:00
import type { Draft } from '~/types'
2022-11-21 06:55:31 +00:00
type FileUploadError = [filename: string, message: string]
2022-11-21 06:55:31 +00:00
const {
draftKey,
initial = getDefaultDraft() as never /* Bug of vue-core */,
expanded: _expanded = false,
2022-11-30 04:50:29 +00:00
placeholder,
dialogLabelledBy,
2022-11-21 06:55:31 +00:00
} = defineProps<{
draftKey: string
initial?: () => Draft
2022-11-21 06:55:31 +00:00
placeholder?: string
inReplyToId?: string
inReplyToVisibility?: StatusVisibility
expanded?: boolean
dialogLabelledBy?: string
2022-11-21 06:55:31 +00:00
}>()
const emit = defineEmits<{
(evt: 'published', status: Status): void
}>()
2022-12-01 07:24:35 +00:00
2022-11-30 04:50:29 +00:00
const { t } = useI18n()
// eslint-disable-next-line prefer-const
let { draft, isEmpty } = $(useDraft(draftKey, initial))
2022-11-21 06:55:31 +00:00
let isSending = $ref(false)
2022-11-29 11:57:05 +00:00
let isExpanded = $ref(false)
const shouldExpanded = $computed(() => _expanded || isExpanded || !isEmpty)
2022-11-24 06:54:54 +00:00
const { editor } = useTiptap({
2022-11-25 18:10:17 +00:00
content: computed({
get: () => draft.params.status,
set: newVal => draft.params.status = newVal,
}),
placeholder: computed(() => placeholder ?? draft.params.inReplyToId ? t('placeholder.replying') : t('placeholder.default_1')),
2022-11-29 11:57:05 +00:00
autofocus: shouldExpanded,
2022-11-25 14:07:31 +00:00
onSubmit: publish,
onFocus() {
if (!isExpanded && draft.initialText) {
editor.value?.chain().insertContent(`${draft.initialText} `).focus('end').run()
draft.initialText = ''
}
isExpanded = true
},
onPaste: handlePaste,
})
2022-11-24 09:15:58 +00:00
const currentVisibility = $computed(() => {
2022-11-25 16:17:15 +00:00
return STATUS_VISIBILITIES.find(v => v.value === draft.params.visibility) || STATUS_VISIBILITIES[0]
2022-11-24 09:15:58 +00:00
})
2022-11-23 17:17:54 +00:00
let isUploading = $ref<boolean>(false)
let isExceedingAttachmentLimit = $ref<boolean>(false)
let failed = $ref<FileUploadError[]>([])
2022-11-23 17:17:54 +00:00
async function handlePaste(evt: ClipboardEvent) {
const files = evt.clipboardData?.files
2022-11-24 08:20:21 +00:00
if (!files || files.length === 0)
2022-11-23 17:17:54 +00:00
return
2022-11-24 04:05:13 +00:00
evt.preventDefault()
2022-11-23 17:17:54 +00:00
await uploadAttachments(Array.from(files))
}
2022-12-27 19:13:50 +00:00
function insertEmoji(name: string) {
editor.value?.chain().focus().insertEmoji(name).run()
2022-12-23 19:15:19 +00:00
}
2022-12-27 19:13:50 +00:00
function insertCustomEmoji(image: any) {
editor.value?.chain().focus().insertCustomEmoji(image).run()
}
2022-12-23 19:15:19 +00:00
2022-11-23 17:17:54 +00:00
async function pickAttachments() {
const mimeTypes = currentInstance.value!.configuration.mediaAttachments.supportedMimeTypes
const files = await fileOpen({
description: 'Attachments',
multiple: true,
mimeTypes,
})
2022-11-23 17:17:54 +00:00
await uploadAttachments(files)
}
async function toggleSensitive() {
draft.params.sensitive = !draft.params.sensitive
}
const masto = useMasto()
2022-11-23 17:17:54 +00:00
async function uploadAttachments(files: File[]) {
isUploading = true
failed = []
// TODO: display some kind of message if too many media are selected
// DONE
const limit = currentInstance.value!.configuration.statuses.maxMediaAttachments || 4
for (const file of files.slice(0, limit)) {
if (draft.attachments.length < limit) {
isExceedingAttachmentLimit = false
try {
const attachment = await masto.mediaAttachments.create({
file,
})
draft.attachments.push(attachment)
}
catch (e) {
// TODO: add some human-readable error message, problem is that masto api will not return response code
console.error(e)
failed = [...failed, [file.name, (e as Error).message]]
}
}
else {
isExceedingAttachmentLimit = true
failed = [...failed, [file.name, t('state.attachments_limit_error')]]
}
2022-11-23 17:17:54 +00:00
}
isUploading = false
}
2022-12-14 23:30:54 +00:00
async function setDescription(att: Attachment, description: string) {
att.description = description
await masto.mediaAttachments.update(att.id, { description: att.description })
2022-12-14 23:30:54 +00:00
}
2022-11-24 04:05:13 +00:00
function removeAttachment(index: number) {
2022-11-24 06:54:54 +00:00
draft.attachments.splice(index, 1)
2022-11-23 17:17:54 +00:00
}
2022-11-21 06:55:31 +00:00
2022-11-24 09:15:58 +00:00
function chooseVisibility(visibility: StatusVisibility) {
draft.params.visibility = visibility
}
function chooseLanguage(language: string | null) {
draft.params.language = language
}
2022-11-21 06:55:31 +00:00
async function publish() {
2022-11-25 16:17:15 +00:00
const payload = {
...draft.params,
status: htmlToText(draft.params.status || ''),
mediaIds: draft.attachments.map(a => a.id),
} as CreateStatusParams
if (process.dev) {
2022-11-25 16:17:15 +00:00
// eslint-disable-next-line no-console
console.info({
raw: draft.params.status,
...payload,
})
// eslint-disable-next-line no-alert
2022-11-25 16:17:15 +00:00
const result = confirm('[DEV] Payload logged to console, do you want to publish it?')
if (!result)
return
}
2022-11-25 16:17:15 +00:00
2022-11-21 06:55:31 +00:00
try {
isSending = true
2022-11-25 16:17:15 +00:00
let status: Status
2022-11-24 11:35:26 +00:00
if (!draft.editingStatus)
status = await masto.statuses.create(payload)
else
status = await masto.statuses.update(draft.editingStatus.id, payload)
2022-11-24 11:35:26 +00:00
draft = initial()
emit('published', status)
2022-11-21 06:55:31 +00:00
}
finally {
isSending = false
}
}
const dropZoneRef = ref<HTMLDivElement>()
async function onDrop(files: File[] | null) {
if (files)
await uploadAttachments(files)
}
const { isOverDropZone } = useDropZone(dropZoneRef, onDrop)
defineExpose({
focusEditor: () => {
editor.value?.commands?.focus?.()
},
})
2022-11-21 06:55:31 +00:00
</script>
<template>
<div v-if="isMastoInitialised && currentUser" flex="~ col gap-4" py3 px2 sm:px4>
2022-11-24 11:35:26 +00:00
<template v-if="draft.editingStatus">
<div flex="~ col gap-1">
<div id="state-editing" text-secondary self-center>
2022-11-29 06:57:32 +00:00
{{ $t('state.editing') }}
2022-11-24 11:35:26 +00:00
</div>
2022-12-02 07:02:44 +00:00
<StatusCard :status="draft.editingStatus" :actions="false" :hover="false" px-0 />
2022-11-24 07:53:27 +00:00
</div>
2022-11-24 11:35:26 +00:00
<div border="b dashed gray/40" />
</template>
2022-11-24 14:32:20 +00:00
2022-12-27 19:13:50 +00:00
<div flex gap-3 flex-1>
<NuxtLink :to="getAccountRoute(currentUser.account)">
<AccountBigAvatar :account="currentUser.account" />
2022-11-24 15:19:18 +00:00
</NuxtLink>
<!-- This `w-0` style is used to avoid overflow problems in flex layoutsso don't remove it unless you know what you're doing -->
2022-11-24 11:35:26 +00:00
<div
ref="dropZoneRef"
flex w-0 flex-col gap-3 flex-1
border="2 dashed transparent"
2022-11-25 09:31:32 +00:00
:class="[isSending ? 'pointer-events-none' : '', isOverDropZone ? '!border-primary' : '']"
2022-11-24 11:35:26 +00:00
>
<div v-if="draft.params.sensitive">
<input
v-model="draft.params.spoilerText"
type="text"
:placeholder="$t('placeholder.content_warning')"
p2 border-rounded w-full bg-transparent
outline-none border="~ base"
>
</div>
<div relative flex-1 flex flex-col>
2022-11-25 13:29:42 +00:00
<EditorContent
:editor="editor"
2022-12-14 13:22:35 +00:00
flex max-w-full
:class="shouldExpanded ? 'min-h-30 md:max-h-[calc(100vh-200px)] sm:max-h-[calc(100vh-400px)] max-h-35 of-y-auto overscroll-contain' : ''"
2022-11-25 13:29:42 +00:00
/>
</div>
2022-11-24 07:53:27 +00:00
<div v-if="isUploading" flex gap-1 items-center text-sm p1 text-primary>
<div i-ri:loader-2-fill animate-spin />
2022-11-29 06:57:32 +00:00
{{ $t('state.uploading') }}
</div>
<div
v-else-if="failed.length > 0"
role="alert"
:aria-describedby="isExceedingAttachmentLimit ? 'upload-failed uploads-per-post' : 'upload-failed'"
flex="~ col"
gap-1 text-sm
pt-1 ps-2 pe-1 pb-2
text-red-600 dark:text-red-400
border="~ base rounded red-600 dark:red-400"
>
<head id="upload-failed" flex justify-between>
<div flex items-center gap-x-2 font-bold>
<div aria-hidden="true" i-ri:error-warning-fill />
<p>{{ $t('state.upload_failed') }}</p>
</div>
<CommonTooltip placement="bottom" :content="$t('action.clear_upload_failed')">
<button
flex rounded-4 p1
hover:bg-active cursor-pointer transition-100
:aria-label="$t('action.clear_upload_failed')"
@click="failed = []"
>
<span aria-hidden="true" w-1.75em h-1.75em i-ri:close-line />
</button>
</CommonTooltip>
</head>
<div v-if="isExceedingAttachmentLimit" id="uploads-per-post" ps-2 sm:ps-1 text-small>
{{ $t('state.attachments_exceed_server_limit') }}
</div>
<ol ps-2 sm:ps-1>
<li v-for="error in failed" :key="error[0]" flex="~ col sm:row" gap-y-1 sm:gap-x-2>
<strong>{{ error[1] }}:</strong>
<span>{{ error[0] }}</span>
</li>
</ol>
</div>
<div v-if="draft.attachments.length" flex="~ col gap-2" overflow-auto>
2022-11-24 11:35:26 +00:00
<PublishAttachment
v-for="(att, idx) in draft.attachments" :key="att.id"
:attachment="att"
:dialog-labelled-by="dialogLabelledBy ?? (draft.editingStatus ? 'state-editing' : null)"
2022-11-24 11:35:26 +00:00
@remove="removeAttachment(idx)"
2022-12-14 23:30:54 +00:00
@set-description="setDescription(att, $event)"
2022-11-24 11:35:26 +00:00
/>
</div>
2022-12-02 07:02:44 +00:00
</div>
</div>
<div flex gap-4>
<div w-12 h-full sm:block hidden />
<div
v-if="shouldExpanded" flex="~ gap-1 1 wrap" m="s--1" pt-2 justify="between" max-w-full
2022-12-02 07:02:44 +00:00
border="t base"
>
2022-12-27 19:13:50 +00:00
<PublishEmojiPicker
@select="insertEmoji"
@select-custom="insertCustomEmoji"
/>
2022-12-23 19:15:19 +00:00
2022-12-02 07:02:44 +00:00
<CommonTooltip placement="bottom" :content="$t('tooltip.add_media')">
<button btn-action-icon :aria-label="$t('tooltip.add_media')" @click="pickAttachments">
<div i-ri:image-add-line />
</button>
</CommonTooltip>
<template v-if="editor">
<CommonTooltip placement="bottom" :content="$t('tooltip.toggle_code_block')">
<button
btn-action-icon
:aria-label="$t('tooltip.toggle_code_block')"
2022-12-27 20:42:58 +00:00
:class="editor.isActive('codeBlock') ? 'text-primary' : ''"
2022-12-02 07:02:44 +00:00
@click="editor?.chain().focus().toggleCodeBlock().run()"
>
<div i-ri:code-s-slash-line />
2022-11-24 15:48:52 +00:00
</button>
</CommonTooltip>
2022-12-02 07:02:44 +00:00
</template>
2022-11-24 09:15:58 +00:00
2022-12-02 07:02:44 +00:00
<div flex-auto />
<div dir="ltr" pointer-events-none pe-1 pt-2 text-sm tabular-nums text-secondary flex gap-0.5>
{{ editor?.storage.characterCount.characters() }}<span text-secondary-light>/</span><span text-secondary-light>{{ characterLimit }}</span>
</div>
2022-12-02 07:02:44 +00:00
<CommonTooltip placement="bottom" :content="$t('tooltip.add_content_warning')">
<button btn-action-icon :aria-label="$t('tooltip.add_content_warning')" @click="toggleSensitive">
<div v-if="draft.params.sensitive" i-ri:alarm-warning-fill text-orange />
<div v-else i-ri:alarm-warning-line />
</button>
</CommonTooltip>
2022-11-24 13:26:33 +00:00
<CommonTooltip placement="top" :content="$t('tooltip.change_language')">
<CommonDropdown placement="bottom">
2023-01-02 04:52:14 +00:00
<button btn-action-icon :aria-label="$t('tooltip.change_language')" w-12 mr--1>
<div i-ri:translate-2 />
<div i-ri:arrow-down-s-line text-sm text-secondary me--1 />
</button>
<template #popper>
2023-01-02 04:52:14 +00:00
<div min-w-80>
<!-- TODO search lang -->
<!-- <input
placeholder="Search"
p2 mb2 border-rounded w-full bg-transparent
outline-none border="~ base"
> -->
<div max-h-40vh overflow-auto>
<CommonDropdownItem
v-for="code in [null, ...ISO6391.getAllCodes()]"
:key="code"
:checked="code === (draft.params.language || null)"
@click="chooseLanguage(code)"
>
{{ code ? ISO6391.getNativeName(code) : 'None' }}
<template #description>
<template v-if="code">
{{ ISO6391.getName(code) }}
</template>
</template>
</CommonDropdownItem>
</div>
</div>
</template>
</CommonDropdown>
</CommonTooltip>
2022-12-02 07:02:44 +00:00
<CommonTooltip placement="bottom" :content="$t('tooltip.change_content_visibility')">
<CommonDropdown>
<button :aria-label="$t('tooltip.change_content_visibility')" btn-action-icon w-12>
<div :class="currentVisibility.icon" />
<div i-ri:arrow-down-s-line text-sm text-secondary me--1 />
</button>
2022-11-24 11:35:26 +00:00
2022-12-02 07:02:44 +00:00
<template #popper>
<CommonDropdownItem
v-for="visibility in STATUS_VISIBILITIES"
:key="visibility.value"
:icon="visibility.icon"
:checked="visibility.value === draft.params.visibility"
@click="chooseVisibility(visibility.value)"
>
{{ $t(`visibility.${visibility.value}`) }}
<template #description>
{{ $t(`visibility.${visibility.value}_desc`) }}
</template>
</CommonDropdownItem>
</template>
</CommonDropdown>
</CommonTooltip>
<button
btn-solid rounded-full text-sm w-full md:w-fit
2022-12-02 07:02:44 +00:00
:disabled="isEmpty || isUploading || (draft.attachments.length === 0 && !draft.params.status)"
@click="publish"
>
{{ !draft.editingStatus ? $t('action.publish') : $t('action.save_changes') }}
</button>
2022-11-24 07:53:27 +00:00
</div>
2022-11-21 06:55:31 +00:00
</div>
</div>
</template>