vikunja/frontend/src/components/tasks/partials/Heading.vue

244 lines
5.6 KiB
Vue

<template>
<div class="heading">
<div class="tw:flex tw:items-center md:tw:items-stretch tw:flex-col tw:gap-1 task-properties">
<div class="tw:flex tw:items-center tw:gap-2">
<ColorBubble
v-if="task.hexColor !== ''"
:color="getHexColor(task.hexColor)"
/>
<BaseButton @click="copyUrl">
<span class="title task-id">
{{ textIdentifier }}
</span>
</BaseButton>
</div>
<Done
:is-done="task.done"
/>
<BaseButton
v-if="hasClose"
:aria-label="$t('task.detail.closeTaskDetail')"
class="close"
@click="$emit('close')"
>
<Icon icon="times" />
</BaseButton>
</div>
<h1
class="title input"
:class="{'disabled': !canWrite}"
:contenteditable="canWrite ? true : undefined"
:spellcheck="false"
@input="handleTitleInput"
@blur="save(($event.target as HTMLInputElement).textContent as string)"
@keydown.enter.prevent.stop="!$event.isComposing && ($event.target as HTMLInputElement).blur()"
@keydown.esc.prevent.stop="!$event.isComposing && cancel($event.target as HTMLInputElement)"
>
{{ task.title.trim() }}
</h1>
<BaseButton
v-if="hasClose"
:aria-label="$t('task.detail.closeTaskDetail')"
class="close"
@click="$emit('close')"
>
<Icon icon="times" />
</BaseButton>
<CustomTransition name="fade">
<span
v-if="loading && saving"
class="is-inline-flex is-align-items-center"
>
<span class="loader is-inline-block mie-2" />
{{ $t('misc.saving') }}
</span>
<span
v-else-if="!loading && showSavedMessage"
class="has-text-success is-inline-flex is-align-content-center"
>
<Icon
icon="check"
class="mie-2"
/>
{{ $t('misc.saved') }}
</span>
</CustomTransition>
</div>
</template>
<script setup lang="ts">
import {ref, computed, onMounted, onBeforeUnmount, watch} from 'vue'
import {useRouter} from 'vue-router'
import BaseButton from '@/components/base/BaseButton.vue'
import CustomTransition from '@/components/misc/CustomTransition.vue'
import ColorBubble from '@/components/misc/ColorBubble.vue'
import Done from '@/components/misc/Done.vue'
import {useCopyToClipboard} from '@/composables/useCopyToClipboard'
import {useTaskStore} from '@/stores/tasks'
import type {ITask} from '@/modelTypes/ITask'
import {getHexColor, getTaskIdentifier} from '@/models/task'
const props = defineProps<{
task: ITask,
canWrite: boolean,
hasClose: boolean,
}>()
const emit = defineEmits<{
'update:task': [task: ITask],
'close': [],
}>()
const router = useRouter()
const copy = useCopyToClipboard()
async function copyUrl() {
const route = router.resolve({name: 'task.detail', query: {taskId: props.task.id}})
const absoluteURL = new URL(route.href, window.location.href).href
await copy(absoluteURL)
}
const taskStore = useTaskStore()
const loading = computed(() => taskStore.isLoading)
const textIdentifier = computed(() => getTaskIdentifier(props.task))
// Since loading is global state, this variable ensures we're only showing the saving icon when saving the description.
const saving = ref(false)
const showSavedMessage = ref(false)
// Track if title has unsaved changes
const titleHasChanges = ref(false)
function handleBeforeUnload(e: BeforeUnloadEvent) {
if (titleHasChanges.value) {
e.preventDefault()
// Modern browsers ignore custom messages but this is still required
e.returnValue = ''
return ''
}
}
onMounted(() => {
window.addEventListener('beforeunload', handleBeforeUnload)
})
onBeforeUnmount(() => {
window.removeEventListener('beforeunload', handleBeforeUnload)
})
// Reset titleHasChanges when the task changes
watch(() => props.task.id, () => {
titleHasChanges.value = false
})
function handleTitleInput(event: Event) {
const target = event.target as HTMLInputElement
titleHasChanges.value = target.textContent !== props.task.title
}
async function save(title: string) {
// We only want to save if the title was actually changed.
// so we only continue if the task title changed.
if (title === props.task.title) {
return
}
try {
saving.value = true
const newTask = await taskStore.update({
...props.task,
title,
})
emit('update:task', newTask)
titleHasChanges.value = false
showSavedMessage.value = true
setTimeout(() => {
showSavedMessage.value = false
}, 2000)
} finally {
saving.value = false
}
}
async function cancel(element: HTMLInputElement) {
element.textContent = props.task.title
titleHasChanges.value = false
element.blur()
}
</script>
<style lang="scss" scoped>
.heading {
display: flex;
justify-content: flex-start;
text-transform: none;
align-items: center;
@media screen and (max-width: $tablet) {
flex-direction: column;
align-items: start;
}
}
.title {
margin-block-end: 0;
}
.title.input {
// 1.8rem is the font-size, 1.125 is the line-height, .3rem padding everywhere, 1px border around the whole thing.
min-block-size: calc(1.8rem * 1.125 + .6rem + 2px);
margin-inline-end: 0;
@media screen and (max-width: $tablet) {
margin: 0 -.3rem .5rem; // the title has 0.3rem padding - this make the text inside of it align with the rest
}
}
.title.task-id {
color: var(--grey-400);
white-space: nowrap;
}
.color-bubble {
block-size: .75rem;
inline-size: .75rem;
}
.close {
font-size: 2rem;
margin-inline-start: 0.5rem;
line-height: 1;
@media screen and (max-width: $tablet) {
display: none;
}
@media screen and (min-width: #{$desktop + 1px}) {
display: none;
}
}
.task-properties .close {
display: none;
position: absolute;
inset-inline-end: 1.25rem;
inset-block-start: 1.1rem;
@media screen and (max-width: $tablet) {
display: block;
}
}
.task-properties {
@media screen and (max-width: $tablet) {
flex-direction: row;
}
}
</style>