Subtraction.js 18 KB

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