Addition.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. import React, { useState, useEffect, useRef } from "react";
  2. import DataGrid, {Column} from "devextreme-react/data-grid";
  3. import {numbersToArrOfArr, afterCommaLen, handleKeyDown} from "./helpers.js";
  4. import "./App.css";
  5. let imdtRes;
  6. let imdtResIdx;
  7. let resArr = [];
  8. let noOfDigits = 0;
  9. let commaIsSet = false;
  10. function Addition() {
  11. const [input, setInput] = useState(""); // initial user input like 34+56.7
  12. const [realResult, setRealResult] = useState(0); // real result of calculation
  13. const [carryArr, setCarryArr] = useState([]); // array of carries ["1","0"]
  14. const [commaIdx, setCommaIdx] = useState(-1); // int: comma position
  15. const [numbers, setNumbers] = useState(0); // array of arrays with input numbers
  16. const [resultsGrid, setResultsGrid] = useState([{id: 1, number: ""}]); // calculated result
  17. const [stepsGrid, setStepsGrid] = useState([{id: 1, step: ""}]); // one step for every digit
  18. const [autoMoveComma, setAutoMoveComma] = useState(false); // move the comma automatically
  19. const [allowStartOver, setAllowStartOver] = useState(false); // show steps and allow restart
  20. let calculationInput = useRef(null);
  21. useEffect(()=>{
  22. let autoMove = window.localStorage.getItem("autoMoveComma");
  23. if(autoMove !== null){
  24. setAutoMoveComma(JSON.parse(autoMove));
  25. console.log("auto move comma:", autoMove);
  26. }
  27. let allowStart = window.localStorage.getItem("allowStartOver");
  28. if(allowStart !== null){
  29. setAllowStartOver(JSON.parse(allowStart));
  30. console.log("allow start over:", allowStart);
  31. }
  32. // focus the input field
  33. if(calculationInput.current && calculationInput.current.value === ""){
  34. calculationInput.current.focus();
  35. }
  36. }, []);
  37. useEffect(()=>{
  38. if(!autoMoveComma){
  39. addNumbersToTable([]);
  40. }
  41. }, [numbers]);
  42. const handleInput = (e) => {
  43. setInput(e.target.value);
  44. };
  45. const handleResChange = (e) => {
  46. if(typeof e === "string"){
  47. imdtRes = e;
  48. }else{
  49. imdtRes = e.target.value;
  50. }
  51. // remove deleted input
  52. if(imdtRes === ""){
  53. resArr.shift();
  54. }else{
  55. // add input number to result array
  56. resArr.unshift(imdtRes);
  57. }
  58. };
  59. const handleCarryChange = (e, nosLeft) => {
  60. if(e.keyCode === 13){
  61. let carryArrCopy = [...carryArr];
  62. carryArrCopy.unshift(e.target.value);
  63. setCarryArr(carryArrCopy);
  64. let noCarry = carryArrCopy[0] === "" || carryArrCopy[0] === "0" || carryArrCopy[0] === undefined;
  65. if((nosLeft === 1 && noCarry) ||
  66. // if stop after first iteration, numbers left is undefined at first
  67. (isNaN(nosLeft) && realResult.toString().length === 1 && noCarry)){
  68. nosLeft = 0;
  69. handleResChange(" ");
  70. }
  71. if(nosLeft === 0){
  72. if(allowStartOver){
  73. showIdmtResults(carryArrCopy);
  74. addButtonsToImdtSteps(carryArrCopy);
  75. }else{
  76. finishCalculation();
  77. }
  78. }
  79. }
  80. };
  81. const handleSubmit = (e) => {
  82. startCalculation();
  83. e.preventDefault(); // avoid page reload
  84. };
  85. const startCalculation = () => {
  86. if(input.includes("+")){
  87. let numbers = input.split("+").map(x => parseFloat(x.replace(",",".")));
  88. let realRes = numbers.reduce((x,y) => x+y, 0);
  89. let afterComma = Math.max(...numbers.map(x => afterCommaLen(x)));
  90. realRes = parseFloat(realRes.toFixed(afterComma));
  91. setRealResult(realRes);
  92. console.log("real result:", realRes);
  93. let numbersArr, commaIdx;
  94. if(!autoMoveComma){
  95. numbersArr = numbers.map(x => x.toString().split(""));
  96. setNumbers(numbersArr);
  97. addNumbersToTable([]);
  98. startCommaMove();
  99. }else{
  100. commaIsSet = true;
  101. [numbersArr, commaIdx] = numbersToArrOfArr(numbers);
  102. setNumbers(numbersArr);
  103. setCommaIdx(commaIdx);
  104. addNumbersToTable(numbersArr);
  105. }
  106. }
  107. };
  108. const handleKeyMoveComma = (e) => {
  109. let pressedKey = e.keyCode;
  110. let tdId = e.target.id;
  111. let noId = tdId.split("numbersTd")[1];
  112. let numbersCopy = [...numbers];
  113. if(pressedKey === 37){ // left arrow
  114. if(numbersCopy[noId].includes(".")){
  115. numbersCopy[noId].push("0");
  116. }else{
  117. numbersCopy[noId].push(".");
  118. }
  119. setNumbers(numbersCopy);
  120. setTimeout(() => {
  121. document.getElementById(tdId).focus();
  122. }, 100);
  123. }else if(pressedKey === 39 && // right arrow
  124. [".", "0"].includes(numbersCopy[noId][numbersCopy[noId].length-1])){
  125. numbersCopy[noId].pop();
  126. setNumbers(numbersCopy);
  127. setTimeout(() => {
  128. document.getElementById(tdId).focus();
  129. }, 100);
  130. }else if(pressedKey === 38){ // up
  131. let nextFocusId = Math.max(noId-1, 0);
  132. document.getElementById("numbersTd" + nextFocusId).focus();
  133. }else if(pressedKey === 40){ // down
  134. let nextFocusId = Math.min(noId+1, numbers.length-1);
  135. document.getElementById("numbersTd" + nextFocusId).focus();
  136. }else if(pressedKey === 13){ // enter
  137. submitCommaMove();
  138. }
  139. };
  140. const addNumbersToTable = (numbersArr) => {
  141. let nos;
  142. if(numbersArr.length > 0){
  143. nos = numbersArr;
  144. }else{
  145. nos = numbers;
  146. }
  147. let table = document.getElementById("numbersGrid");
  148. table.innerHTML = "";
  149. for(let noIdx in nos){
  150. let row = table.insertRow();
  151. let cell = row.insertCell();
  152. let cellText = nos[noIdx].join("");
  153. cell.innerHTML = cellText;
  154. cell.tabIndex = "0";
  155. cell.id = "numbersTd" + noIdx;
  156. }
  157. };
  158. const ResultCarryForm = ({handleResChange, handleCarryChange}) => {
  159. let resInputField = useRef(null);
  160. let carryInputField = useRef(null);
  161. let labelText = "";
  162. let nosLeft = imdtResIdx + 1;
  163. let carryText = "Übertrag = ";
  164. if(commaIsSet && typeof numbers === "object" && imdtResIdx !== -1){
  165. document.getElementById("commaWarning").innerHTML = "";
  166. // set the digit index, start with the last digit
  167. if(typeof imdtResIdx === "undefined" || imdtResIdx === null){
  168. noOfDigits = Math.min(...numbers.map(n => n.length));
  169. imdtResIdx = noOfDigits - 1;
  170. }
  171. if(typeof nosLeft === "undefined" ){
  172. nosLeft = imdtResIdx + 1;
  173. }
  174. // skip over comma
  175. if(imdtResIdx === commaIdx){
  176. if(!resArr.includes(".")){
  177. handleResChange(".");
  178. }
  179. imdtResIdx = imdtResIdx - 1;
  180. nosLeft = nosLeft - 1;
  181. }
  182. // iterate numbers for this digit index
  183. for(let n in numbers){
  184. // skip last empty number if no carry
  185. if(nosLeft === 1 && (carryArr[0] === "0" || carryArr[0] === "")){
  186. nosLeft = 0;
  187. imdtResIdx = -1;
  188. return <></>;
  189. }
  190. // create label text like: d + d + .. = input
  191. let digit = numbers[parseInt(n)][imdtResIdx];
  192. if([".", "0", "&nbsp;", ""].includes(digit)){
  193. digit = 0;
  194. }
  195. labelText += digit;
  196. if(parseInt(n) === numbers.length - 1){
  197. if(carryArr[0] !== undefined && carryArr[0] > 0){
  198. labelText += " + " + carryArr[0].toString();
  199. }
  200. labelText += " = ";
  201. }else{
  202. labelText += " + ";
  203. }
  204. }
  205. imdtResIdx -= 1;
  206. nosLeft -= 1;
  207. return (
  208. <form>
  209. <label htmlFor="input_result" id="input_result_label">
  210. {labelText}
  211. </label>
  212. <input
  213. onKeyDown={(e) => handleKeyDown(e)}
  214. onChange={(e) => handleResChange(e)}
  215. type="text" id="input_result" size="2" maxLength="1"
  216. aria-labelledby="input_result_label" aria-required="true"
  217. ref={resInputField} autoFocus/>
  218. <br/>
  219. <label htmlFor="input_carry" id="input_carry_label">
  220. {carryText}
  221. </label>
  222. <input
  223. onKeyDown={(e) => handleCarryChange(e, nosLeft)}
  224. type="text" id="input_carry" size="2" maxLength="1"
  225. aria-labelledby="input_carry_label" aria-required="true"
  226. ref={carryInputField}/>
  227. </form>
  228. );
  229. }else{
  230. return (
  231. <>
  232. </>
  233. );
  234. }
  235. };
  236. const addButtonsToImdtSteps = (carryArrCopy) => {
  237. // workaround to access DOM td after grid values are set
  238. setTimeout(() => {
  239. let table = document.getElementById("idmtResultSteps").getElementsByTagName("table")[0];
  240. let trs = table.getElementsByTagName("tr");
  241. for(let trIdx in trs){
  242. let tr = trs[trIdx];
  243. if(tr instanceof HTMLElement && trIdx < trs.length-1){
  244. let td = tr.getElementsByTagName("td")[0];
  245. let btn = document.createElement("button");
  246. btn.innerHTML = "hier starten";
  247. btn.classList = "btn btn-secondary btn-sm";
  248. btn.id = "btn-" + trIdx;
  249. btn.addEventListener("click", () => startOver(trIdx, carryArrCopy));
  250. td.appendChild(btn);
  251. }
  252. }
  253. }, 300);
  254. };
  255. const showIdmtResults = (carryArrCopy) => {
  256. let realResArr = realResult.toString().split("");
  257. while(realResArr.length < resArr.length){
  258. realResArr.unshift("&nbsp;"); // add " " before
  259. }
  260. let carries = carryArrCopy.map(x => x || "0");
  261. carries.push("0");
  262. while(carries.length < resArr.length){
  263. carries.unshift("0"); // add "0" before
  264. }
  265. let foundComma = false;
  266. let stepsGridCopy = [];
  267. for(let i=0; i<resArr.length; i++) {
  268. let text = "";
  269. let trueNumbers = false;
  270. let idxNumbers = resArr.length - i - 1;
  271. let idxCarry = carries.length - i - 1;
  272. let realSum = 0;
  273. for(let j=0; j<numbers.length; j++){
  274. let no = numbers[j][idxNumbers];
  275. if(no!=="." && no!=="&nbsp;"){
  276. trueNumbers = true;
  277. realSum += parseInt(no);
  278. }
  279. text += no;
  280. if(j<numbers.length-1){
  281. text += " + ";
  282. }
  283. }
  284. // ignore indexes without real digits
  285. if(resArr[idxNumbers]!=="&nbsp;" || (carries[idxCarry]!=="&nbsp;" & carries[idxCarry]!=="0")){
  286. trueNumbers = true;
  287. }
  288. // carry array has no "."
  289. if(resArr[idxNumbers] === "."){
  290. foundComma = true;
  291. trueNumbers = false;
  292. }
  293. if(foundComma){
  294. idxCarry += 1;
  295. }
  296. // add carry only if > 0
  297. if(carries[idxCarry]!=="0"){
  298. realSum += parseInt(carries[idxCarry]);
  299. text += " + " + carries[idxCarry]; // + " Übertrag"
  300. }
  301. let realCarry = parseInt(realSum/10).toString();
  302. realSum = (realSum % 10).toString();
  303. text += " = ";
  304. if(trueNumbers){
  305. text += resArr[idxNumbers];
  306. text += " mit Übertrag " + carries[idxCarry-1];
  307. text = text.replace(/&nbsp;/g, "0");
  308. text += resArr[idxNumbers]===realSum && carries[idxCarry-1]===realCarry ? ": Richtig " : ": Falsch ";
  309. stepsGridCopy.push({step: text});
  310. }
  311. }
  312. let paragraph = document.getElementById("stepsParagraph");
  313. paragraph.innerHTML = "Rechenschritte: ";
  314. let btnSubmit = document.createElement("button");
  315. btnSubmit.innerHTML = "Ergebnis abgeben";
  316. btnSubmit.addEventListener("click", finishCalculation);
  317. btnSubmit.classList = "btn btn-secondary btn-sm";
  318. btnSubmit.id = "btnSubmitSteps1";
  319. document.getElementById("idmtResultSteps").insertBefore(btnSubmit, paragraph);
  320. setStepsGrid(stepsGridCopy);
  321. let btnSubmit2 = document.createElement("button");
  322. btnSubmit2.innerHTML = "Ergebnis abgeben";
  323. btnSubmit2.addEventListener("click", finishCalculation);
  324. btnSubmit2.classList = "btn btn-secondary btn-sm";
  325. btnSubmit2.id = "btnSubmitSteps2";
  326. document.getElementById("idmtResultSteps").appendChild(btnSubmit2);
  327. document.getElementById("idmtResultSteps").style.display = "inline-block";
  328. document.getElementById("idmtResultSteps").tabIndex = 0;
  329. document.getElementById("idmtResultSteps").focus();
  330. };
  331. const startOver = (idx, carryArrCopy) => {
  332. let invertedIdx = numbers[0].length - idx - 1;
  333. imdtResIdx = invertedIdx;
  334. let slicer = invertedIdx;
  335. if(resArr[0] !== "&nbsp;" && !resArr.includes(".")){
  336. slicer += 1;
  337. }else if(resArr[0] === "&nbsp;" && resArr.includes(".")){
  338. slicer -= 1;
  339. }
  340. carryArrCopy = carryArrCopy.slice(slicer);
  341. // reset result and carries until the chosen step
  342. let commaPos = resArr.indexOf(".");
  343. if(resArr.includes(".") && commaPos >= invertedIdx){
  344. imdtResIdx -= 1;
  345. resArr = resArr.slice(invertedIdx);
  346. }else{
  347. resArr = resArr.slice(invertedIdx + 1);
  348. }
  349. setCarryArr(carryArrCopy);
  350. document.getElementById("btnSubmitSteps1").remove();
  351. document.getElementById("btnSubmitSteps2").remove();
  352. document.getElementById("idmtResultSteps").style.display = "none";
  353. };
  354. const finishCalculation = () => {
  355. document.getElementById("idmtResultSteps").innerHTML = "";
  356. let resCalc = resArr.filter(n => n !== "&nbsp;").join("");
  357. setResultsGrid([{number: resCalc}]);
  358. resCalc = parseFloat(resCalc);
  359. let message = "";
  360. if(resCalc === realResult){
  361. message = "Richtig!";
  362. }else{
  363. message = "Das ist leider falsch.";
  364. }
  365. message = "<p>" + message + "</p>";
  366. document.getElementById("finishCalculation").innerHTML = message;
  367. document.getElementById("finishCalculation").tabIndex = "0";
  368. document.getElementById("finishCalculation").focus();
  369. };
  370. const startCommaMove = () => {
  371. document.getElementById("commaSubmit").style.display = "inline-block";
  372. setTimeout(() => {
  373. document.getElementById("numbersTd0").focus();
  374. }, 300);
  375. };
  376. const submitCommaMove = () => {
  377. let numbersCopy = [...numbers];
  378. let commaPositions = [];
  379. // reset numbers to calculate new length after error
  380. for(let noIdx in numbersCopy){
  381. while(numbersCopy[noIdx][0] === "&nbsp;"){
  382. numbersCopy[noIdx].shift();
  383. }
  384. }
  385. // get all numbers up to the same length
  386. let numbersLen = Math.max(...numbers.map(n => n.length));
  387. for(let noIdx in numbersCopy){
  388. while(numbersCopy[noIdx].length <= numbersLen){
  389. numbersCopy[noIdx].unshift("&nbsp;");
  390. }
  391. commaPositions.push(numbersCopy[noIdx].indexOf("."));
  392. }
  393. setNumbers(numbersCopy);
  394. let commaCorrect = false;
  395. commaPositions = [...new Set(commaPositions)];
  396. // if no commas in numbers
  397. if(commaPositions.length === 1 && commaPositions[0] !== numbersLen){
  398. setCommaIdx(commaPositions[0]);
  399. commaCorrect = true;
  400. }
  401. if(commaCorrect){
  402. document.getElementById("commaSubmit").style.display = "none";
  403. commaIsSet = true;
  404. }else{
  405. let paragraph = document.getElementById("commaWarning");
  406. paragraph.innerHTML = "das ist leider falsch, versuche es noch einmal:";
  407. paragraph.tabIndex = "0";
  408. paragraph.focus();
  409. }
  410. };
  411. return (
  412. <main>
  413. <h1>Addition</h1>
  414. <form onSubmit={(e) => handleSubmit(e)}>
  415. <label htmlFor="calculationInput">
  416. Rechnung:
  417. </label>
  418. <input
  419. type="text"
  420. id="calculationInput"
  421. onChange={(e) => handleInput(e)}
  422. aria-label="Rechnung"
  423. aria-required="true"
  424. ref={calculationInput}/>
  425. <input type="submit" value="berechnen"/>
  426. </form>
  427. <div id="finishCalculation"></div>
  428. <div id="commaWarning"></div>
  429. <div id="overview">
  430. <table
  431. id="numbersGrid"
  432. onKeyDown={handleKeyMoveComma}
  433. >
  434. </table>
  435. <input type="submit"
  436. value="Komma best&auml;tigen"
  437. id="commaSubmit"
  438. onClick={() => submitCommaMove()}/>
  439. <hr></hr>
  440. <DataGrid
  441. dataSource={resultsGrid}
  442. focusedRowEnabled={true}
  443. showBorders={true}
  444. showColumnHeaders={false}
  445. >
  446. <Column dataField="number" />
  447. </DataGrid>
  448. </div>
  449. <div id="calculation">
  450. <ResultCarryForm
  451. handleResChange={handleResChange}
  452. handleCarryChange={handleCarryChange}/>
  453. </div>
  454. <div id="idmtResultSteps">
  455. <p id="stepsParagraph"></p>
  456. <DataGrid
  457. dataSource={stepsGrid}
  458. focusedRowEnabled={true}
  459. showBorders={true}
  460. showColumnHeaders={false}
  461. noDataText=""
  462. >
  463. <Column dataField="step" />
  464. </DataGrid>
  465. </div>
  466. </main>
  467. );
  468. }
  469. export default Addition;