import { type EarringsShape, type GlassesShape, BeardShape, Gender, TopsShape, } from '@/enums' import type { AvatarOption, None } from '@/types' import { AVATAR_LAYER, NONE, SETTINGS, SPECIAL_AVATARS } from './constant' /** * Get a random value from an array. */ function getRandomValue( arr: Item[], { avoid = [], usually = [], }: { avoid?: unknown[]; usually?: (Item | 'none')[] } = {} ): Item { const avoidValues = avoid.filter(Boolean) const filteredArr = arr.filter((it) => !avoidValues.includes(it)) const usuallyValues = usually .filter(Boolean) .reduce((acc, cur) => acc.concat(new Array(15).fill(cur)), []) const finalArr = filteredArr.concat(usuallyValues) const randomIdx = Math.floor(Math.random() * finalArr.length) const randomValue = finalArr[randomIdx] return randomValue } export function getRandomFillColor(colors = SETTINGS.commonColors) { return colors[Math.floor(Math.random() * colors.length)] } export function getRandomAvatarOption( presetOption: Partial = {}, useOption: Partial = {} ): AvatarOption { const gender = getRandomValue(SETTINGS.gender) const beardList: BeardShape[] = [] let topList: TopsShape[] = [TopsShape.Danny, TopsShape.Wave, TopsShape.Pixie] if (gender === Gender.Male) { beardList.push(BeardShape.Scruff) topList = SETTINGS.topsShape.filter((shape) => !topList.includes(shape)) } const beardShape = getRandomValue(beardList, { usually: [NONE], }) const hairShape = getRandomValue(topList, { avoid: [useOption.widgets?.tops?.shape], }) const hairColor = getRandomFillColor() const avatarOption: AvatarOption = { gender, wrapperShape: presetOption?.wrapperShape || getRandomValue(SETTINGS.wrapperShape), background: { color: getRandomValue(SETTINGS.backgroundColor, { avoid: [ useOption.background?.color, (hairShape === TopsShape.Punk || hairShape === TopsShape.Fonze) && hairColor, // Handle special cases and prevent color conflicts. ], }), }, widgets: { face: { shape: getRandomValue(SETTINGS.faceShape), fillColor: getRandomFillColor(SETTINGS.skinColors), }, tops: { shape: hairShape, fillColor: hairColor, }, ear: { shape: getRandomValue(SETTINGS.earShape, { avoid: [useOption.widgets?.ear?.shape], }), }, earrings: { shape: getRandomValue(SETTINGS.earringsShape, { usually: [NONE], }), }, eyebrows: { shape: getRandomValue(SETTINGS.eyebrowsShape, { avoid: [useOption.widgets?.eyebrows?.shape], }), }, eyes: { shape: getRandomValue(SETTINGS.eyesShape, { avoid: [useOption.widgets?.eyes?.shape], }), }, nose: { shape: getRandomValue(SETTINGS.noseShape, { avoid: [useOption.widgets?.nose?.shape], }), }, glasses: { shape: getRandomValue(SETTINGS.glassesShape, { usually: [NONE], }), }, mouth: { shape: getRandomValue(SETTINGS.mouthShape, { avoid: [useOption.widgets?.mouth?.shape], }), }, beard: { shape: beardShape, // HACK: ...(beardShape === BeardShape.Scruff ? { zIndex: AVATAR_LAYER['mouth'].zIndex - 1 } : undefined), }, clothes: { shape: getRandomValue(SETTINGS.clothesShape, { avoid: [useOption.widgets?.clothes?.shape], }), fillColor: getRandomFillColor(), }, }, } return avatarOption } export function getSpecialAvatarOption(): AvatarOption { return SPECIAL_AVATARS[Math.floor(Math.random() * SPECIAL_AVATARS.length)] } export function showConfetti() { import('canvas-confetti').then((confetti) => { const canvasEle: HTMLCanvasElement | null = document.querySelector('#confetti') if (!canvasEle) { return } const myConfetti = confetti.create(canvasEle, { resize: true, useWorker: true, disableForReducedMotion: true, }) const duration = performance.now() + 1 * 1000 const confettiColors = ['#6967fe', '#85e9f4', '#e16984'] void (function frame() { myConfetti({ particleCount: confettiColors.length, angle: 60, spread: 55, origin: { x: 0 }, colors: confettiColors, }) myConfetti({ particleCount: confettiColors.length, angle: 120, spread: 55, origin: { x: 1 }, colors: confettiColors, }) if (performance.now() < duration) { requestAnimationFrame(frame) } })() }) } export function highlightJSON(json: string): string { if (!json) { return '' } if (typeof json != 'string') { json = JSON.stringify(json, undefined, 2) } json = json.replace(/&/g, '&').replace(//g, '>') return json.replace( /("(\\u[a-zA-Z0-9]{4}|\\[^u]|[^\\"])*"(\s*:)?|\b(true|false|null)\b|-?\d+(?:\.\d*)?(?:[eE][+-]?\d+)?)/g, (match) => { let cls = '' if (/^"/.test(match)) { if (/:$/.test(match)) { cls = 'key' } else { cls = 'string' } } else if (/true|false/.test(match)) { cls = 'boolean' } else if (/null/.test(match)) { cls = 'null' } else { cls = 'number' } return `${match}` } ) }