<script setup lang="ts"> import type { IToastProps } from '@interfaces/componentsProps'; import { computed, ref, type Ref, watch } from 'vue'; import { convertThemeToColor, getValueFromSize } from '@helpers/common'; import type { TToastType } from '@interfaces/componentsProp'; import { iconsSet } from '@/common/constants/icons'; import CrossIcon from '@icons/Mono/CrossIcon.vue'; import type { TThemeColor } from '@interfaces/common'; const props = withDefaults(defineProps<IToastProps>(), { type: 'success', size: 'normal', width: '300px', position: 'topRight', duration: 5, }); const typeToTheme: Record<TToastType, TThemeColor> = { success: 'green', info: 'blue', warn: 'yellow', error: 'red', }; const typeToHeader: Record<TToastType, string> = { success: 'Success Message', info: 'Info Message', warn: 'Warn Message', error: 'Error Message', }; const typeToIcon: Record<TToastType, string> = { success: 'CheckMark', info: 'Info', warn: 'Warning', error: 'CrossRound', }; const active = defineModel() as Ref<boolean>; let toastsContainer = document.querySelector(`.toasts-container.${props.position}`); const toast = ref(); watch([toast, () => props.static], () => { console.log('watch 2'); if (toast.value && !props.static) { toastsContainer?.appendChild(toast.value); } if (props.static) { toastsContainer?.removeChild(toast.value); } }); const themeColor = computed<TThemeColor>(() => props.theme ?? typeToTheme[props.type]); const header = computed<string>(() => props.header ?? typeToHeader[props.type]); const icon = computed(() => props.icon ?? typeToIcon[props.type]); const color = computed(() => convertThemeToColor( themeColor.value === 'white' ? 'black' : themeColor.value === 'black' ? 'white' : themeColor.value, '400', ), ); const whiteOrBlack = computed(() => (themeColor.value === 'white' ? 'black' : 'white')); const backgroundColor = computed(() => convertThemeToColor(themeColor.value, themeColor.value === 'white' || themeColor.value === 'black' ? '500' : '800'), ); const borderColor = computed(() => convertThemeToColor(themeColor.value, '500')); const fontSize = computed(() => getValueFromSize(props.size, ['12px', '16px', '20px', '24px'])); const padding = computed( () => getValueFromSize(props.size, ['7px 10px', '10px 15px', '14px 20px', '20px 30px']) as string, ); const gap = computed(() => props.size === 'normal' ? '10px' : props.size === 'large' || props.size === 'huge' ? '15px' : '5px', ); const iconSize = computed(() => fontSize.value.slice(0, -2)); const textMargin = computed(() => { return +iconSize.value + +gap.value.slice(0, -2) + 'px'; }); const positionParts = computed(() => { const result = []; if (props.position.length < 7) return [props.position]; const position = props.position.toLowerCase(); if (position.includes('top')) result.push('top'); if (position.includes('bottom')) result.push('bottom'); if (position.includes('left')) result.push('left'); if (position.includes('right')) result.push('right'); return result; }); const styles = computed(() => { if (props.static) return ''; if (positionParts.value.length === 1) { const position = positionParts.value[0]; if (position === 'left' || position === 'right') return `${position}: -100%; top: 50%; transform: translateY(-50%);`; return `${position}: -100%; left: 50%; transform: translateX(-50%);`; } return `${positionParts.value[0]}: -100%; ${positionParts.value[1]}: 20px`; }); const initContainer = () => { if (!toastsContainer) { toastsContainer = document.createElement('div'); toastsContainer.classList.add('toasts-container'); toastsContainer.classList.add(`${props.position}`); toastsContainer.style = ` position: fixed; display: flex; flex-direction: column; gap: 20px; transition: all 0.4s ease-in-out; ${styles.value} `; document.body.appendChild(toastsContainer); } }; watch( () => props.position, () => { console.log('watch 1'); initContainer(); }, { immediate: true, }, ); // const activeStyles = computed(() => { // const activeToasts = document.querySelectorAll(`.toast-container.${props.position}.active`); // let activeToastsHeight = 0; // for (const toast of activeToasts) { // activeToastsHeight += toast.offsetHeight; // } // // const offset = activeToastsHeight + 20 * activeToasts.length + 20 + 'px'; // console.log('activeToasts: ', `${positionParts.value[0]}: ${offset}`); // // if (positionParts.value.length === 1) return `${positionParts.value[0]}: ${offset}`; // return `${positionParts.value[0]}: ${offset}; ${positionParts.value[1]}: 20px`; // }); const closeToast = () => (active.value = false); let timeout: number; if (props.duration) { watch(active, () => { if (active.value) { toast.value.classList.add('active'); const activeToasts = document.querySelectorAll(`.toast-container.${props.position}.active`); let activeToastsHeight = 0; for (const toast of activeToasts) { activeToastsHeight += toast.offsetHeight; } console.log('activeToasts.length: ', activeToasts.length); const offset = activeToastsHeight + 20 * activeToasts.length; console.log('offset: ', offset, offset - toastsContainer?.clientHeight, toastsContainer?.clientHeight); toast.value.style.order = 9999; toastsContainer.style[positionParts.value[0]] = offset - toastsContainer?.clientHeight + 'px'; console.log('toast.value.style: ', toast.value.style); timeout = setTimeout(() => (active.value = false), (props.duration as number) * 1000); } else { console.log('inactive'); toast.value.classList.remove('active'); const activeToasts = document.querySelectorAll(`.toast-container.${props.position}.active`); let activeToastsHeight = 0; for (const toast of activeToasts) { activeToastsHeight += toast.offsetHeight; } const offset = activeToastsHeight + 20 * activeToasts.length; toastsContainer.style[positionParts.value[0]] = offset - toastsContainer?.clientHeight + 'px'; toast.value.style.order = 1; clearTimeout(timeout); } }); } </script> <template> <section ref="toast" :class="[ `toast-container ${position}`, { active, }, ]" :style="`position: relative; ${styles}`" > <h3 class="toast-header" :style="`font-size: calc(${fontSize} + 4px)`"> <component :is="iconsSet[icon]" :color="color" :size="iconSize" /> <span class="toast-header-text">{{ header }}</span> </h3> <p class="toast-text">{{ text ?? `This is a toast about ${type}` }}</p> <CrossIcon @click="closeToast" class="toast-close_button" :style="`top: ${padding.split(' ')[0]}; right: ${padding.split(' ')[1]}`" :color="color" :size="iconSize" /> </section> </template> <style scoped> .toast-container { z-index: 9999; padding: v-bind(padding); border: 1px solid v-bind(borderColor); border-radius: 7px; width: v-bind(width); ::before { content: ''; position: absolute; z-index: -1; top: 0; left: 0; background-color: v-bind(backgroundColor); width: 100%; height: 100%; opacity: 0.55; border-radius: 6px; } } .toast-header { display: flex; gap: v-bind(gap); color: v-bind(color); margin-bottom: 5px; } .toast-header-text { font-weight: 500; } .toast-text { margin-left: v-bind(textMargin); color: v-bind(whiteOrBlack); font-size: v-bind(fontSize); } .toast-close_button { background-color: transparent; position: absolute; cursor: pointer; display: block; } </style>