App.js 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. import './App.css';
  2. import { clear } from '@testing-library/user-event/dist/clear';
  3. import React, {useState, useEffect} from 'react';
  4. import watch from "./img/ClockFace.png";
  5. import hourArrow from "./img/ClockFace_H.png";
  6. import minuteArrow from "./img/ClockFace_M.png";
  7. import secondArrow from "./img/ClockFace_S.png";
  8. // Spoiler
  9. const Spoiler = ({header="+", open=true, children}) => {
  10. const [isOpen, setOpen] = useState(open)
  11. return <>
  12. <button onClick={() => setOpen(!isOpen)}>{header}</button>
  13. {!isOpen && children}
  14. </>
  15. }
  16. // RangeInput
  17. const RangeInput = ({min, max}) => {
  18. const [text, setText] = useState("")
  19. return <>
  20. <input
  21. style={ {color: (text.length <= min || text.length > max) ? 'red' : 'black'}}
  22. value={text}
  23. onChange={(e) => setText(e.target.value)}>
  24. </input>
  25. </>
  26. }
  27. // PasswordConfirm
  28. const PasswordConfirm = ({min}) => {
  29. const [pswd, setPswd] = useState("")
  30. const [pswd2, setPswd2] = useState("")
  31. return <>
  32. <h3>password:</h3>
  33. <input type='password'
  34. style={ {color: pswd.length <= min ? 'red' : 'black'}}
  35. onChange={(e) => setPswd(e.target.value)} />
  36. <h3>repeat password:</h3>
  37. <input type = 'password'
  38. style={ {color: pswd !== pswd2 ? 'red' : 'black'}}
  39. onChange={(e) => setPswd2(e.target.value)} />
  40. </>
  41. }
  42. //Timer
  43. const Timer = ({seconds}) => {
  44. const [sec, setSec] = useState(seconds)
  45. let [pause, setPause] = useState(false)
  46. let hours = Math.floor(sec / (60 * 60))
  47. let minuts = Math.floor(sec % (60 * 60) / 60)
  48. let second = sec % 60
  49. let interval
  50. useEffect(() => {
  51. if (sec <= 0) {
  52. setPause(true)
  53. }
  54. if (!pause) {
  55. interval = setInterval(() => {
  56. setSec(sec => sec - 1)
  57. }, 1000);
  58. }
  59. return () => {
  60. clearInterval(interval)
  61. }
  62. }, [pause, sec])
  63. return (
  64. <div>
  65. <h3>Timer</h3>
  66. <span>{hours}</span> : <span>{minuts}</span> : <span>{second} </span>
  67. <button onClick={() => { setPause(!pause) }}>{pause ? 'start' : 'stop'}</button>
  68. </div>
  69. )
  70. }
  71. //TimerControl
  72. const TimerControl = () => {
  73. const [hours, setHours] = useState(0)
  74. const [minutes, setMinutes] = useState(0)
  75. const [seconds, setSeconds] = useState(0)
  76. const [arr, setArr] = useState([])
  77. let sec = hours * 3600 + minutes * 60 + seconds;
  78. return (
  79. <>
  80. <h3>Timer Control</h3>
  81. <input placeholder='hours'type='number' min='0' value={hours} onChange={(e) => setHours(+e.target.value)} />
  82. <input placeholder='minutes' type='number' min='0' value={minutes} onChange={(e) => setMinutes(+e.target.value)} />
  83. <input placeholder='seconds' type='number' min='0' value={seconds} onChange={(e) => setSeconds(+e.target.value)} />
  84. {arr.map(item=> (<Timer key={item} seconds={sec} />))}
  85. <button onClick={()=>{setArr([Math.random()]) }}>Start</button>
  86. </>
  87. )
  88. }
  89. //TimerContainer
  90. const SecondsTimer = ({seconds}) => <h2>{seconds}</h2>
  91. const TimerContainer = ({seconds = 1000, refresh, render:Render}) => {
  92. const [time, setTime] = useState(seconds)
  93. const [pause, setPause] = useState(false)
  94. let interval
  95. useEffect(() => {
  96. let timeBefore = Math.floor(performance.now())
  97. interval = setInterval(() => {
  98. let timeAfter = Math.floor(performance.now())
  99. !pause && time > 0 ? setTime((seconds - (timeAfter - timeBefore) / 1000)) : setTime(time)
  100. }, refresh)
  101. return() => {
  102. clearInterval(interval)
  103. }
  104. }, [pause, refresh, seconds])
  105. return (
  106. <>
  107. <h3>Timer Container</h3>
  108. <div>
  109. <Render seconds={time} />
  110. <button onClick={() => {setPause(!pause)}}>{!pause ? 'Stop' : 'Start'}</button>
  111. </div>
  112. </>
  113. )
  114. }
  115. //LCD
  116. const LCD = ({seconds}) => {
  117. let hours = Math.floor(seconds / 60 / 60)
  118. let minutes = Math.floor((seconds % 3600) / 60)
  119. let sec = seconds % 60
  120. return (
  121. <>
  122. <div>
  123. <span>{hours}</span> : <span>{minutes}</span> : <span>{sec.toFixed(2)} </span>
  124. </div>
  125. </>
  126. )
  127. }
  128. //Watch
  129. const Watch = ({seconds}) => {
  130. let hours = Math.floor(seconds / 60 / 60)
  131. let minutes = Math.floor((seconds % 3600) / 60)
  132. let second = seconds % 60
  133. return (
  134. <>
  135. <h3>Watch</h3>
  136. <div className='Watch'>
  137. <img
  138. className='Clock'
  139. src={watch}
  140. alt="Watch" />
  141. <img
  142. className="Arrows"
  143. style={{ transform: `rotate(${(360 / 12) * hours}deg)` }}
  144. src={hourArrow}
  145. alt="Hours"
  146. />
  147. <img
  148. className="Arrows"
  149. style={{ transform: `rotate(${(360 / 12 /5) * minutes}deg)` }}
  150. src={minuteArrow}
  151. alt="Minutes"
  152. />
  153. <img
  154. className="Arrows"
  155. style={{ transform: `rotate(${(360 / 12 /5) * second}deg)` }}
  156. src={secondArrow}
  157. alt="Seconds"
  158. />
  159. </div>
  160. </>
  161. )
  162. }
  163. const TimerContainerWatch = ({seconds = 1000, refresh, render:Render}) => {
  164. const [time, setTime] = useState(seconds)
  165. const [pause, setPause] = useState(false)
  166. let interval
  167. useEffect(() => {
  168. let timeBefore = Math.floor(performance.now())
  169. interval = setInterval(() => {
  170. let timeAfter = Math.floor(performance.now())
  171. !pause && time > 0 ? setTime((seconds - (timeAfter - timeBefore) / 1000)) : setTime(time)
  172. }, refresh)
  173. return() => {
  174. clearInterval(interval)
  175. }
  176. }, [pause, refresh, seconds])
  177. return (
  178. <>
  179. <div>
  180. <Render seconds={time} />
  181. </div>
  182. </>
  183. )
  184. }
  185. function App() {
  186. return (
  187. <div className="App">
  188. <div>Spoiler:</div>
  189. <Spoiler header={<h1>Заголовок</h1>} open>
  190. Контент 1
  191. <p>
  192. лорем ипсум траливали и тп.
  193. </p>
  194. </Spoiler>
  195. <div>Spoiler 2:</div>
  196. <Spoiler>
  197. <h2>Контент 2</h2>
  198. <p>
  199. лорем ипсум траливали и тп.
  200. </p>
  201. </Spoiler>
  202. <div>Check input length:</div>
  203. <RangeInput min={2} max={10}/>
  204. <PasswordConfirm min={2} />
  205. <Timer seconds={10000} />
  206. <TimerControl />
  207. <TimerContainer seconds={1800} refresh={100} render={LCD} />
  208. {/* <Watch sec={1800}/> */}
  209. <TimerContainerWatch seconds={1800} refresh={1000} render={Watch} />
  210. </div>
  211. );
  212. }
  213. export default App;