Subtraction.js 18 KB

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