elk/components/publish/PublishWidget.vue

258 lines
7.3 KiB
Vue
Raw Normal View History

2022-11-21 06:55:31 +00:00
<script setup lang="ts">
2022-11-24 11:35:26 +00:00
import type { CreateStatusParams, StatusVisibility } from 'masto'
import { fileOpen } from 'browser-fs-access'
import { useDropZone } from '@vueuse/core'
import { EditorContent } from '@tiptap/vue-3'
2022-11-25 13:29:42 +00:00
import { POST_CHARS_LIMIT } from '~~/constants'
2022-11-21 06:55:31 +00:00
const {
draftKey,
placeholder = 'What is on your mind?',
inReplyToId,
expanded: _expanded = false,
2022-11-21 06:55:31 +00:00
} = defineProps<{
draftKey: string
placeholder?: string
inReplyToId?: string
expanded?: boolean
2022-11-21 06:55:31 +00:00
}>()
let isExpanded = $ref(_expanded)
2022-11-21 06:55:31 +00:00
let isSending = $ref(false)
2022-11-24 11:35:26 +00:00
let { draft } = $(useDraft(draftKey, inReplyToId))
2022-11-24 06:54:54 +00:00
const { editor } = useTiptap({
content: toRef(draft.params, 'status'),
placeholder,
autofocus: isExpanded,
onSubimit: publish,
onFocus() { isExpanded = true },
onPaste: handlePaste,
})
2022-11-23 17:17:54 +00:00
const status = $computed(() => {
return {
2022-11-24 06:54:54 +00:00
...draft.params,
mediaIds: draft.attachments.map(a => a.id),
2022-11-23 17:17:54 +00:00
} as CreateStatusParams
})
2022-11-24 09:15:58 +00:00
const currentVisibility = $computed(() => {
2022-11-24 13:26:33 +00:00
return STATUS_VISIBILITIES.find(v => v.value === status.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)
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))
}
async function pickAttachments() {
const files = await fileOpen([
{
description: 'Attachments',
multiple: true,
mimeTypes: ['image/*'],
extensions: ['.png', '.gif', '.jpeg', '.jpg', '.webp', '.avif', '.heic', '.heif'],
},
{
description: 'Attachments',
mimeTypes: ['video/*'],
extensions: ['.webm', '.mp4', '.m4v', '.mov', '.ogv', '.3gp'],
},
{
2022-11-24 04:05:13 +00:00
description: 'Attachments',
mimeTypes: ['audio/*'],
extensions: ['.mp3', '.ogg', '.oga', '.wav', '.flac', '.opus', '.aac', '.m4a', '.3gp', '.wma'],
},
])
2022-11-23 17:17:54 +00:00
await uploadAttachments(files)
}
async function toggleSensitive() {
draft.params.sensitive = !draft.params.sensitive
}
2022-11-23 17:17:54 +00:00
async function uploadAttachments(files: File[]) {
isUploading = true
for (const file of files) {
const attachment = await masto.mediaAttachments.create({
file,
})
2022-11-24 06:54:54 +00:00
draft.attachments.push(attachment)
2022-11-23 17:17:54 +00:00
}
isUploading = false
}
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
}
2022-11-21 06:55:31 +00:00
async function publish() {
if (process.dev) {
alert(JSON.stringify(draft.params, null, 2))
return
}
2022-11-21 06:55:31 +00:00
try {
isSending = true
2022-11-24 11:35:26 +00:00
if (!draft.editingStatus)
await masto.statuses.create(status)
else
await masto.statuses.update(draft.editingStatus.id, status)
2022-11-24 11:35:26 +00:00
2022-11-25 11:39:21 +00:00
draft = getDefaultDraft({ inReplyToId })
2022-11-24 11:35:26 +00:00
isPublishDialogOpen.value = false
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)
2022-11-21 06:55:31 +00:00
onUnmounted(() => {
// Remove draft if it's empty
2022-11-24 06:54:54 +00:00
if (!draft.attachments.length && !draft.params.status) {
2022-11-21 06:55:31 +00:00
nextTick(() => {
2022-11-24 06:54:54 +00:00
delete currentUserDrafts.value[draftKey]
2022-11-21 06:55:31 +00:00
})
}
})
</script>
<template>
2022-11-24 14:32:20 +00:00
<div v-if="currentUser" flex="~ col gap-1">
2022-11-24 11:35:26 +00:00
<template v-if="draft.editingStatus">
<div flex="~ col gap-1">
<div text-gray self-center>
Editing
</div>
<StatusCard :status="draft.editingStatus" :actions="false" :hover="false" />
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-11-24 11:35:26 +00:00
<div p4 flex gap-4>
2022-11-24 15:48:52 +00:00
<NuxtLink w-12 h-12 :to="getAccountPath(currentUser.account)">
2022-11-24 15:19:18 +00:00
<AccountAvatar :account="currentUser.account" w-12 h-12 />
</NuxtLink>
2022-11-24 11:35:26 +00:00
<div
ref="dropZoneRef"
2022-11-24 15:19:18 +00:00
flex 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="Write your warning here"
p2 border-rounded w-full bg-transparent
outline-none border="~ base"
>
</div>
2022-11-25 13:29:42 +00:00
<div relative>
<EditorContent
:editor="editor"
:class="isExpanded ? 'min-h-120px' : ''"
/>
<div v-if="isExpanded" absolute right-0 bottom-0 pointer-events-none text-sm op25>
{{ POST_CHARS_LIMIT - editor?.storage.characterCount.characters() }}
</div>
</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 />
Uploading...
</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"
@remove="removeAttachment(idx)"
/>
</div>
<div
v-if="isExpanded" flex="~ gap-2" m="l--1" pt-2
border="t base"
>
2022-11-24 15:48:52 +00:00
<CommonTooltip placement="bottom" content="Add images, a video or an audio file">
<button btn-action-icon @click="pickAttachments">
<div i-ri:image-add-line />
2022-11-24 15:48:52 +00:00
</button>
</CommonTooltip>
2022-11-24 09:15:58 +00:00
<template v-if="editor">
<CommonTooltip placement="bottom" content="Toggle code block">
<button
btn-action-icon
:class="editor.isActive('codeBlock') ? 'op100' : 'op50'"
@click="editor?.chain().focus().toggleCodeBlock().run()"
>
<div i-ri:code-s-slash-line />
</button>
</CommonTooltip>
</template>
2022-11-24 13:26:33 +00:00
<div flex-auto />
<CommonTooltip placement="bottom" content="Add content warning">
<button btn-action-icon @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 11:35:26 +00:00
<CommonDropdown>
2022-11-24 13:26:33 +00:00
<button btn-action-icon w-12>
2022-11-24 11:35:26 +00:00
<div :class="currentVisibility.icon" />
2022-11-24 13:26:33 +00:00
<div i-ri:arrow-down-s-line text-sm op50 mr--1 />
2022-11-24 11:35:26 +00:00
</button>
<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)"
>
{{ visibility.label }}
<template #description>
{{ visibility.description }}
</template>
</CommonDropdownItem>
</template>
</CommonDropdown>
<button
2022-11-24 13:26:33 +00:00
btn-solid rounded-full text-sm
2022-11-24 11:35:26 +00:00
:disabled="isUploading || (draft.attachments.length === 0 && !draft.params.status)"
@click="publish"
>
{{ !draft.editingStatus ? 'Publish!' : 'Save changes' }}
</button>
</div>
2022-11-24 07:53:27 +00:00
</div>
2022-11-21 06:55:31 +00:00
</div>
</div>
</template>