Files
BewlyBewly/src/components/Dialog.vue

223 lines
6.5 KiB
Vue

<script setup lang="ts">
import { onKeyStroke } from '@vueuse/core'
import Button from '~/components/Button.vue'
import { useBewlyApp } from '~/composables/useAppProvider'
const props = withDefaults(defineProps<{
title?: string
desc?: string
center?: boolean
frostedGlass?: boolean
appendToBewlyBody?: boolean
width?: string | number
maxWidth?: string | number
contentHeight?: string | number
contentMaxHeight?: string | number
showFooter?: boolean
centerFooter?: boolean
loading?: boolean
preventCloseWhenLoading?: boolean
}>(), {
preventCloseWhenLoading: true,
frostedGlass: true,
showFooter: true,
})
const emit = defineEmits(['close', 'confirm'])
onKeyStroke('Enter', (e: KeyboardEvent) => {
e.preventDefault()
if (!props.loading)
handleConfirm()
})
onKeyStroke('Escape', (e: KeyboardEvent) => {
e.preventDefault()
if (props.loading && props.preventCloseWhenLoading)
return
handleClose()
})
const showShortcut = ref<boolean>(false)
const { mainAppRef } = useBewlyApp()
const showDialog = ref<boolean>(false)
const dialogWidth = computed(() => {
return typeof props.width === 'number' ? `${props.width}px` : props.width || '400px'
})
const dialogMaxWidth = computed(() => {
return typeof props.maxWidth === 'number' ? `${props.maxWidth}px` : props.maxWidth || 'unset'
})
const dialogContentHeight = computed(() => {
return typeof props.contentHeight === 'number' ? `${props.contentHeight}px` : props.contentHeight || 'auto'
})
const dialogContentMaxHeight = computed(() => {
return typeof props.contentMaxHeight === 'number' ? `${props.contentMaxHeight}px` : props.contentMaxHeight || 'auto'
})
onKeyStroke('Alt', (e: KeyboardEvent) => {
e.preventDefault()
showShortcut.value = true
}, { eventName: 'keydown' })
onKeyStroke('Alt', (e: KeyboardEvent) => {
e.preventDefault()
showShortcut.value = false
}, { eventName: 'keyup' })
onMounted(() => {
showDialog.value = true
})
onBeforeUnmount(() => {
handleClose()
})
function handleClose() {
if (props.loading && props.preventCloseWhenLoading)
return
showDialog.value = false
nextTick(() => {
emit('close')
})
}
function handleConfirm() {
emit('confirm')
if (!props.loading)
handleClose()
}
</script>
<template>
<Teleport :to="mainAppRef" :disabled="!appendToBewlyBody">
<Transition name="modal">
<div
v-if="showDialog"
class="dialog"
pos="fixed top-0 left-0" w-full h-full z-10002
transform-gpu pointer-events-auto
>
<div
bg="black opacity-40 dark:opacity-40"
pos="absolute top-0 left-0" w-full h-full z-0
@click="handleClose"
/>
<div
style="
box-shadow: var(--bew-shadow-4), var(--bew-shadow-edge-glow-2);
"
:style="{
width: dialogWidth,
maxWidth: dialogMaxWidth,
backdropFilter: frostedGlass ? 'var(--bew-filter-glass-2)' : 'none',
backgroundColor: frostedGlass ? 'var(--bew-elevated)' : 'var(--bew-elevated-solid)',
}"
pos="absolute top-1/2 left-1/2" rounded="$bew-radius" border="1 $bew-border-color"
transform="translate--1/2" z-2
antialiased
>
<!-- loading masking -->
<Transition name="fade">
<div
v-if="loading"
pos="absolute top-0 left-0" w-full h-full bg="white dark:black opacity-60 dark:opacity-60" flex="~ justify-center items-center"
z-2
>
<div i-svg-spinners-ring-resize text="4xl" />
</div>
</Transition>
<header
style="
text-shadow: 0 0 15px var(--bew-elevated-solid), 0 0 20px var(--bew-elevated-solid)
"
pos="sticky top-0 left-0" w-full h-70px px-12 flex
items-center justify-between
rounded="t-$bew-radius" z-1
>
<div
:style="{ textAlign: center ? 'center' : 'left' }"
w-full
>
<slot name="title">
<p text-xl fw-bold>
{{ title }}
</p>
</slot>
<p text="sm $bew-text-2">
<slot name="desc">
{{ desc }}
</slot>
</p>
</div>
<div
style="
backdrop-filter: var(--bew-filter-glass-1);
box-shadow: var(--bew-shadow-edge-glow-1), var(--bew-shadow-1);
"
text="!16px hover:$bew-theme-color" w="32px" h="32px"
flex="~ items-center justify-center shrink-0"
bg="$bew-fill-1 hover:$bew-theme-color-30"
ml-8 rounded-8 cursor="pointer" border="1 $bew-border-color"
box-border
duration-300
@click="handleClose"
>
<div i-ic-baseline-clear />
</div>
</header>
<main
:style="{
height: dialogContentHeight,
maxHeight: dialogContentMaxHeight,
paddingBottom: !showFooter ? '2rem' : '0.5rem',
}"
p="x-12 y-2" relative overflow="x-hidden y-overlay"
>
<!-- <div h-80px mt--8 /> -->
<slot />
</main>
<footer
v-if="showFooter"
:style="{ justifyContent: centerFooter || center ? 'center' : 'flex-end' }"
flex="~ gap-2" p="x-12 t-2 b-6"
>
<Button type="tertiary" @click="handleClose">
<div>
{{ $t('common.cancel') }}
<span
v-show="showShortcut"
text="xs $bew-text-2 lh-0" p="x-1" rounded-4px bg="$bew-fill-1"
border="1 $bew-border-color"
mix-blend-color-dodge
>
ESC
</span>
</div>
</Button>
<Button type="primary" @click="handleConfirm">
<div>
{{ $t('common.confirm') }}
<span
v-show="showShortcut"
text="xs $bew-text-2 lh-0" p="x-1" rounded-4px bg="$bew-fill-1"
border="1 $bew-border-color"
mix-blend-color-dodge
>
ENTER
</span>
</div>
</Button>
</footer>
</div>
</div>
</Transition>
</Teleport>
</template>
<style lang="scss" scoped>
</style>