-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathCharacterPoints.js
121 lines (99 loc) · 4.63 KB
/
CharacterPoints.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
import React from "react";
import ReactDOM from "react-dom";
function CharacterAttributes({ totalPoints }) {
const [health, setHealth] = React.useState(0);
const [stamina, setStamina] = React.useState(0);
const [speed, setSpeed] = React.useState(0);
const [score, setScore] = React.useState(0);
React.useEffect(() => {
setScore(() => stamina + speed + health);
}, [health, stamina, speed]);
const handleChange = (e) => {
// Could save e.target.name & e.target.value to variables as well
const isWithinThreshold = +e.target.value <= Math.floor(totalPoints * 0.7);
const isTotalPointsOne = totalPoints === 1;
if (e.target.name === "health") {
if ((isTotalPointsOne && stamina === 0 && speed === 0 && +e.target.value === 1) ||
(isWithinThreshold && +e.target.value + stamina + speed <= totalPoints)) {
setHealth(+e.target.value);
}
} else if (e.target.name === "stamina") {
if ((isTotalPointsOne && health === 0 && speed === 0 && +e.target.value === 1) ||
(isWithinThreshold && health + +e.target.value + speed <= totalPoints)) {
setStamina(+e.target.value);
}
} else if (e.target.name === "speed") {
if ((isTotalPointsOne && health === 0 && stamina === 0 && +e.target.value === 1) ||
(isWithinThreshold && health + stamina + +e.target.value <= totalPoints)) {
setSpeed(+e.target.value);
}
}
};
return (
<div>
Character stats: <span id="points">{Number(totalPoints) - score}</span> points left.
<div>
<input type="range" id="health" name="health" min="0" max={totalPoints} value={health} step="1" onChange={(e) => handleChange(e)} />Health
</div>
<div>
<input type="range" id="stamina" name="stamina" min="0" max={totalPoints} value={stamina} step="1" onChange={(e) => handleChange(e)} />Stamina
</div>
<div>
<input type="range" id="speed" name="speed" min="0" max={totalPoints} value={speed} step="1" onChange={(e) => handleChange(e)} />Speed
</div>
</div>
);
}
document.body.innerHTML = "<div id='root'></div>";
const root = ReactDOM.createRoot(document.getElementById("root"));
root.render(<CharacterAttributes totalPoints={15} />);
/*
Problems with this implementation
- max -> distorts the range
- can't move the sliders again after score is equal to totalPoints
- should have added name property to inputs -> easily overlooked
- I don't mind props -> especially when you have a lot of local variables
- Lesson learned -> best to do all logic in a separate handler function
import React, { useEffect, useState } from "react";
import ReactDOM from "react-dom";
function CharacterAttributes(props) {
const [health, setHealth] = React.useState(0);
const [stamina, setStamina] = React.useState(0);
const [speed, setSpeed] = React.useState(0);
const [score, setScore] = React.useState(0);
// At one point, I tried to use a state object and a single state handler
// e.target.name = e.target.value
// I was having difficulty so I dumbed it down to use separate state
const max = props.totalPoints === 1 ? 1 : Math.floor(props.totalPoints * .7);
React.useEffect(()=> {
setScore(()=> stamina + speed + health);
}, [health,stamina,speed]);
return (
<div>
Character stats: <span id="points">{Number(props.totalPoints) - score}</span> points left.
<div>
<input type="range" id="health" min="0" max={max} value={health} step="1" onChange={(e)=> {
if(props.totalPoints === 1 && stamina === 1 || speed === 1) return;
if(props.totalPoints === score) return;
setHealth(+e.target.value)
}} />Health
</div>
<div>
<input type="range" id="stamina" min="0" max={max} value={stamina} step="1" onChange={(e)=> {
if(props.totalPoints === 1 && health === 1 || speed === 1) return;
if(props.totalPoints === score) return;
setStamina(+e.target.value)}} />Stamina
</div>
<div>
<input type="range" id="speed" min="0" max={max} value={speed} step="1" onChange={(e)=> {
if(props.totalPoints === 1 && health === 1 || stamina === 1) return;
if(props.totalPoints === score) return;
setSpeed(+e.target.value)
}} />Speed
</div>
</div>
)
}
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(<CharacterAttributes totalPoints={15} />);
*/