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