/* HTML generation */
const TYPES = {
Grass: '🍃',
Fire: '🔥',
Water: '💧',
Lightning: '⚡',
Fighting: '✊',
Psychic: '👁️',
Colorless: '⭐',
Darkness: '🌑',
Metal: '⚙️',
Dragon: '🐲',
Fairy: '🧚',
};
const energyHTML = (type, types = TYPES) => {
return `${types[type]}`;
};
const attackDescriptionHTML = (text) => {
if (!text) {
return '';
}
let fontSize;
if (text.length > 185) {
fontSize = 0.7;
} else if (text.length > 140) {
fontSize = 0.8;
} else if (text.length > 90) {
fontSize = 0.9;
}
return `${text}`;
};
const attackRowsHTML = (attacks) => {
return attacks
.map((attack) => {
const { cost, damage, name, text } = attack;
return `
${cost.map((energy) => energyHTML(energy)).join('')}
${name}
${attackDescriptionHTML(text)}
${damage ? damage : ''}
`;
})
.join('');
};
const cardHTML = (details) => {
const { hp, energy_type, species, length, weight, attacks, weakness, resistance, retreat, description, rarity } =
details;
const poke_name = details.name; // `name` would be reserved JS word
return `
Basic Pokémon
${species} Pokémon. Length: ${length.feet}'${length.inches}", Weight: ${weight}
${attackRowsHTML(attacks)}
weakness
${weakness ? energyHTML(weakness) : ''}
resistance
${resistance ? energyHTML(resistance) : ''}
${resistance ? '-30' : ''}
retreat cost
${energyHTML('Colorless').repeat(retreat)}
${description}
`;
};
/* Utility */
const getBasePath = () => {
return document.location.origin + document.location.pathname;
};
const generateDetails = async () => {
const details = await fetch(`${getBasePath()}/details`);
return await details.json();
};
const createTask = async (prompt) => {
const taskResponse = await fetch(`${getBasePath()}task/create?prompt=${prompt}`);
const task = await taskResponse.json();
return task;
};
const queueTask = async (task_id) => {
const queueResponse = await fetch(`${getBasePath()}task/queue?task_id=${task_id}`);
return queueResponse.json();
};
const pollTask = async (task) => {
const taskResponse = await fetch(`${getBasePath()}task/poll?task_id=${task.task_id}`);
return await taskResponse.json();
};
const longPollTask = async (task, interval = 10_000, max) => {
const etaDisplay = document.querySelector('.eta');
task = await pollTask(task);
if (task.status === 'completed' || (max && task.poll_count > max)) {
return task;
}
etaDisplay.textContent = Math.round(task.eta);
await new Promise((resolve) => setTimeout(resolve, interval));
return await longPollTask(task, interval, max);
};
/* DOM */
const generateButton = document.querySelector('button.generate');
const durationTimer = () => {
const elapsedDisplay = document.querySelector('.elapsed');
let duration = 0.0;
return () => {
const startTime = performance.now();
const incrementSeconds = setInterval(() => {
duration += 0.1;
elapsedDisplay.textContent = duration.toFixed(1);
}, 100);
const updateDuration = (task) => {
if (task?.status == 'completed') {
duration = task.completed_at - task.created_at;
return;
}
duration = Number(((performance.now() - startTime) / 1_000).toFixed(1));
};
window.addEventListener('focus', updateDuration);
return {
cleanup: (completedTask) => {
updateDuration(completedTask);
clearInterval(incrementSeconds);
window.removeEventListener('focus', updateDuration);
elapsedDisplay.textContent = duration.toFixed(1);
},
};
};
};
const rotateCard = () => {
const RANGE = 0.1;
const INTERVAL = 13; // ~75 per second
let previousTime = 0;
// Throttle closure
return (card, containerMouseEvent) => {
const currentTime = performance.now();
if (currentTime - previousTime > INTERVAL) {
previousTime = currentTime;
const rect = card.getBoundingClientRect();
const rotateX = (containerMouseEvent.clientY - rect.y - rect.height / 2) * RANGE;
const rotateY = -(containerMouseEvent.clientX - rect.x - rect.width / 2) * RANGE;
card.style.setProperty('--rotate-x', rotateX + 'deg');
card.style.setProperty('--rotate-y', rotateY + 'deg');
}
};
};
const cardRotationInitiator = (renderSection) => {
let currentCard;
return () => {
let handleMouseMove;
if (currentCard) {
handleMouseMove = rotateCard().bind(null, currentCard);
renderSection.removeEventListener('mousemove', handleMouseMove, true);
}
const newCard = document.querySelector('.pokecard');
currentCard = newCard;
handleMouseMove = rotateCard().bind(null, newCard);
renderSection.addEventListener('mousemove', handleMouseMove, true);
};
};
let generating = false;
generateButton.addEventListener('click', async () => {
if (generating) {
return;
}
const renderSection = document.querySelector('section.render');
const durationDisplay = document.querySelector('.duration');
const initialiseCardRotation = cardRotationInitiator(renderSection);
try {
generating = true;
const details = await generateDetails();
const task = await createTask(details.energy_type);
const timer = durationTimer(durationDisplay);
const timerCleanup = timer().cleanup;
const longPromises = [queueTask(task.task_id), longPollTask(task)];
const completedTask = await Promise.any(longPromises);
generating = false;
timerCleanup(completedTask);
renderSection.innerHTML = cardHTML(details);
const picture = document.querySelector('img.picture');
picture.src = completedTask.value;
initialiseCardRotation();
} catch (err) {
generating = false;
console.error(err);
}
});