Conway's Game of Life as a React web app
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Controls.tsx 2.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. import React, { useEffect } from 'react';
  2. interface Props {
  3. running: boolean;
  4. speed: number;
  5. width: number;
  6. height: number;
  7. reset: () => void;
  8. toggle: () => void;
  9. updateCellSize: (f: (size: number) => number) => void;
  10. updateDimensions: (dimensions: { width?: number; height?: number }) => void;
  11. updateSpeed: (speed: number) => void;
  12. }
  13. const Controls = (props: Props) => {
  14. const {
  15. running,
  16. speed,
  17. toggle,
  18. updateSpeed,
  19. width,
  20. height,
  21. reset,
  22. updateCellSize,
  23. updateDimensions,
  24. } = props;
  25. const goSlower = () => {
  26. updateSpeed(speed + 100);
  27. };
  28. const goFaster = () => {
  29. if (speed > 100) {
  30. updateSpeed(speed - 100);
  31. }
  32. };
  33. const getSmaller = () => {
  34. updateCellSize((size) => size - 1);
  35. };
  36. const getBigger = () => {
  37. updateCellSize((size) => size + 1);
  38. };
  39. useEffect(() => {
  40. document.addEventListener('keyup', (e) => {
  41. if (e.key === ' ') {
  42. toggle();
  43. }
  44. });
  45. }, [toggle]);
  46. return (
  47. <div className="controls">
  48. <div className="controls__section dimensions">
  49. <label>
  50. Width{' '}
  51. <input
  52. className="number-field"
  53. type="number"
  54. value={width}
  55. onChange={(e) =>
  56. updateDimensions({ width: parseInt(e.currentTarget.value, 10) })
  57. }
  58. />
  59. </label>
  60. <label>
  61. Height{' '}
  62. <input
  63. className="number-field"
  64. type="number"
  65. value={height}
  66. onChange={(e) =>
  67. updateDimensions({ height: parseInt(e.currentTarget.value, 10) })
  68. }
  69. />
  70. </label>
  71. </div>
  72. <div className="controls__section speed">
  73. <button className="button" onClick={goSlower}>
  74. Slower
  75. </button>
  76. <button className="button" disabled={speed <= 100} onClick={goFaster}>
  77. Faster
  78. </button>
  79. </div>
  80. <div className="controls__section size">
  81. <button className="button" onClick={getSmaller}>
  82. Smaller
  83. </button>
  84. <button className="button" onClick={getBigger}>
  85. Bigger
  86. </button>
  87. </div>
  88. <div className="controls__section play-reset">
  89. <div className="reset">
  90. <button className="button" onClick={reset}>
  91. Reset
  92. </button>
  93. </div>
  94. <div className="play">
  95. <button className="button" onClick={toggle}>
  96. {running ? 'Pause' : 'Play'}
  97. </button>
  98. </div>
  99. </div>
  100. </div>
  101. );
  102. };
  103. export default Controls;