import { reactive, ref, computed, onBeforeUnmount } from "vue";
* Composable que realiza una cuenta regresiva hasta una fecha/hora específica.
* @param time - Fecha objetivo en formato ISO, por ejemplo: '2025-06-01T12:00:00'
export function useCountdown(time: string) {
// Objeto reactivo que almacena los valores de días, horas, minutos y segundos
// Referencia para almacenar el ID del intervalo
const intervalId = ref<number | null>(null);
* Calcula la diferencia de tiempo entre la fecha actual y la fecha objetivo.
* Actualiza el estado del temporizador.
function useCalculateTimeLeft() {
const startDate = new Date(time); // Fecha objetivo
const now = new Date(); // Fecha actual
const difference = startDate.getTime() - now.getTime();
// Si ya ha pasado la fecha objetivo, se detiene el temporizador
clearInterval(intervalId.value as number);
// Cálculo de días, horas, minutos y segundos restantes
timer.days = Math.floor(difference / (1000 * 60 * 60 * 24));
timer.hours = Math.floor(
(difference % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60)
timer.minutes = Math.floor((difference % (1000 * 60 * 60)) / (1000 * 60));
timer.seconds = Math.floor((difference % (1000 * 60)) / 1000);
// Inicia el intervalo para actualizar el temporizador cada segundo
intervalId.value = window.setInterval(useCalculateTimeLeft, 1000);
// Ejecuta la función inmediatamente al crear el composable
// Limpia el intervalo cuando se destruye el componente
if (intervalId.value !== null) {
clearInterval(intervalId.value);
// Computed que indica si el temporizador sigue en marcha (aún queda tiempo)
const endTimer = computed(
() => !Object.values(timer).every((value) => value <= 0)
timer, // Objeto reactivo con tiempo restante
endTimer, // Booleano que indica si la cuenta regresiva ha terminado