mirror of
https://github.com/Vikeo/LifeTrinket.git
synced 2025-11-20 17:57:59 +00:00
294 lines
9.3 KiB
TypeScript
294 lines
9.3 KiB
TypeScript
import { useEffect, useRef, useState } from 'react';
|
|
import { useSwipeable } from 'react-swipeable';
|
|
import { twc } from 'react-twc';
|
|
import { useGlobalSettings } from '../../Hooks/useGlobalSettings';
|
|
import { usePlayers } from '../../Hooks/usePlayers';
|
|
import { Cog } from '../../Icons/generated';
|
|
import { Player, Rotation } from '../../Types/Player';
|
|
import {
|
|
RotationButtonProps,
|
|
RotationDivProps,
|
|
} from '../Buttons/CommanderDamage';
|
|
import { LoseGameButton } from '../Buttons/LoseButton';
|
|
import CommanderDamageBar from '../Counters/CommanderDamageBar';
|
|
import ExtraCountersBar from '../Counters/ExtraCountersBar';
|
|
import { Paragraph } from '../Misc/TextComponents';
|
|
import PlayerMenu from '../Players/PlayerMenu';
|
|
import Health from './Health';
|
|
import { baseColors } from '../../../tailwind.config';
|
|
|
|
const SettingsButtonTwc = twc.button<RotationButtonProps>((props) => [
|
|
'absolute flex-grow border-none outline-none cursor-pointer bg-transparent z-[1] select-none webkit-user-select-none',
|
|
props.$rotation === Rotation.Side || props.$rotation === Rotation.SideFlipped
|
|
? `right-auto top-[1vmax] left-[27%]`
|
|
: 'top-1/4 right-[1vmax]',
|
|
]);
|
|
|
|
type SettingsButtonProps = {
|
|
onClick: () => void;
|
|
rotation: Rotation;
|
|
};
|
|
|
|
const SettingsButton = ({ onClick, rotation }: SettingsButtonProps) => {
|
|
return (
|
|
<SettingsButtonTwc
|
|
onClick={onClick}
|
|
$rotation={rotation}
|
|
aria-label={`Settings`}
|
|
>
|
|
<Cog size="5vmin" color="black" opacity="0.3" />
|
|
</SettingsButtonTwc>
|
|
);
|
|
};
|
|
|
|
const LifeCounterContentWrapper = twc.div`
|
|
relative flex flex-grow flex-col items-center w-full h-full overflow-hidden`;
|
|
|
|
const LifeCounterWrapper = twc.div<RotationDivProps>((props) => [
|
|
'relative flex items-center w-full h-full z-[1]',
|
|
props.$rotation === Rotation.SideFlipped || props.$rotation === Rotation.Side
|
|
? `flex-row`
|
|
: `flex-col`,
|
|
]);
|
|
|
|
const PlayerLostWrapper = twc.div<RotationDivProps>((props) => [
|
|
'z-[1] flex absolute w-full h-full justify-center items-center pointer-events-none select-none webkit-user-select-none bg-lifeCounter-lostWrapper opacity-75',
|
|
props.$rotation === Rotation.SideFlipped || props.$rotation === Rotation.Side
|
|
? `rotate-[${props.$rotation - 90}deg]`
|
|
: '',
|
|
]);
|
|
|
|
const DynamicText = twc.div`text-[8vmin] whitespace-nowrap`;
|
|
|
|
const hasCommanderDamageReached21 = (player: Player) => {
|
|
const commanderDamageTotals = player.commanderDamage.map(
|
|
(commanderDamage) => commanderDamage.damageTotal
|
|
);
|
|
|
|
const partnerDamageTotals = player.commanderDamage.map(
|
|
(commanderDamage) => commanderDamage.partnerDamageTotal
|
|
);
|
|
|
|
return (
|
|
commanderDamageTotals.some((damageTotal) => damageTotal >= 21) ||
|
|
partnerDamageTotals.some((partnerDamageTotal) => partnerDamageTotal >= 21)
|
|
);
|
|
};
|
|
|
|
const playerCanLose = (player: Player) => {
|
|
const poisonCounter = player.extraCounters.find(
|
|
(counter) => counter.type === 'poison'
|
|
);
|
|
|
|
return (
|
|
player.lifeTotal < 1 ||
|
|
hasCommanderDamageReached21(player) ||
|
|
(poisonCounter && poisonCounter.value >= 10)
|
|
);
|
|
};
|
|
|
|
type LifeCounterProps = {
|
|
player: Player;
|
|
opponents: Player[];
|
|
isStartingPlayer?: boolean;
|
|
};
|
|
|
|
const RECENT_DIFFERENCE_TTL = 3_000;
|
|
|
|
const LifeCounter = ({ player, opponents }: LifeCounterProps) => {
|
|
const { updatePlayer, updateLifeTotal } = usePlayers();
|
|
const { settings, playing, setPlaying, stopPlayerRandomization } =
|
|
useGlobalSettings();
|
|
const playingTimerRef = useRef<NodeJS.Timeout | undefined>(undefined);
|
|
|
|
const [showPlayerMenu, setShowPlayerMenu] = useState(false);
|
|
const [recentDifference, setRecentDifference] = useState(0);
|
|
const [differenceKey, setDifferenceKey] = useState(Date.now());
|
|
const [isLandscape, setIsLandscape] = useState(false);
|
|
|
|
const calcRot = player.isSide
|
|
? player.settings.rotation - 180
|
|
: player.settings.rotation;
|
|
|
|
const rotationAngle = isLandscape ? calcRot : calcRot + 90;
|
|
|
|
const handlers = useSwipeable({
|
|
trackMouse: true,
|
|
onSwipedDown: (e) => {
|
|
e.event.stopPropagation();
|
|
setShowPlayerMenu(true);
|
|
},
|
|
onSwipedUp: (e) => {
|
|
e.event.stopPropagation();
|
|
setShowPlayerMenu(false);
|
|
},
|
|
|
|
swipeDuration: 500,
|
|
onSwiping: (e) => e.event.stopPropagation(),
|
|
rotationAngle,
|
|
});
|
|
|
|
useEffect(() => {
|
|
const timer = setTimeout(() => {
|
|
setRecentDifference(0);
|
|
}, RECENT_DIFFERENCE_TTL);
|
|
|
|
const resizeObserver = new ResizeObserver(() => {
|
|
if (document.body.clientWidth > document.body.clientHeight)
|
|
setIsLandscape(true);
|
|
else setIsLandscape(false);
|
|
return;
|
|
});
|
|
|
|
resizeObserver.observe(document.body);
|
|
|
|
return () => {
|
|
clearTimeout(timer);
|
|
// Cleanup: disconnect the ResizeObserver when the component unmounts.
|
|
resizeObserver.disconnect();
|
|
};
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [recentDifference, document.body.clientHeight, document.body.clientWidth]);
|
|
|
|
useEffect(() => {
|
|
if (
|
|
player.isStartingPlayer &&
|
|
((!playing &&
|
|
settings.useRandomStartingPlayerInterval &&
|
|
stopPlayerRandomization) ||
|
|
(!settings.useRandomStartingPlayerInterval && !playing))
|
|
) {
|
|
playingTimerRef.current = setTimeout(() => {
|
|
setPlaying(true);
|
|
}, 10_000);
|
|
}
|
|
|
|
return () => clearTimeout(playingTimerRef.current);
|
|
}, [
|
|
player.isStartingPlayer,
|
|
playing,
|
|
setPlaying,
|
|
settings.useRandomStartingPlayerInterval,
|
|
stopPlayerRandomization,
|
|
]);
|
|
|
|
player.settings.rotation === Rotation.SideFlipped ||
|
|
player.settings.rotation === Rotation.Side;
|
|
|
|
const handleLifeChange = (updatedLifeTotal: number) => {
|
|
const difference = updateLifeTotal(player, updatedLifeTotal);
|
|
setRecentDifference(recentDifference + difference);
|
|
setDifferenceKey(Date.now());
|
|
};
|
|
|
|
const toggleGameLost = () => {
|
|
const updatedPlayer = { ...player, hasLost: !player.hasLost };
|
|
updatePlayer(updatedPlayer);
|
|
};
|
|
|
|
const calcRotation =
|
|
player.settings.rotation === Rotation.SideFlipped ||
|
|
player.settings.rotation === Rotation.Side
|
|
? player.settings.rotation - 90
|
|
: player.settings.rotation;
|
|
|
|
const calcTextRotation =
|
|
player.settings.rotation === Rotation.SideFlipped ||
|
|
player.settings.rotation === Rotation.Side
|
|
? player.settings.rotation - 180
|
|
: player.settings.rotation;
|
|
|
|
return (
|
|
<LifeCounterContentWrapper style={{ background: player.color }}>
|
|
<LifeCounterWrapper
|
|
$rotation={player.settings.rotation}
|
|
style={{ rotate: `${calcRotation}deg` }}
|
|
{...handlers}
|
|
>
|
|
{!playing && settings.showStartingPlayer && player.isStartingPlayer && (
|
|
<div
|
|
className="z-20 flex absolute w-full h-full justify-center items-center select-none cursor-pointer webkit-user-select-none"
|
|
style={{
|
|
rotate: `${calcRotation}deg`,
|
|
backgroundImage:
|
|
stopPlayerRandomization ||
|
|
!settings.useRandomStartingPlayerInterval
|
|
? `radial-gradient(circle at center, ${player.color}, ${baseColors.primary.main})`
|
|
: 'none',
|
|
}}
|
|
onClick={() => {
|
|
clearTimeout(playingTimerRef.current);
|
|
setPlaying(true);
|
|
}}
|
|
>
|
|
<DynamicText
|
|
style={{
|
|
rotate: `${calcTextRotation}deg`,
|
|
}}
|
|
>
|
|
<div className="flex flex-col justify-center items-center">
|
|
<Paragraph>👑</Paragraph>
|
|
{(stopPlayerRandomization ||
|
|
!settings.useRandomStartingPlayerInterval) && (
|
|
<>
|
|
<Paragraph>You start!</Paragraph>
|
|
<Paragraph className="text-xl">(Press to hide)</Paragraph>
|
|
</>
|
|
)}
|
|
</div>
|
|
</DynamicText>
|
|
</div>
|
|
)}
|
|
|
|
{player.hasLost && (
|
|
<PlayerLostWrapper $rotation={player.settings.rotation} />
|
|
)}
|
|
{settings.useRandomStartingPlayerInterval &&
|
|
!stopPlayerRandomization &&
|
|
!playing && (
|
|
<div
|
|
className="flex absolute w-full h-full justify-center items-center pointer-events-none select-none webkit-user-select-none z-10"
|
|
style={{ backgroundColor: player.color }}
|
|
/>
|
|
)}
|
|
<CommanderDamageBar
|
|
opponents={opponents}
|
|
player={player}
|
|
key={player.index}
|
|
handleLifeChange={handleLifeChange}
|
|
/>
|
|
{settings.showPlayerMenuCog && (
|
|
<SettingsButton
|
|
onClick={() => {
|
|
setShowPlayerMenu(!showPlayerMenu);
|
|
}}
|
|
rotation={player.settings.rotation}
|
|
/>
|
|
)}
|
|
{playerCanLose(player) && (
|
|
<LoseGameButton
|
|
rotation={player.settings.rotation}
|
|
onClick={toggleGameLost}
|
|
/>
|
|
)}
|
|
<Health
|
|
player={player}
|
|
rotation={player.settings.rotation}
|
|
differenceKey={differenceKey}
|
|
recentDifference={recentDifference}
|
|
handleLifeChange={handleLifeChange}
|
|
/>
|
|
<ExtraCountersBar player={player} />
|
|
|
|
<PlayerMenu
|
|
isShown={showPlayerMenu}
|
|
player={player}
|
|
setShowPlayerMenu={setShowPlayerMenu}
|
|
/>
|
|
</LifeCounterWrapper>
|
|
</LifeCounterContentWrapper>
|
|
);
|
|
};
|
|
|
|
export default LifeCounter;
|