From 8cadccdbadad452e2a974a90bbf0c78d5f028d84 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sat, 26 Oct 2024 22:42:24 +0000 Subject: [PATCH] Deploy to GitHub pages --- asset-manifest.json | 13 ++++++++ favicon.ico | Bin 0 -> 3870 bytes index.html | 1 + logo192.png | Bin 0 -> 5347 bytes logo512.png | Bin 0 -> 9664 bytes manifest.json | 25 +++++++++++++++ robots.txt | 3 ++ static/css/main.101ab0f3.css | 2 ++ static/css/main.101ab0f3.css.map | 1 + static/js/main.202c3475.js | 3 ++ static/js/main.202c3475.js.LICENSE.txt | 41 +++++++++++++++++++++++++ static/js/main.202c3475.js.map | 1 + 12 files changed, 90 insertions(+) create mode 100644 asset-manifest.json create mode 100644 favicon.ico create mode 100644 index.html create mode 100644 logo192.png create mode 100644 logo512.png create mode 100644 manifest.json create mode 100644 robots.txt create mode 100644 static/css/main.101ab0f3.css create mode 100644 static/css/main.101ab0f3.css.map create mode 100644 static/js/main.202c3475.js create mode 100644 static/js/main.202c3475.js.LICENSE.txt create mode 100644 static/js/main.202c3475.js.map diff --git a/asset-manifest.json b/asset-manifest.json new file mode 100644 index 00000000..1b46c652 --- /dev/null +++ b/asset-manifest.json @@ -0,0 +1,13 @@ +{ + "files": { + "main.css": "/solver/static/css/main.101ab0f3.css", + "main.js": "/solver/static/js/main.202c3475.js", + "index.html": "/solver/index.html", + "main.101ab0f3.css.map": "/solver/static/css/main.101ab0f3.css.map", + "main.202c3475.js.map": "/solver/static/js/main.202c3475.js.map" + }, + "entrypoints": [ + "static/css/main.101ab0f3.css", + "static/js/main.202c3475.js" + ] +} \ No newline at end of file diff --git a/favicon.ico b/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..a11777cc471a4344702741ab1c8a588998b1311a GIT binary patch literal 3870 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 baj7?x%VD!5Cxk2YbYLNVeiXvvpMCWYo=by@ literal 0 HcmV?d00001 diff --git a/index.html b/index.html new file mode 100644 index 00000000..c7d52bc1 --- /dev/null +++ b/index.html @@ -0,0 +1 @@ +Sudoku solver
\ No newline at end of file diff --git a/logo192.png b/logo192.png new file mode 100644 index 0000000000000000000000000000000000000000..fc44b0a3796c0e0a64c3d858ca038bd4570465d9 GIT binary patch literal 5347 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 b{|{}U^kRn2wo}j7|0ATu<;8xA7zX}7|B6mN literal 0 HcmV?d00001 diff --git a/manifest.json b/manifest.json new file mode 100644 index 00000000..3823d9f9 --- /dev/null +++ b/manifest.json @@ -0,0 +1,25 @@ +{ + "short_name": "Sudoku solver", + "name": "Yet another sudoku solver", + "icons": [ + { + "src": "favicon.ico", + "sizes": "64x64 32x32 24x24 16x16", + "type": "image/x-icon" + }, + { + "src": "logo192.png", + "type": "image/png", + "sizes": "192x192" + }, + { + "src": "logo512.png", + "type": "image/png", + "sizes": "512x512" + } + ], + "start_url": ".", + "display": "standalone", + "theme_color": "#49515f", + "background_color": "#3e4451" +} diff --git a/robots.txt b/robots.txt new file mode 100644 index 00000000..e9e57dc4 --- /dev/null +++ b/robots.txt @@ -0,0 +1,3 @@ +# https://www.robotstxt.org/robotstxt.html +User-agent: * +Disallow: diff --git a/static/css/main.101ab0f3.css b/static/css/main.101ab0f3.css new file mode 100644 index 00000000..4acc2bb8 --- /dev/null +++ b/static/css/main.101ab0f3.css @@ -0,0 +1,2 @@ +*,:after,:before{background-repeat:no-repeat;box-sizing:border-box}:after,:before{text-decoration:inherit;vertical-align:inherit}:where(:root){-webkit-tap-highlight-color:transparent;-webkit-text-size-adjust:100%;cursor:default;line-height:1.5;overflow-wrap:break-word;tab-size:4}:where(body){margin:0}:where(h1){font-size:2em;margin:.67em 0}:where(dl,ol,ul) :where(dl,ol,ul){margin:0}:where(hr){color:inherit;height:0}:where(nav) :where(ol,ul){list-style-type:none;padding:0}:where(nav li):before{content:"\200B";float:left}:where(pre){font-family:monospace,monospace;font-size:1em;overflow:auto}:where(abbr[title]){text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted}:where(b,strong){font-weight:bolder}:where(code,kbd,samp){font-family:monospace,monospace;font-size:1em}:where(small){font-size:80%}:where(audio,canvas,iframe,img,svg,video){vertical-align:middle}:where(iframe){border-style:none}:where(svg:not([fill])){fill:currentColor}:where(table){border-collapse:collapse;border-color:inherit;text-indent:0}:where(button,input,select){margin:0}:where(button,[type=button i],[type=reset i],[type=submit i]){-webkit-appearance:button}:where(fieldset){border:1px solid #a0a0a0}:where(progress){vertical-align:initial}:where(textarea){margin:0;resize:vertical}:where([type=search i]){-webkit-appearance:textfield;outline-offset:-2px}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}::-webkit-input-placeholder{color:inherit;opacity:.54}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}:where(dialog){background-color:#fff;border:solid;color:#000;height:-moz-fit-content;height:fit-content;left:0;margin:auto;padding:1em;position:absolute;right:0;width:-moz-fit-content;width:fit-content}:where(dialog:not([open])){display:none}:where(details>summary:first-of-type){display:list-item}:where([aria-busy=true i]){cursor:progress}:where([aria-controls]){cursor:pointer}:where([aria-disabled=true i],[disabled]){cursor:not-allowed}:where([aria-hidden=false i][hidden]){display:initial}:where([aria-hidden=false i][hidden]:not(:focus)){clip:rect(0,0,0,0);position:absolute}body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,Fira Sans,Droid Sans,Helvetica Neue,sans-serif;margin:0}.monospace,code{font-family:var(--monospace)}:root{--monospace:source-code-pro,Menlo,Monaco,Consolas,"Courier New",monospace;color-scheme:dark;font-size:calc(.5em + 1vmin)}@media print{:root{--text-color:#000;--background-color:#fff;--link-color:#00e;--link-active-color:red;--link-visited-color:#551a8b}}:root{--relative-pixel:0.04ch;--regular-border:calc(var(--relative-pixel)*2);--larger-border:calc(var(--regular-border)*8);--large-border:calc(var(--larger-border)/3);--sudokucell-length:4rem;--cell-fontsize:2.5rem;--candidate-fontsize:0.9rem;--background-color:#282c34;--link-color:#61dafb;--link-visited-color:#d7a1e8;--link-active-color:tomato;--text-color:#f5f5f5;--border-color:#fff;--maroon:maroon;--red:#ec3e31;--orange:orange;--blue-gray:#8cadf3;--lighter-green:#8ea;--light-green:#0c0;--green:green;--purple:#639}.App{grid-gap:1rem;--app-main-height:calc(var(--sudokucell-length)*9 + var(--regular-border)*7*2 + var(--regular-border)*2*1 + var(--regular-border)*16/3);align-content:space-between;background-color:var(--background-color);color:var(--text-color);display:grid;gap:1rem;margin:0;min-height:100vh;min-width:100vw;position:relative;width:-moz-fit-content;width:fit-content}@media (min-width:900px){.App{gap:0 0;grid-template-areas:"Header Header Header Header Header" ". Main . Aside ." ". . . . .";grid-template-columns:250fr 1432fr 150fr 1314fr 250fr;grid-template-rows:auto var(--app-main-height) .3fr}.App-aside{--font-size:1rem}}@media (max-width:900px){.App{grid-template-areas:". Header ." ". Main ." ". Aside .";grid-template-columns:250fr 2745fr 250fr;grid-template-rows:auto var(--app-main-height) auto;justify-content:center}.App-aside{--font-size:calc(1rem + 1vh);font-size:var(--font-size);margin:1.5rem 0 25vh}}.App-header{font-size:calc(10px + 2vmin);grid-area:Header;height:-moz-fit-content;height:fit-content;padding:2vh 0}.App-header>h1.Title{display:inline;font-size:1em;margin:0;padding:0 2vw}.App-header>.Version{color:silver;font-size:.7em}.App.error{border:.5rem solid var(--red)}.App.error>.App-header:before{color:var(--red);content:"Error! "}.App-link{color:var(--link-color)}.App-link:visited{color:var(--link-visited-color)}.App-link:active{color:var(--link-active-color)}.App-aside{grid-area:Aside}.App-main{--sudoku-length:calc(var(--sudokucell-length)*9 + var(--regular-border)*7*2 + var(--regular-border)*2*1 + var(--large-border)*2);align-content:center;display:flex;grid-area:Main;justify-content:center}.App-main,.Sudoku{height:var(--sudoku-length)}.Sudoku{width:var(--sudoku-length);z-index:3}table.Sudoku{--cell-height:11.11111%;border:var(--larger-border) solid var(--border-color);border-collapse:collapse;border-spacing:0;cursor:pointer;font-family:var(--monospace);table-layout:fixed;text-align:center;vertical-align:middle}tr.Row{height:var(--cell-height)}tr.Row:nth-child(3n):not(:last-child)>td.Cell{border-bottom:var(--large-border) solid var(--border-color)}td.Cell{border:var(--regular-border) solid var(--border-color);border-spacing:0;font-size:var(--cell-fontsize);height:var(--cell-height);max-height:var(--cell-height);max-width:var(--cell-height);padding:0;text-align:center;vertical-align:middle;width:var(--cell-height)}td.Cell:nth-child(3n):not(:last-child){border-right:var(--large-border) solid var(--border-color)}[dir=rtl] td.Cell:nth-child(3n):not(:last-child),td.Cell:nth-child(3n):not(:last-child):dir(rtl){border-left:var(--large-border) solid var(--border-color);border-right:initial}div.Cell{background-color:initial;border:none;height:100%;padding:0;width:100%}div.Cell>*{height:var(--cell-height);overflow:hidden;text-overflow:clip}div.Cell>span.ugh.tables{align-items:center;display:inline-flex;height:100%}div.Cell[data-error=true]{background-color:var(--maroon)}div.Cell[data-active=true]{background-color:#555}div.Cell[data-active=false]{background:repeating-linear-gradient(-45deg,#d35e5e,#44476b,#333)}.Cell>.Loading{font-size:var(--candidate-fontsize);text-overflow:ellipsis}p.Candidates{display:grid;font-size:1rem;grid-template-columns:repeat(3,1fr);grid-template-rows:repeat(3,1fr);height:100%;margin:0;padding:2pt;width:100%}span.Candidate{color:inherit;font-size:var(--candidate-fontsize);line-height:normal;overflow:hidden;padding:0;text-align:center;text-overflow:clip}.Candidate.eliminated{background-color:#ff0}.Candidate.added{background-color:green}.Candidate.solved{color:#90ee90}.Candidate.blue,.Cell.blue{background-color:var(--blue-gray);color:#000}.Candidate.green,.Cell.green{background-color:#90ee90;color:#000}.Candidate.orange{background-color:orange;color:#000}.Cell.orange{background-color:#f0c575;color:#000}.Cell.salmon{background-color:salmon;color:#000}.Cell.orange.salmon{background-color:#f5a675;color:#000}.Candidate.blue.green,.Cell.blue.green{background-color:#8ecec2}.App-aside{border:inset;height:-moz-fit-content;height:fit-content}.StrategyList{margin-bottom:0;margin-left:1rem;margin-top:0;min-width:-moz-fit-content;min-width:fit-content}.StrategyItem.isCurrent{border-right:5px solid #b7c1c2}@supports selector(:dir(rtl)){.StrategyItem.isCurrent:dir(rtl){border-left:5px solid #b7c1c2;border-right:initial}}@supports not selector(:dir(rtl)){.StrategyItem.isCurrent[dir=rtl]{border-left:5px solid #b7c1c2;border-right:initial}}.StrategyItem:nth-child(odd){background-color:#1f2329}.StrategyItem:nth-child(2n){background-color:#31353f}.StrategyItem.disabled{color:#aaa}.StrategyItem.disabled:nth-child(odd){background-color:#282b31}.StrategyItem.disabled:nth-child(2n){background-color:#31353c}.StrategyList{--item-height:var(--font-size);max-width:none;padding-left:calc(2rem + var(--item-height));position:relative}.StrategyItem{--label-width:80%;--status-width:20%;--padding:1.5rem;line-height:calc(var(--item-height) + .1rem);max-width:calc(100% - var(--padding));min-width:calc(var(--label-width) + var(--status-width));padding-left:var(--padding)}.StrategyToggler{height:var(--item-height);left:-1rem;line-height:inherit;margin:0;position:absolute;text-align:center;width:var(--item-height)}@supports selector(:dir(rtl)){.StrategyList:dir(rtl){padding-left:0;padding-right:calc(2rem + var(--item-height))}.StrategyItem:dir(rtl){padding-left:0;padding-right:var(--padding)}.StrategyToggler:dir(rtl){left:auto;right:0}}@supports not selector(:dir(rtl)){[dir=rtl] .StrategyList{padding-left:0;padding-right:calc(2rem + var(--item-height))}[dir=rtl] .StrategyItem{padding-left:0;padding-right:var(--padding)}[dir=rtl] .StrategyToggler{left:auto;right:0}}.StrategyLabel{display:inline-grid;width:var(--label-width)}.StrategyTogglerLabel{color:#0000;height:var(--item-height);overflow:hidden;position:absolute;right:0;text-overflow:clip;-webkit-user-select:none;user-select:none;width:100%}a[href]+label>.StrategyTogglerLabel{width:20%}.StrategyResult{grid-area:status}.StrategyResult.success{color:var(--light-green)}.StrategyResult.fail{color:var(--orange)}.StrategyResult.error{color:var(--red)}.StrategyDetails{border:1px solid #a0a0a0;margin-inline-end:2px;margin-inline-start:2px;padding:.5em}.StrategyDetails>p{margin:0;white-space:pre-wrap;word-break:break-word}.Tabs{display:grid;grid-template-columns:repeat(2,1fr)}.Tab.selected{background-color:var(--background-color);border:none}.Tab.unselected{background-color:#505050;border-style:outset}.Tab.unselected:hover{background-color:#767676}.Tab.unselected:active{background-color:#373737;border-style:inset}.AlertNotice{word-wrap:break-word;background-color:#444;border-style:solid;border-width:var(--regular-border);bottom:2vh;display:flex;flex-direction:column;justify-content:space-between;max-width:calc(100% - 4rem);padding:0 1rem 1rem;position:fixed;right:2vw;width:-moz-fit-content;width:fit-content;word-break:break-all;z-index:4}.AlertNotice,.AlertNotice>p{height:-moz-fit-content;height:fit-content}.AlertNotice>p{font-size:1rem}.AlertNotice>button{font-size:2rem}.AlertNotice>*{position:sticky}.AlertNotice.info{border-color:var(--blue-gray)}.AlertNotice.warning{border-color:var(--orange)}.AlertNotice.error{border-color:var(--red)}.PromptWindow{background-color:#7777;display:flex;height:100vh;position:fixed;width:100vw;z-index:4}.PromptNotice{word-wrap:break-word;align-content:center;align-self:center;background-color:var(--background-color);display:inline-flex;flex-direction:column;height:-moz-fit-content;height:fit-content;justify-content:center;margin:0 auto;padding:1rem;position:relative;text-align:center;width:-moz-fit-content;width:fit-content;word-break:break-all;z-index:4}.PromptNotice>label{font-size:1.5rem}.PromptNotice>button{font-size:1.5rem;width:100%}.PromptNotice>label>p{margin-bottom:1rem;margin-top:0}.PromptNotice>label>textarea{width:100%}.github-corner{border:0;color:var(--text-color);position:absolute;right:0;top:0}.github-corner>svg{fill:#151513}.octo-arm{transform-origin:130px 106px}@keyframes octocat-wave{0%,to{transform:rotate(0deg)}20%,60%{transform:rotate(-25deg)}40%,80%{transform:rotate(10deg)}}@media (prefers-reduced-motion:no-preference){.github-corner:hover .octo-arm{animation:octocat-wave .56s ease-in-out}@media (max-width:500px){.github-corner:hover .octo-arm{animation:none}.github-corner .octo-arm{animation:octocat-wave .56s ease-in-out}}}@media (prefers-reduced-motion:reduce){.github-corner:hover .octo-arm{animation:octocat-wave 7s ease-in-out}@media (max-width:500px){.github-corner:hover .octo-arm{animation:none}.github-corner .octo-arm{animation:octocat-wave 7s ease-in-out}}}.ErrorNotice{border:3px solid var(--red);font-family:var(--monospace);height:-moz-fit-content;height:fit-content;padding:1.5rem;white-space:pre;width:-moz-fit-content;width:fit-content} +/*# sourceMappingURL=main.101ab0f3.css.map*/ \ No newline at end of file diff --git a/static/css/main.101ab0f3.css.map b/static/css/main.101ab0f3.css.map new file mode 100644 index 00000000..ac47633d --- /dev/null +++ b/static/css/main.101ab0f3.css.map @@ -0,0 +1 @@ +{"version":3,"file":"static/css/main.101ab0f3.css","mappings":"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","sources":["../node_modules/sanitize.css/sanitize.css","index.css","App.css","Elems/Main.css","Elems/MainElems/Sudoku.css","Elems/MainElems/Candidates.css","Elems/MainElems/Candidate.css","Elems/Aside.css","Elems/AsideElems/StrategyList.css","Elems/AsideElems/StrategyItem.css","Elems/AsideElems/StrategyDetails.css","Elems/AsideElems/Tabs.css","Elems/NoticeElems/AlertNotice.css","Elems/NoticeElems/PromptWindow.css","Elems/GithubCorner.css","ErrorNotice.css"],"sourcesContent":["/* Document\n * ========================================================================== */\n\n/**\n * 1. Add border box sizing in all browsers (opinionated).\n * 2. Backgrounds do not repeat by default (opinionated).\n */\n\n*,\n::before,\n::after {\n box-sizing: border-box; /* 1 */\n background-repeat: no-repeat; /* 2 */\n}\n\n/**\n * 1. Add text decoration inheritance in all browsers (opinionated).\n * 2. Add vertical alignment inheritance in all browsers (opinionated).\n */\n\n::before,\n::after {\n text-decoration: inherit; /* 1 */\n vertical-align: inherit; /* 2 */\n}\n\n/**\n * 1. Use the default cursor in all browsers (opinionated).\n * 2. Change the line height in all browsers (opinionated).\n * 3. Breaks words to prevent overflow in all browsers (opinionated).\n * 4. Use a 4-space tab width in all browsers (opinionated).\n * 5. Remove the grey highlight on links in iOS (opinionated).\n * 6. Prevent adjustments of font size after orientation changes in iOS.\n */\n\n:where(:root) {\n cursor: default; /* 1 */\n line-height: 1.5; /* 2 */\n overflow-wrap: break-word; /* 3 */\n -moz-tab-size: 4; /* 4 */\n tab-size: 4; /* 4 */\n -webkit-tap-highlight-color: transparent; /* 5 */\n -webkit-text-size-adjust: 100%; /* 6 */\n}\n\n/* Sections\n * ========================================================================== */\n\n/**\n * Remove the margin in all browsers (opinionated).\n */\n\n:where(body) {\n margin: 0;\n}\n\n/**\n * Correct the font size and margin on `h1` elements within `section` and\n * `article` contexts in Chrome, Edge, Firefox, and Safari.\n */\n\n:where(h1) {\n font-size: 2em;\n margin: 0.67em 0;\n}\n\n/* Grouping content\n * ========================================================================== */\n\n/**\n * Remove the margin on nested lists in Chrome, Edge, and Safari.\n */\n\n:where(dl, ol, ul) :where(dl, ol, ul) {\n margin: 0;\n}\n\n/**\n * 1. Correct the inheritance of border color in Firefox.\n * 2. Add the correct box sizing in Firefox.\n */\n\n:where(hr) {\n color: inherit; /* 1 */\n height: 0; /* 2 */\n}\n\n/**\n * Remove the list style on navigation lists in all browsers (opinionated).\n */\n\n:where(nav) :where(ol, ul) {\n list-style-type: none;\n padding: 0;\n}\n\n/**\n * Prevent VoiceOver from ignoring list semantics in Safari (opinionated).\n */\n\n:where(nav li)::before {\n content: \"\\200B\";\n float: left;\n}\n\n/**\n * 1. Correct the inheritance and scaling of font size in all browsers.\n * 2. Correct the odd `em` font sizing in all browsers.\n * 3. Prevent overflow of the container in all browsers (opinionated).\n */\n\n:where(pre) {\n font-family: monospace, monospace; /* 1 */\n font-size: 1em; /* 2 */\n overflow: auto; /* 3 */\n}\n\n/* Text-level semantics\n * ========================================================================== */\n\n/**\n * Add the correct text decoration in Safari.\n */\n\n:where(abbr[title]) {\n text-decoration: underline;\n text-decoration: underline dotted;\n}\n\n/**\n * Add the correct font weight in Chrome, Edge, and Safari.\n */\n\n:where(b, strong) {\n font-weight: bolder;\n}\n\n/**\n * 1. Correct the inheritance and scaling of font size in all browsers.\n * 2. Correct the odd `em` font sizing in all browsers.\n */\n\n:where(code, kbd, samp) {\n font-family: monospace, monospace; /* 1 */\n font-size: 1em; /* 2 */\n}\n\n/**\n * Add the correct font size in all browsers.\n */\n\n:where(small) {\n font-size: 80%;\n}\n\n/* Embedded content\n * ========================================================================== */\n\n/*\n * Change the alignment on media elements in all browsers (opinionated).\n */\n\n:where(audio, canvas, iframe, img, svg, video) {\n vertical-align: middle;\n}\n\n/**\n * Remove the border on iframes in all browsers (opinionated).\n */\n\n:where(iframe) {\n border-style: none;\n}\n\n/**\n * Change the fill color to match the text color in all browsers (opinionated).\n */\n\n:where(svg:not([fill])) {\n fill: currentColor;\n}\n\n/* Tabular data\n * ========================================================================== */\n\n/**\n * 1. Collapse border spacing in all browsers (opinionated).\n * 2. Correct table border color inheritance in all Chrome, Edge, and Safari.\n * 3. Remove text indentation from table contents in Chrome, Edge, and Safari.\n */\n\n:where(table) {\n border-collapse: collapse; /* 1 */\n border-color: inherit; /* 2 */\n text-indent: 0; /* 3 */\n}\n\n/* Forms\n * ========================================================================== */\n\n/**\n * Remove the margin on controls in Safari.\n */\n\n:where(button, input, select) {\n margin: 0;\n}\n\n/**\n * Correct the inability to style buttons in iOS and Safari.\n */\n\n:where(button, [type=\"button\" i], [type=\"reset\" i], [type=\"submit\" i]) {\n -webkit-appearance: button;\n}\n\n/**\n * Change the inconsistent appearance in all browsers (opinionated).\n */\n\n:where(fieldset) {\n border: 1px solid #a0a0a0;\n}\n\n/**\n * Add the correct vertical alignment in Chrome, Edge, and Firefox.\n */\n\n:where(progress) {\n vertical-align: baseline;\n}\n\n/**\n * 1. Remove the margin in Firefox and Safari.\n * 3. Change the resize direction in all browsers (opinionated).\n */\n\n:where(textarea) {\n margin: 0; /* 1 */\n resize: vertical; /* 3 */\n}\n\n/**\n * 1. Correct the odd appearance in Chrome, Edge, and Safari.\n * 2. Correct the outline style in Safari.\n */\n\n:where([type=\"search\" i]) {\n -webkit-appearance: textfield; /* 1 */\n outline-offset: -2px; /* 2 */\n}\n\n/**\n * Correct the cursor style of increment and decrement buttons in Safari.\n */\n\n::-webkit-inner-spin-button,\n::-webkit-outer-spin-button {\n height: auto;\n}\n\n/**\n * Correct the text style of placeholders in Chrome, Edge, and Safari.\n */\n\n::-webkit-input-placeholder {\n color: inherit;\n opacity: 0.54;\n}\n\n/**\n * Remove the inner padding in Chrome, Edge, and Safari on macOS.\n */\n\n::-webkit-search-decoration {\n -webkit-appearance: none;\n}\n\n/**\n * 1. Correct the inability to style upload buttons in iOS and Safari.\n * 2. Change font properties to `inherit` in Safari.\n */\n\n::-webkit-file-upload-button {\n -webkit-appearance: button; /* 1 */\n font: inherit; /* 2 */\n}\n\n/* Interactive\n * ========================================================================== */\n\n/*\n * Add the correct styles in Safari.\n */\n\n:where(dialog) {\n background-color: white;\n border: solid;\n color: black;\n height: -moz-fit-content;\n height: fit-content;\n left: 0;\n margin: auto;\n padding: 1em;\n position: absolute;\n right: 0;\n width: -moz-fit-content;\n width: fit-content;\n}\n\n:where(dialog:not([open])) {\n display: none;\n}\n\n/*\n * Add the correct display in Safari.\n */\n\n:where(details > summary:first-of-type) {\n display: list-item;\n}\n\n/* Accessibility\n * ========================================================================== */\n\n/**\n * Change the cursor on busy elements in all browsers (opinionated).\n */\n\n:where([aria-busy=\"true\" i]) {\n cursor: progress;\n}\n\n/*\n * Change the cursor on control elements in all browsers (opinionated).\n */\n\n:where([aria-controls]) {\n cursor: pointer;\n}\n\n/*\n * Change the cursor on disabled, not-editable, or otherwise\n * inoperable elements in all browsers (opinionated).\n */\n\n:where([aria-disabled=\"true\" i], [disabled]) {\n cursor: not-allowed;\n}\n\n/*\n * Change the display on visually hidden accessible elements\n * in all browsers (opinionated).\n */\n\n:where([aria-hidden=\"false\" i][hidden]) {\n display: initial;\n}\n\n:where([aria-hidden=\"false\" i][hidden]:not(:focus)) {\n clip: rect(0, 0, 0, 0);\n position: absolute;\n}\n","@import url(sanitize.css);\n\nbody {\n margin: 0;\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',\n 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',\n sans-serif;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n}\n\n.monospace,\ncode {\n font-family: var(--monospace);\n}\n\n:root {\n --monospace: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', monospace;\n\n /* Make font-size relative to screen size */\n font-size: calc(0.5em + 1vmin);\n\n /* Only supports dark-mode for now */\n color-scheme: dark;\n}\n\n@media print {\n :root {\n --text-color: black;\n --background-color: white;\n --link-color: rgb(0, 0, 238);\n --link-active-color: red;\n --link-visited-color: rgb(85, 26, 139);\n }\n}\n","\n/* root variables */\n:root {\n --relative-pixel: calc(1ch / 25);\n\n /* Border variables to make it look like a sudoku */\n --regular-border: calc(2 * var(--relative-pixel));\n --larger-border: calc(8 * var(--regular-border));\n --large-border: calc(var(--larger-border) / 3);\n\n --sudokucell-length: 4rem;\n --cell-fontsize: 2.5rem; /* fontsize of a solved digit */\n --candidate-fontsize: 0.9rem;\n\n /* colors */\n --background-color: #282c34; /* dark mode */\n --link-color: #61dafb;\n --link-visited-color: #d7a1e8;\n --link-active-color: tomato;\n --text-color: whitesmoke;\n --border-color: white;\n --maroon: maroon; /* rgb(139, 0, 0)? */\n --red: #ec3e31;\n --orange: orange;\n --blue-gray: #8cadf3;\n --lighter-green: #8ea;\n --light-green: #0c0;\n --green: green;\n --purple: rebeccapurple;\n}\n\n.App {\n background-color: var(--background-color);\n color: var(--text-color);\n\n /* For the github-corner */\n position: relative;\n\n min-height: 100vh;\n min-width: 100vw;\n width: fit-content;\n margin: 0;\n\n display: grid;\n gap: 1rem;\n align-content: space-between;\n /* default moved to min-width media querys */\n\n /* See app.main */\n --app-main-height: calc(\n calc( /* --sudoku-length */\n 9 * var(--sudokucell-length) +\n calc(7 * var(--regular-border) * 2) +\n calc(2 * var(--regular-border) * 1) +\n calc(2 * 8 * var(--regular-border) / 3)\n )\n );\n}\n\n/* width >= 900px, 2 column layout*/\n@media (min-width: 900px) {\n .App {\n grid-template-columns: 250fr 1432fr 150fr 1314fr 250fr; /* ~ (8%, 44%, 40%, 8%) */\n grid-template-rows: auto var(--app-main-height) 0.3fr; /* prev ~ (7%, 7%, 2rem, 72%, 7%) */\n gap: 0 0;\n grid-template-areas:\n \"Header Header Header Header Header\"\n \". Main . Aside .\"\n \". . . . .\";\n }\n\n /* Declares variable for consistency (see below) */\n .App-aside {\n --font-size: 1rem;\n }\n}\n\n/* width <= 900px, 1 column layout */\n@media (max-width: 900px) {\n .App {\n grid-template-columns: 250fr 2745fr 250fr; /* ~ (8%, 84%, 8%) */\n grid-template-rows: auto var(--app-main-height) auto;\n grid-template-areas:\n \". Header .\"\n \". Main .\"\n \". Aside .\";\n\n justify-content: center;\n }\n\n /**\n * Bigger text (aside)\n * Variable exists because of the strategyItem shenanigans\n * Margin: Gap + Allow scrolling past the aside in one column mode\n */\n .App-aside {\n --font-size: calc(1rem + 1vh);\n font-size: var(--font-size);\n margin: 1.5rem 0 25vh 0;\n }\n}\n\n/*\n@keyframes App-logo-spin {\n from {\n transform: rotate(0deg);\n }\n to {\n transform: rotate(360deg);\n }\n}\n\n.App-logo {\n height: 40vmin;\n pointer-events: none;\n}\n\n@media (prefers-reduced-motion: no-preference) {\n .App-logo {\n animation: App-logo-spin infinite 20s linear;\n }\n}\n*/\n\n.App-header {\n grid-area: Header;\n padding: 2vh 0;\n height: fit-content;\n font-size: calc(10px + 2vmin);\n}\n\n.App-header > h1.Title {\n display: inline;\n font-size: 1em;\n padding: 0 2vw;\n margin: 0;\n}\n\n.App-header > .Version {\n font-size: 0.7em;\n color: silver;\n}\n\n/* TODO: Fix layout when error */\n.App.error {\n border: 0.5rem solid var(--red);\n}\n\n.App.error > .App-header::before {\n content: \"Error! \";\n color: var(--red);\n}\n\n\n.App-link {\n color: var(--link-color);\n}\n\n.App-link:visited {\n color: var(--link-visited-color);\n}\n\n.App-link:active {\n color: var(--link-active-color);\n}\n\n.App-aside { grid-area: Aside; }\n.App-main { grid-area: Main; }\n\n\n\n",".App-main {\n /* variables (inherited by children) */\n --sudoku-length: calc(\n 9 * var(--sudokucell-length) +\n calc(7 * var(--regular-border) * 2) +\n calc(2 * var(--regular-border) * 1) +\n calc(2 * var(--large-border))\n ); /*\n Length = Row width (border doesn't count because of box-sizing: border-box)\n\n Row width = 9 cells\n 7 cells = width + 2regular-border\n 2 cells = width + 1regular-border + 1large-border\n\n total = 9 * width +\n (7 * 2 * regular) +\n (2 * 1 * regular) +\n (2 * large)\n */\n\n display: flex;\n height: var(--sudoku-length);\n justify-content: center;\n align-content: center;\n}\n\n.Sudoku {\n width: var(--sudoku-length);\n height: var(--sudoku-length);\n\n z-index: 3;\n}\n","\n/*\n References:\n [1]: https://stackoverflow.com/questions/4457506\n [2]: https://stackoverflow.com/questions/13667941\n*/\n\ntable.Sudoku {\n border: var(--larger-border) solid var(--border-color);\n border-collapse: collapse;\n border-spacing: 0;\n\n text-align: center;\n font-family: var(--monospace);\n\n /* [ref 1] - height and width are set in Main.css */\n table-layout: fixed;\n\n /* Also --cell-width */\n --cell-height: calc(100% / 9);\n\n /* The following variables are inherited by :root\n --cell-fontsize: 2.5rem;\n --candidate-fontsize: 0.9rem;\n */\n\n vertical-align: middle;\n cursor: pointer;\n}\n\ntr.Row {\n height: var(--cell-height);\n}\n\ntr.Row:nth-child(3n):not(:last-child) > td.Cell {\n border-bottom: var(--large-border) solid var(--border-color);\n}\n\ntd.Cell {\n width: var(--cell-height);\n height: var(--cell-height); /* part of trying @[ref 1] */\n max-width: var(--cell-height);\n max-height: var(--cell-height);\n border: var(--regular-border) solid var(--border-color);\n border-spacing: 0;\n font-size: var(--cell-fontsize);\n\n padding: 0;\n\n text-align: center;\n vertical-align: middle;\n}\n\ntd.Cell:nth-child(3n):not(:last-child) {\n border-right: var(--large-border) solid var(--border-color);\n}\n\n[dir=rtl] td.Cell:nth-child(3n):not(:last-child),\ntd.Cell:nth-child(3n):not(:last-child):dir(rtl) {\n border-right: initial;\n border-left: var(--large-border) solid var(--border-color);\n}\n\ndiv.Cell {\n border: none;\n width: 100%;\n height: 100%;\n padding: 0;\n background-color: transparent;\n}\n\ndiv.Cell > * {\n /* [ref 2] */\n height: var(--cell-height);\n\n /* & [ref 1] */\n overflow: hidden;\n text-overflow: clip;\n}\n\n/* This selector was made before I needed to use it\n * This is for vertically aligning the big digits\n */\ndiv.Cell > span.ugh.tables {\n display: inline-flex;\n height: 100%;\n align-items: center;\n}\n\ndiv.Cell[data-error=\"true\"] {\n background-color: var(--maroon);\n}\n\ndiv.Cell[data-active=\"true\"] {\n background-color: #555;\n}\n\ndiv.Cell[data-active=\"false\"] {\n background: repeating-linear-gradient(-45deg, #d35e5e, #44476b, #333);\n}\n\n/* Section moved to Candidates.css */\n\n/* For the lazy loaded props */\n.Cell > .Loading {\n font-size: var(--candidate-fontsize);\n text-overflow: ellipsis;\n}\n\n","\n/* Moved from Sudoku.css */\n\np.Candidates {\n margin: 0;\n padding: 2pt;\n width: 100%;\n height: 100%;\n\n display: grid;\n grid-template-columns: repeat(3, 1fr);\n grid-template-rows: repeat(3, 1fr);\n font-size: 1rem;\n}\n\nspan.Candidate {\n padding: 0;\n line-height: normal;\n\n font-size: var(--candidate-fontsize);\n text-align: center;\n color: inherit;\n\n /* [ref 1] */\n overflow: hidden;\n text-overflow: clip;\n}\n","\n.Candidate.eliminated {\n background-color: yellow;\n}\n\n.Candidate.added {\n background-color: green;\n}\n\n.Candidate.solved {\n color: lightgreen;\n}\n\n.Cell.blue,\n.Candidate.blue {\n background-color: var(--blue-gray);\n color: black;\n}\n\n.Cell.green,\n.Candidate.green {\n background-color: lightgreen;\n color: black;\n}\n\n.Candidate.orange {\n background-color: orange;\n color: black;\n}\n\n/* More like tan instead of orange, but at least my eyes don't hurt from the contrast */\n.Cell.orange {\n background-color: hsl(39, 80%, 70%); /* orange - 20 sat + 20 lig */\n color: black;\n}\n\n.Cell.salmon {\n background-color: salmon;\n color: black;\n}\n\n.Cell.orange.salmon {\n background-color: hsl(23, 87%, 71%);\n color: black;\n}\n\n/* Halfway between the styles in blue and green */\n.Cell.blue.green,\n.Candidate.blue.green {\n background-color: #8ecec2;\n}\n",".App-aside {\n height: fit-content;\n border: inset;\n}\n\n.StrategyList {\n margin-left: 1rem;\n margin-top: 0;\n margin-bottom: 0;\n}\n","\n/* Overflow */\n.StrategyList {\n min-width: fit-content;\n}\n\n/* The current strategy item */\n.StrategyItem.isCurrent {\n border-right: 5px solid rgb(183, 193, 194);\n}\n\n@supports selector(:dir(rtl)) {\n .StrategyItem.isCurrent:dir(rtl) {\n border-right: initial;\n border-left: 5px solid rgb(183, 193, 194);\n }\n}\n\n@supports not selector(:dir(rtl)) {\n .StrategyItem.isCurrent[dir=rtl] {\n border-right: initial;\n border-left: 5px solid rgb(183, 193, 194);\n }\n}\n\n/* Alternating colors - note that the markers are not in color */\n.StrategyItem:nth-child(odd) {\n background-color: #1f2329;\n}\n\n.StrategyItem:nth-child(even) {\n background-color: #31353f;\n}\n\n/* Disabled strategy item */\n.StrategyItem.disabled {\n color: rgb(170, 170, 170);\n}\n\n.StrategyItem.disabled:nth-child(odd) {\n background-color: #282b31;\n}\n\n.StrategyItem.disabled:nth-child(even) {\n background-color: #31353c;\n}\n","\n /* Checkbox positioning */\n.StrategyList {\n /*\n These variables were moved from .StrategyItem\n\n The checkbox needs to be --item-height\n This impacts the strategy list, since the padding-left must account for the checkbox\n */\n --item-height: var(--font-size); /* see App.css */\n\n position: relative;\n max-width: initial;\n padding-left: calc(2rem + var(--item-height));\n}\n\n/* Sidenote discovery: Google Chrome doesn't know that 0 is a valid when using max() */\n.StrategyItem {\n --label-width: 80%; /* For status */\n --status-width: 20%; /* For status */\n --padding: 1.5rem;\n\n padding-left: var(--padding);\n\n /* But this is for list item positioning */\n line-height: calc(var(--item-height) + 0.1rem);\n\n /* Making room for status */\n min-width: calc(var(--label-width) + var(--status-width));\n max-width: calc(100% - var(--padding));\n}\n\n.StrategyToggler {\n position: absolute;\n left: -1rem;\n margin: 0; /* Now sanitize.css works! */\n text-align: center;\n line-height: inherit;\n\n width: var(--item-height);\n height: var(--item-height);\n}\n\n@supports selector(:dir(rtl)) {\n .StrategyList:dir(rtl) {\n padding-left: initial;\n padding-right: calc(2rem + var(--item-height));\n }\n\n .StrategyItem:dir(rtl) {\n padding-left: initial;\n padding-right: var(--padding);\n }\n\n .StrategyToggler:dir(rtl) {\n left: initial;\n right: 0;\n }\n}\n\n@supports not selector(:dir(rtl)) {\n [dir=rtl] .StrategyList {\n padding-left: initial;\n padding-right: calc(2rem + var(--item-height));\n }\n\n [dir=rtl] .StrategyItem {\n padding-left: initial;\n padding-right: var(--padding);\n }\n\n [dir=rtl] .StrategyToggler {\n left: initial;\n right: 0;\n }\n}\n\n/* Label structure: Name Status */\n.StrategyLabel {\n width: var(--label-width);\n display: inline-grid; /* Still inline but behaves like a block element */\n\n /*\n Right now this doesn't do anything, (besides display)\n since the StrategyLabel doesn't contain a tooltip anymore.\n\n This would make more sense in a parent \"StrategyItemContent\" span\n As of right now, the \"StrategyStatus\" are aligned by the constant width.\n\n align-content: center;\n display: inline-grid;\n grid-template-columns: 3fr 1fr;\n grid-template-areas:\n 'name status';\n */\n}\n\n.StrategyTogglerLabel {\n position: absolute;\n color: transparent;\n user-select: none;\n overflow: hidden;\n text-overflow: clip;\n\n /* Tap size */\n height: var(--item-height);\n width: 100%;\n right: 0;\n}\n\n/* If there's a link don't block it */\n/* (a[href] + label) selects label */\na[href] + label > .StrategyTogglerLabel {\n width: 20%;\n}\n\n/* Result */\n.StrategyResult {\n grid-area: status;\n}\n\n.StrategyResult.success {\n color: var(--light-green);\n}\n\n.StrategyResult.fail {\n color: var(--orange);\n}\n\n.StrategyResult.error {\n color: var(--red);\n}\n",".StrategyDetails {\n margin-inline-start: 2px;\n margin-inline-end: 2px;\n border: 1px solid #a0a0a0;\n padding: 0.5em;\n}\n\n.StrategyDetails > p {\n margin: 0;\n white-space: pre-wrap;\n word-break: break-word;\n}\n","\n.Tabs {\n display: grid;\n grid-template-columns: repeat(2, 1fr);\n}\n\n.Tab.selected {\n border: none;\n background-color: var(--background-color);\n}\n\n.Tab.unselected {\n border-style: outset;\n background-color: rgb(80, 80, 80);\n}\n\n.Tab.unselected:hover {\n background-color: rgb(118, 118, 118);\n}\n\n.Tab.unselected:active {\n background-color: rgb(55, 55, 55);\n border-style: inset;\n}\n","\n.AlertNotice {\n position: fixed;\n bottom: 2vh;\n right: 2vw;\n height: fit-content;\n width: fit-content;\n max-width: calc(100% - 4rem);\n\n background-color: #444;\n z-index: 4;\n\n display: flex;\n padding: 0 1rem 1rem 1rem;\n justify-content: space-between;\n flex-direction: column;\n\n border-width: var(--regular-border);\n border-style: solid;\n\n word-wrap: break-word;\n word-break: break-all;\n}\n\n/** Positioning */\n.AlertNotice > p {\n height: fit-content;\n font-size: 1rem;\n}\n\n.AlertNotice > button {\n font-size: 2rem;\n}\n\n.AlertNotice > * {\n position: sticky;\n}\n\n\n/** Types of alert notices */\n.AlertNotice.info {\n border-color: var(--blue-gray);\n}\n\n.AlertNotice.warning {\n border-color: var(--orange);\n}\n\n.AlertNotice.error {\n border-color: var(--red);\n}\n","\n.PromptWindow {\n position: fixed; /** prevent escape! Note that aria-modal is implicitly true */\n width: 100vw;\n height: 100vh;\n background-color: #7777;\n display: flex;\n z-index: 4;\n}\n\n.PromptNotice {\n position: relative;\n height: fit-content;\n width: fit-content;\n\n background-color: var(--background-color);\n z-index: 4;\n\n /* Everything here is just for centering */\n display: inline flex;\n margin: 0 auto;\n padding: 1rem;\n flex-direction: column;\n text-align: center;\n align-self: center;\n align-content: center;\n justify-content: center;\n word-wrap: break-word;\n word-break: break-all;\n}\n\n.PromptNotice > label {\n font-size: 1.5rem;\n}\n\n.PromptNotice > button {\n font-size: 1.5rem;\n width: 100%;\n}\n\n.PromptNotice > label > p {\n margin-top: 0;\n margin-bottom: 1rem;\n}\n\n.PromptNotice > label > textarea {\n width: 100%;\n}\n\n\n","\n.github-corner {\n color: var(--text-color);\n position: absolute;\n top: 0;\n border: 0;\n right: 0;\n}\n\n.github-corner > svg {\n fill: #151513;\n}\n\n.octo-arm {\n transform-origin: 130px 106px;\n}\n\n@keyframes octocat-wave {\n 0%, 100% {\n transform: rotate(0deg);\n }\n\n 20%, 60% {\n transform: rotate(-25deg);\n }\n\n 40%, 80% {\n transform: rotate(10deg);\n }\n}\n\n@media (prefers-reduced-motion: no-preference) {\n .github-corner:hover .octo-arm {\n animation: octocat-wave 560ms ease-in-out;\n }\n\n @media (max-width: 500px) {\n .github-corner:hover .octo-arm {\n animation: none;\n }\n\n .github-corner .octo-arm {\n animation: octocat-wave 560ms ease-in-out;\n }\n }\n}\n\n@media (prefers-reduced-motion: reduce) {\n .github-corner:hover .octo-arm {\n animation: octocat-wave 7s ease-in-out;\n }\n\n @media (max-width: 500px) {\n .github-corner:hover .octo-arm {\n animation: none;\n }\n\n .github-corner .octo-arm {\n animation: octocat-wave 7s ease-in-out;\n }\n }\n}\n","\n.ErrorNotice {\n font-family: var(--monospace);\n white-space: pre;\n padding: 1.5rem;\n border: 3px solid var(--red);\n height: fit-content;\n width: fit-content;\n}\n"],"names":[],"sourceRoot":""} \ No newline at end of file diff --git a/static/js/main.202c3475.js b/static/js/main.202c3475.js new file mode 100644 index 00000000..10a294b3 --- /dev/null +++ b/static/js/main.202c3475.js @@ -0,0 +1,3 @@ +/*! For license information please see main.202c3475.js.LICENSE.txt */ +(()=>{"use strict";var e={123:e=>{var t=Object.getOwnPropertySymbols,n=Object.prototype.hasOwnProperty,r=Object.prototype.propertyIsEnumerable;function o(e){if(null===e||void 0===e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var t={},n=0;n<10;n++)t["_"+String.fromCharCode(n)]=n;if("0123456789"!==Object.getOwnPropertyNames(t).map((function(e){return t[e]})).join(""))return!1;var r={};return"abcdefghijklmnopqrst".split("").forEach((function(e){r[e]=e})),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},r)).join("")}catch(o){return!1}}()?Object.assign:function(e,l){for(var s,i,a=o(e),u=1;u{var r=n(43),o=n(123),l=n(853);function s(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n
\n )\n }\n}\n","import React from 'react';\nimport ExternalLink from '../ExternalLink';\n\nexport type StrategyLabelProps = Readonly<{\n name: string\n href?: string\n id?: string\n}>\n\n/**\n * The text {labelling} (naming) [for] a strategy; inside such StrategyItem.\n */\nexport default class StrategyLabel extends React.PureComponent {\n render() {\n if (this.props.href) {\n return (\n \n {this.props.name}\n \n )\n }\n\n return {this.props.name}\n }\n}\n","\nimport React from 'react';\n\ntype StrategyTogglerProps = Readonly<{\n callback: React.ChangeEventHandler\n id: string\n}>\n\ntype StrategyTogglerState = Readonly<{\n checked: boolean\n}>\n\n/**\n * Turns a strategy off or on\n */\nexport default class StrategyToggler extends React.Component {\n constructor(props: StrategyTogglerProps) {\n super(props)\n\n this.state = {\n checked: true\n }\n\n this.callback = this.callback.bind(this)\n }\n\n render() {\n // Apparently there shouldn't be an aria-checked\n return (\n \n )\n }\n\n callback(_event: React.ChangeEvent) {\n this.setState(state => ({ checked: !state.checked }))\n this.props.callback(_event)\n }\n}\n","import { SudokuDigits } from \"../Types\";\nimport PureSudoku from \"./Spaces/PureSudoku\";\nimport { CellID } from \"./Utils\";\n\nexport const SuccessError = -1\nexport type StrategyResult = Readonly<{\n success: true,\n successcount: number,\n message?: string\n} | {\n success: false,\n successcount?: number,\n message?: string\n}>\nexport type StrategyError = Readonly<{\n success: false,\n successcount: typeof SuccessError,\n message?: string\n}>\n\nexport type PureStrategy = (arg: PureSudoku) => StrategyResult\ntype StatefulStrategy = (arg: PureSudoku, memory: StrategyMemory[number]) => StrategyResult\n// eslint-disable-next-line @typescript-eslint/ban-types -- `Function` is only used for the `name` property\nexport type Strategy = Function & (PureStrategy | StatefulStrategy)\n\n// Group types\nexport type CellInfo = {\n candidates: SudokuDigits[]\n position: CellID\n}\n\nexport type CellGroup = CellInfo[]\n\n\n// Information strategies might want to know\nexport class StrategyMemory extends Array {\n public [0] = { solved: 0 }\n}\n","import React from 'react';\nimport { SuccessError } from '../../Api/Types';\n\nexport type StrategyStatusProps = Readonly<{\n success: null\n successcount: null | number\n}> | Readonly<{\n success: boolean\n successcount: number\n}>\n\n/**\n * How did trying the strategy go?\n */\nexport default class StrategyStatus extends React.Component {\n render() {\n const resultText =\n this.props.success === null\n ? '-'\n : this.props.success\n ? this.props.successcount === 1\n ? 'Yes'\n : this.props.successcount\n : this.props.successcount === SuccessError\n ? 'Error!'\n : 'No';\n\n const cssClass =\n this.props.success === null\n ? 'null'\n : this.props.success\n ? 'success'\n : 'fail';\n\n return (\n \n {resultText}\n \n )\n }\n}\n","import React from 'react';\n\nexport type StrategyTogglerLabelProps = Readonly<{\n name: string\n id?: string\n}>\n\n/**\n * Same as {@link StrategyLabel}, but this time labelling the {@link StrategyToggler}\n */\nexport default class StrategyTogglerLabel extends React.PureComponent {\n render() {\n return (\n \n {`Toggle ${this.props.name}`}\n \n )\n }\n}\n","import './StrategyItem.css'\nimport React from 'react';\nimport StrategyLabel, { StrategyLabelProps } from './StrategyLabel';\nimport StrategyToggler from './StrategyToggler';\nimport StrategyStatus, { StrategyStatusProps } from './StrategyStatus';\nimport Solver from '../../Api/Solver';\nimport StrategyTogglerLabel from './StrategyTogglerLabel';\n\nexport type StrategyItemProps = StrategyLabelProps & Readonly<{\n solver: Solver\n index: number\n required?: true | 'true'\n}>\n\nexport type StrategyItemState = StrategyStatusProps & Readonly<{\n disabled: boolean\n isCurrentStrategy: boolean\n}>\n\ntype StrategyResult = {\n success: boolean\n successcount: number | null\n}\n\n/**\n * The strategy element\n *\n * Passes props to StrategyLabel\n */\nexport default class StrategyItem extends React.Component {\n id: `strategy-${string}`;\n labelId: `label-for-strategy-${string}`;\n togglerId?: `strategy-toggler-${string}`;\n constructor(props: StrategyItemProps) {\n super(props)\n\n const name = this.props.name.replaceAll(' ', '-')\n this.id = `strategy-${name}` as const\n this.labelId = `label-for-strategy-${name}` as const\n if (this.props.required === undefined) {\n this.togglerId = `strategy-toggler-${name}` as const\n }\n\n this.state = {\n success: null,\n successcount: null,\n\n disabled: false,\n isCurrentStrategy: false\n }\n\n this.reset = this.reset.bind(this)\n this.toggle = this.toggle.bind(this)\n this.whenStepFinished = this.whenStepFinished.bind(this)\n }\n\n componentDidMount() {\n this.props.solver.eventRegistry.addEventListener('step finish', this.whenStepFinished)\n this.props.solver.eventRegistry.addEventListener('new turn', this.reset)\n }\n\n componentWillUnmount() {\n this.props.solver.eventRegistry.removeEventListener('step finish', this.whenStepFinished)\n this.props.solver.eventRegistry.removeEventListener('new turn', this.reset)\n }\n\n whenStepFinished({success, successcount}: StrategyResult, index: number) {\n if (index === this.props.index) {\n this.setState({ success, successcount, isCurrentStrategy: true })\n } else {\n this.setState({ isCurrentStrategy: false })\n }\n }\n\n render() {\n /**\n * a11y considerations:\n *\n * I want the checkbox to be togglable by clicking any part of the text.\n *\n * But the text itself isn't a good label;\n * A better label would be \"toggle strategyName\" instead of \"strategyName\"\n *\n * Also, \"strategyName\" should label the
  • rather than the checkbox\n */\n let thisClass = 'StrategyItem'\n if (this.state.disabled) {\n thisClass += ' disabled'\n }\n if (this.state.isCurrentStrategy) {\n thisClass += ' isCurrent'\n }\n\n const togglerPart = this.props.required ? <> : (\n \n )\n\n return (\n
  • \n \n {togglerPart}\n \n
  • \n )\n\n // StrategyLabel goes before StrategyToggler because\n // it makes sense a11y wise to put the text first\n\n // And also because the site supports both ltr and rtl (hopefully)\n }\n\n toggle(_event: React.ChangeEvent) {\n this.setState(state => ({ disabled: !state.disabled }), () => {\n this.props.solver.disabled[this.props.index] = this.state.disabled\n })\n }\n\n reset() {\n this.setState({ success: null, successcount: null })\n }\n}\n","\nimport './StrategyList.css'\nimport React from 'react';\nimport Solver from '../../Api/Solver';\nimport { GuaranteedConstructCallback } from '../../Types';\nimport StrategyItem from './StrategyItem';\n\ntype StrategyListProps = Readonly<{\n solver: Solver\n}> & GuaranteedConstructCallback\n\n/**\n * A list of strategies\n */\nexport default class StrategyList extends React.Component {\n constructor(props: StrategyListProps) {\n super(props)\n this.props.whenConstruct()\n }\n\n render() {\n let index = 0\n\n const getRepeatedProps = () => {\n return {\n solver: this.props.solver,\n index: index++\n }\n }\n\n return (\n
      \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
    \n )\n }\n}\n","\nimport './StrategyDetails.css'\nimport React from \"react\"\nimport Solver from \"../../Api/Solver\"\nimport { StrategyResult } from \"../../Api/Types\"\n\ntype StrategyDetailsProps = Readonly<{\n solver: Solver\n}>\n\ntype StrategyDetailsState = Readonly<{\n text: string\n}>\n\nexport default class StrategyDetails extends React.Component {\n constructor(props: StrategyDetailsProps) {\n super(props)\n\n this.state = {\n text: \"\"\n }\n\n this.reset = this.reset.bind(this)\n this.whenStepFinished = this.whenStepFinished.bind(this)\n }\n\n componentDidMount() {\n this.props.solver.eventRegistry.addEventListener('step finish', this.whenStepFinished)\n this.props.solver.eventRegistry.addEventListener('new turn', this.reset)\n }\n\n componentWillUnmount() {\n this.props.solver.eventRegistry.removeEventListener('step finish', this.whenStepFinished)\n this.props.solver.eventRegistry.removeEventListener('new turn', this.reset)\n }\n\n whenStepFinished({message}: StrategyResult) {\n this.setState({ text: message ?? \"\" })\n }\n\n reset() {\n this.setState({ text: \"\" })\n }\n\n render() {\n return (\n
    \n Strategy explanation\n

    {this.state.text}

    \n
    \n )\n }\n}\n","\nimport React from 'react'\nimport Solver from '../../Api/Solver'\nimport Sudoku from '../../Api/Spaces/Sudoku'\nimport { StrategyResult } from '../../Api/Types'\nimport StrategyControls from './StrategyControls'\nimport StrategyList from './StrategyList'\nimport StrategyDetails from './StrategyDetails'\n\ntype SolverPartProps = Readonly<{\n sudoku: Sudoku\n solver: Solver\n}>\n\n/**\n * The solver part of the sudoku solver\n */\nexport default class SolverPart extends React.Component {\n children: {\n controls: null | StrategyControls\n list: null | StrategyList\n }\n strategyItemStates: StrategyResult[]\n constructor(props: SolverPartProps) {\n super(props)\n\n this.children = {\n controls: null,\n list: null\n }\n this.strategyItemStates = []\n\n this.whenControlsConstruct = this.whenControlsConstruct.bind(this)\n this.whenListConstructs = this.whenListConstructs.bind(this)\n }\n\n render() {\n return (\n
    \n \n
    \n strategies\n \n
    \n \n
    \n )\n }\n\n whenControlsConstruct(controls: StrategyControls) {\n this.children.controls = controls\n }\n\n whenListConstructs(list: StrategyList) {\n this.children.list = list\n }\n\n /** Called when a strategy starts - see the Solver api */\n notify(strategyIndex: number, strategyResult: StrategyResult) {\n this.strategyItemStates[strategyIndex] = strategyResult\n }\n}\n","import React from \"react\";\nimport { _Callback } from \"../../Types\";\nimport Control from \"../Control\";\n\ntype TabProps = Readonly<{\n index: number\n focused: boolean\n selected: boolean\n title: string\n whenFocused: _Callback\n whenSelected: (index: number) => void\n}>\n\nexport default class Tab extends React.Component {\n selfElement: HTMLButtonElement | null = null\n setSelfElement: (element: HTMLButtonElement | null) => HTMLButtonElement | null;\n constructor (props: TabProps) {\n super(props)\n this.setSelfElement = (element: HTMLButtonElement | null) => this.selfElement = element\n this.callbackIfNotSelected = this.callbackIfNotSelected.bind(this)\n }\n\n componentDidUpdate() {\n if (this.props.focused) {\n this.selfElement?.focus()\n }\n }\n\n render () {\n const className = this.props.selected ? `Tab selected` : `Tab unselected`\n return (\n {this.props.title}\n )\n }\n\n callbackIfNotSelected () {\n if (!this.props.selected) {\n this.props.whenSelected(this.props.index)\n }\n }\n}\n","import React from \"react\";\nimport \"./Tabs.css\"\n\nimport { _Callback } from \"../../Types\";\nimport Tab from \"./Tab\";\n\ntype TabsProps = Readonly<{\n tabNames: string[]\n whenTabChange: _Callback\n}>\n\ntype TabsState = Readonly<{\n focusedTab: number | null\n selectedTab: number\n}>\n\ntype TabsElement = HTMLDivElement | null\n\nconst importantKeys = new Set([\"Tab\", \"ArrowLeft\", \"ArrowRight\", \"Home\", \"End\"] as const)\nconst oppositeKeys = {\n ArrowLeft: \"ArrowRight\",\n ArrowRight: \"ArrowLeft\",\n Home: \"End\",\n End: \"Home\",\n} as const\n\nexport default class Tabs extends React.Component {\n tabsElement: TabsElement = null;\n keysPressed: typeof importantKeys = new Set();\n setTabsElement: (element: TabsElement) => TabsElement;\n constructor (props: TabsProps) {\n super(props)\n this.state = {\n focusedTab: 0,\n selectedTab: 0,\n }\n\n this.setTabsElement = (element: TabsElement) => this.tabsElement = element\n\n this.whenBlur = this.whenBlur.bind(this)\n this.whenFocus = this.whenFocus.bind(this)\n this.whenKeyUp = this.whenKeyUp.bind(this)\n this.whenKeyDown = this.whenKeyDown.bind(this)\n this.whenTabChange = this.whenTabChange.bind(this)\n this.whenTabFocused = this.whenTabFocused.bind(this)\n }\n\n componentDidMount () {\n // @ts-expect-error Why does React have special dom types???\n document.addEventListener('keyup', this.whenKeyUp)\n }\n\n componentWillUnmount () {\n // @ts-expect-error Why does React have special dom types???\n document.removeEventListener('keyup', this.whenKeyUp)\n }\n\n render () {\n const tabs = []\n for (const [index, title] of this.props.tabNames.entries()) {\n tabs.push(\n \n )\n }\n\n /**\n * https://www.w3.org/TR/wai-aria-practices-1.1/examples/tabs/tabs-1/tabs.html\n *\n * Using div here because it's recommended\n * tabindex for focusability but not tabbability (-1)\n */\n return (\n {tabs}\n )\n }\n\n whenTabChange (index: number) {\n this.setState({selectedTab: index})\n this.props.whenTabChange(index)\n }\n\n /** Keyboard support */\n\n whenKeyDown (event: React.KeyboardEvent) {\n if (importantKeys.has(event.key)) { // @ts-expect-error Can't narrow\n this.keysPressed.add(event.key)\n }\n\n const movements = new Set()\n let tab = false\n for (const key of this.keysPressed) {\n if (key === 'Tab') {\n tab = true\n } else if (movements.has(oppositeKeys[key])) {\n movements.delete(oppositeKeys[key])\n } else {\n movements.add(key)\n }\n }\n\n if (movements.size !== 0) {\n for (const key of ['Home', 'End', 'ArrowLeft', 'ArrowRight'] as const) {\n if (movements.has(key)) {\n this.handleMovement(key, tab)\n break\n }\n }\n } else if (tab) {\n this.changeToMainContent()\n }\n }\n\n private handleMovement (movement: keyof typeof oppositeKeys, tab: boolean) {\n let tabChanged = false\n this.setState((state, props) => {\n let newTab: number\n\n if (movement === 'End')\n newTab = props.tabNames.length - 1\n else if (movement === 'Home' || state.focusedTab === null)\n newTab = 0\n else if (movement === 'ArrowLeft')\n newTab = state.focusedTab === 0\n ? props.tabNames.length - 1\n : state.focusedTab - 1\n else if (movement === 'ArrowRight')\n newTab = state.focusedTab === props.tabNames.length - 1\n ? 0\n : state.focusedTab + 1\n else\n throw TypeError(`Impossible movement: ${movement}`)\n\n if (state.selectedTab === newTab) {\n tabChanged = true\n return { focusedTab: newTab, selectedTab: newTab }\n }\n\n return null\n }, () => {\n if (tabChanged) {\n this.props.whenTabChange(this.state.selectedTab)\n }\n if (tab) {\n this.changeToMainContent()\n }\n })\n }\n\n whenKeyUp (event: React.KeyboardEvent) {\n this.keysPressed.delete(event.key)\n }\n\n whenTabFocused () {\n // align focused and selected tab\n this.setState(prevState => ({ focusedTab: prevState.selectedTab }))\n }\n\n // When entering a group of tabs, focus the currently active (\"selected\") tab\n // Note that a mouseup happens after a focus event, i.e. this always wins a race to be overridden.\n whenFocus (_event: React.FocusEvent) {\n this.whenTabFocused()\n }\n\n whenBlur() {\n this.setState({ focusedTab: null })\n }\n\n changeToMainContent () {\n document.getElementById('TabContent')?.focus()\n }\n}\n","/**\n * IMPORTANT: This is represented by a section element\n * since it's actually directly related to the content\n *\n * But it's still... _aside_ ```
    ```\n */\n\nimport './Aside.css'\nimport React from 'react'\nimport SolverPart from './AsideElems/SolverPart'\nimport SudokuData from '../Api/Spaces/Sudoku'\nimport Tabs from './AsideElems/Tabs'\nimport Control from './Control'\nimport Solver from '../Api/Solver'\n\ntype AsideProps = Readonly<{\n sudoku: SudokuData,\n solver: Solver,\n}>\n\ntype AsideState = Readonly<{\n selectedTab: number\n}>\n\n\nconst tabNames = [\"solving tools\", \"strategies\"]\n\n/**\n * Currently a window of tabs\n */\nexport default class Aside extends React.Component {\n constructor (props: AsideProps) {\n super(props)\n\n this.state = {\n selectedTab: 0\n }\n\n this.whenTabChange = this.whenTabChange.bind(this)\n }\n\n render() {\n let content: JSX.Element\n if (this.state.selectedTab === 0) {\n content = <>\n \n \n \n \n } else if (this.state.selectedTab === 1) {\n content =\n \n } else {\n throw new ReferenceError(`unknown Selected tab index: ${this.state.selectedTab}`)\n }\n\n /**\n * Tabpanel id used in Tabs (aria-owns)\n * div because that's what's recommended\n * tabindex for focusability but not tabbability (-1)\n */\n return (\n
    \n \n
    {content}
    \n
    \n );\n }\n\n whenTabChange(index: number) {\n this.setState({ selectedTab: index })\n }\n}\n","import { ALL_CANDIDATES, GrpTyp, IndexToNine, INDICES_TO_NINE, SudokuDigits, ThreeDimensionalArray } from \"../../Types\"\nimport { boxAt, CellID, id, to9by9 } from \"../Utils\"\n\nexport type CandidateLocations = Record[]>\n\nfunction Cell (id: CellID, cell: SudokuDigits[]) {\n return {\n candidates: cell,\n position: id\n }\n}\n\n/**\n * Defines base sudoku methods\n * Should I move these to utils?\n */\nexport default class PureSudoku {\n data: ThreeDimensionalArray\n constructor(representation?: string) {\n this.data = [\n [], [], [],\n [], [], [],\n [], [], [],\n ]\n\n for (let i = 0; i < 9; i++) {\n for (let j = 0; j < 9; j++) {\n this.data[i][j] = [1, 2, 3, 4, 5, 6, 7, 8, 9]\n }\n }\n\n if (typeof representation === \"string\") {\n this.import(representation)\n }\n }\n\n static fromRepresentation(this: T, representation: string): InstanceType {\n return new this(representation) as InstanceType\n }\n\n /**\n * Convert the sudoku into 81 digits\n * \"0\" = a cell with no candidates\n * \".\" = an unsolved cell\n */\n to81 () {\n let str = \"\"\n for (const row of this.data) {\n for (const cell of row) {\n if (cell.length === 1) {\n str += cell[0]\n } else if (cell.length === 0) {\n str += \"0\"\n } else {\n str += \".\"\n }\n }\n str += \"\\n\"\n }\n return str\n }\n\n /**\n * Same as to81 but every cell is represented by\n * its candidates. No newlines.\n * Missing candidates are represented by \".\"\n */\n to729 () {\n let str = \"\"\n for (const row of this.data) {\n for (const cell of row) {\n for (const candidate of ALL_CANDIDATES) {\n str += cell.includes(candidate) ? candidate : \".\"\n }\n }\n }\n return str\n }\n\n /**\n * Imports from an 81 character string representing a sudoku.\n *\n * Any character that is not a digit is a blank cell.\n */\n import81(representation: string): void {\n let totalIndex = 0\n for (const i of INDICES_TO_NINE) {\n for (const j of INDICES_TO_NINE) {\n const char = representation[totalIndex]\n totalIndex++ // after char\n\n // Using `this.set` for compatibility with `Sudoku`\n if (\"123456789\".includes(char)) {\n this.set(i, j).to(Number(char) as SudokuDigits)\n } else {\n this.set(i, j).to(1, 2, 3, 4, 5, 6, 7, 8, 9)\n }\n }\n }\n }\n\n /**\n * Imports from a 729 character string, where every 9 candidates\n * specify candidates for a cell\n */\n import729(representation: string): void {\n let totalIndex = 0\n for (const i of INDICES_TO_NINE) {\n for (const j of INDICES_TO_NINE) {\n const candidateData = (\n representation.slice(totalIndex * 9, totalIndex * 9 + 9)\n .split('')\n .filter(candidate => \"123456789\".includes(candidate))\n .map(candidate => Number(candidate) as SudokuDigits)\n )\n\n totalIndex++\n this.set(i, j).to(...candidateData)\n }\n }\n }\n\n importGrid(gridRepresentation: ThreeDimensionalArray) {\n for (const i of INDICES_TO_NINE) {\n for (const j of INDICES_TO_NINE) {\n this.set(i, j).to(...gridRepresentation[i][j])\n }\n }\n }\n\n // Note: import is a keyword, but it doesn't cause a syntax error here\n import(representation: string) {\n representation = representation.trim().normalize()\n const representationWithoutWhitespace = representation.replaceAll(/\\s/g, \"\")\n const onlyDigitRepresentation = representation.replaceAll(/\\D/g, \"\")\n const oneToNineOrDot = representation.replaceAll(/[^.1-9]/g, \"\")\n\n for (const testRepresentation of [representation, representationWithoutWhitespace, onlyDigitRepresentation, oneToNineOrDot] as const) {\n if (testRepresentation.length === 81) {\n this.import81(testRepresentation)\n return {\n success: true,\n representationType: '81'\n } as const\n }\n\n if (testRepresentation.length === 729) {\n this.import729(testRepresentation)\n return {\n success: true,\n representationType: '729'\n } as const\n }\n }\n\n\n const gridRepresentation = representation\n .split('')\n .filter(char => \"123456789 \".includes(char))\n .join('')\n .trim()\n .split(/\\s+/) // split ignores g flag\n\n if (gridRepresentation.length === 81) {\n this.importGrid(\n to9by9(\n gridRepresentation.map(\n a => a.split('').map(b => Number(b) as SudokuDigits)\n )\n )\n )\n\n return {\n success: true,\n representationType: 'grid'\n } as const\n }\n\n return {\n success: false\n } as const\n }\n\n // WARNING: The \"candidates\" aren't checked (other than the static type checking)\n set(x: IndexToNine, y: IndexToNine) {\n return {\n to: (...candidates: SudokuDigits[]) => {\n this.data[x][y] = candidates\n }\n }\n }\n\n clearCell(x: IndexToNine, y: IndexToNine) {\n this.set(x, y).to(1, 2, 3, 4, 5, 6, 7, 8, 9)\n }\n\n clear() {\n for (const i of INDICES_TO_NINE) {\n for (const j of INDICES_TO_NINE) {\n this.clearCell(i, j)\n }\n }\n }\n\n getColumn(index: IndexToNine) {\n return this.data.map(row => row[index])\n }\n\n getBox(index: IndexToNine) {\n // 0 1 2 -> 0 3 6\n const startRow = index - (index % 3) // / 3 * 3\n const startColumn = (index % 3) * 3\n return this.data.slice(startRow, startRow + 3).flatMap(row => row.slice(startColumn, startColumn + 3))\n }\n\n getBoxGroup(index: IndexToNine, data: T[][]) {\n const startRow = index - (index % 3)\n const startColumn = (index % 3) * 3\n return data.slice(startRow, startRow + 3).flatMap(row => row.slice(startColumn, startColumn + 3))\n }\n\n /**\n * A group is a set of maximally mutually exclusive set of cells.\n * \n * Currently, we assume this is just the rows, columns, and boxes, but this will be changed later on.\n */\n getGroups() {\n const groups = []\n const cellData = this.data.map((row, indexOfRow) =>\n row.map((cell, indexInRow) => Cell(id(indexOfRow as IndexToNine, indexInRow as IndexToNine), cell))\n )\n\n for (const i of INDICES_TO_NINE) {\n groups.push(\n cellData[i],\n cellData.map(row => row[i]),\n this.getBoxGroup(i, cellData)\n )\n }\n\n return groups\n }\n\n /**\n * Returns the candidate locations of the sudoku\n *\n * @misnomer\n * getCandidatesLocations\n *\n * @example\n *\n * ```ts\n * sudoku.getCandidatesLocations()[candidate].rows[5]\n * // > Set\n * // set of cells with that candidate\n * ```\n *\n * Return type is an array of\n * ```\n * CandidateLocations: {\n * rows: [Set for each index]\n * columns: [Set for each index]\n * boxes: [Set for each index]\n * }\n * ```\n */\n getCandidateLocations(): CandidateLocations[] {\n const candidateLocations = [] as CandidateLocations[]\n\n for (const candidate of ALL_CANDIDATES) {\n candidateLocations[candidate] = {\n row: [new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set()],\n column: [new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set()],\n box: [new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set(), new Set()]\n }\n }\n\n for (const row of INDICES_TO_NINE) {\n for (const column of INDICES_TO_NINE) {\n const cellID = id(row, column)\n for (const candidate of this.data[row][column]) {\n candidateLocations[candidate].row[row].add(cellID)\n candidateLocations[candidate].column[column].add(cellID)\n candidateLocations[candidate].box[boxAt(row, column)].add(cellID)\n }\n }\n }\n\n return candidateLocations\n }\n\n /**\n * Removes a candidate at a cell\n *\n * @example\n * (new PureSudoku()).remove(7).at(3, 5)\n */\n remove(candidate: SudokuDigits) {\n // Using an arrow function here to use `this`\n return {\n at: (row: IndexToNine, column: IndexToNine) => {\n this.set(row, column).to(\n ...this.data[row][column].filter(candidatee => candidatee !== candidate)\n )\n }\n }\n }\n}\n\n","import Cell from \"../../Elems/MainElems/Cell\"\nimport { IndexToNine, SudokuDigits, TwoDimensionalArray } from \"../../Types\"\nimport PureSudoku from \"./PureSudoku\"\n\n/**\n * Wraps PureSudoku, to sync the data with the Cell components.\n * \n * It is also used by the Sudoku component to indirectly access the Cell components. -.-\n * \n * For sanity, this class should be kept very simple.\n */\nexport default class Sudoku extends PureSudoku {\n cells: TwoDimensionalArray\n constructor (...args: ConstructorParameters) {\n super(...args)\n if (this.data === undefined) {\n throw TypeError('Super constructor PureSudoku didnt initialize this.data')\n }\n\n this.cells = [\n [], [], [],\n [], [], [],\n [], [], [],\n ]\n }\n\n override set(x: IndexToNine, y: IndexToNine) {\n return {\n to: async (...candidates: SudokuDigits[]) => {\n this.data[x][y] = candidates\n\n // super calls set() before defining this.cells\n if (this.cells !== undefined) {\n await new Promise(resolve => {\n this.cells[x][y]?.setCandidatesTo(candidates, () => resolve(undefined))\n })\n }\n }\n }\n }\n\n override clearCell(x: IndexToNine, y: IndexToNine) {\n this.data[x][y] = [1, 2, 3, 4, 5, 6, 7, 8, 9]\n this.cells[x][y]?.clearCandidates()\n }\n\n /**\n * Used for Cell initialization\n *\n * An alternate possibility is having the cell reflect the data, but\n * that allows inconsistencies between the visuals and the data.\n */\n addCell(cell: Cell) {\n this.cells[cell.props.row][cell.props.column] = cell\n this.data[cell.props.row][cell.props.column] = cell.state.candidates\n }\n\n removeCell(cell: Cell) {\n this.cells[cell.props.row][cell.props.column] = undefined\n // this.data[cell.props.row][cell.props.column] = cell.state.candidates\n }\n}\n","/**\n * Same as Utils, but ones dependent on other files.\n *\n * Might want to run `madge --circular --extensions ts ./src` after importing from here\n */\n\nimport { SudokuDigits, INDICES_TO_NINE, IndexToNine } from \"../Types\";\nimport PureSudoku from \"./Spaces/PureSudoku\";\nimport Sudoku from \"./Spaces/Sudoku\";\nimport { CandidateID, CellID, id } from \"./Utils\";\n\n/**\n * Highlights a cell, see {@link Cell#highlight}\n * Default background is blue.\n */\nexport function highlightCell (sudoku: PureSudoku, {row, column}: CellID, color = 'blue') {\n if (sudoku instanceof Sudoku) {\n sudoku.cells[row][column]?.addClass(color)\n }\n}\n\n/**\n * Highlights multiple cells, see {@link Cell#highlight}\n * Default background is blue.\n */\nexport function highlightGroup (sudoku: PureSudoku, group: Iterable, color = 'blue') {\n if (sudoku instanceof Sudoku) {\n for (const {row, column} of group) {\n sudoku.cells[row][column]?.addClass(color)\n }\n }\n}\n\n/**\n * Colors a group of cells' candidates, see {@link Cell#highlight}\n *\n * The precedence for colors is:\n * 1. orange\n * 2. green\n * 3. blue\n */\nexport function colorGroup (sudoku: PureSudoku, group: Iterable, candidate: SudokuDigits, color = 'blue') {\n if (sudoku instanceof Sudoku) {\n for (const cell of group) {\n const element = sudoku.cells[cell.row][cell.column];\n element?.highlight([candidate], color);\n }\n }\n}\n\n/**\n * Same as {@link colorGroup}, but this time with a specific candidate\n */\nexport function colorCandidateF (sudoku: PureSudoku, row: IndexToNine, column: IndexToNine, digit: SudokuDigits, color = 'blue') {\n if (sudoku instanceof Sudoku) {\n sudoku.cells[row][column]?.highlight([digit], color)\n }\n}\n\n/**\n * Same as {@link colorGroup}, but this time with a specific candidate\n */\nexport function colorCandidate (sudoku: PureSudoku, { row, column, digit }: CandidateID, color = 'blue') {\n colorCandidateF(sudoku, row, column, digit, color)\n}\n\nexport function numberOfCellsWithNCandidates (sudoku: PureSudoku, N: number) {\n let cellsWithNCandidates = 0\n for (const row of INDICES_TO_NINE) {\n for (const column of INDICES_TO_NINE) {\n if (sudoku.data[row][column].length === N) {\n cellsWithNCandidates++\n }\n }\n }\n\n return cellsWithNCandidates\n}\n\nexport function getCellsWithNCandidates (sudoku: PureSudoku, N: number) {\n const cellsWithNCandidates = [] as CellID[]\n for (const row of INDICES_TO_NINE) {\n for (const column of INDICES_TO_NINE) {\n if (sudoku.data[row][column].length === N) {\n cellsWithNCandidates.push(id(row, column))\n }\n }\n }\n\n return cellsWithNCandidates\n}\n\n/**\n * Removes a {@param candidate} from multiple {@param cells}\n * if that candidate exists.\n *\n * Returns true if any candidates were eliminated\n */\nexport function removeCandidateFromCells(sudoku: PureSudoku, candidate: SudokuDigits, cells: Iterable) {\n // Should this return early if there are no cells?\n let success = false\n for (const {row, column} of cells) {\n if (sudoku.data[row][column].includes(candidate)) {\n sudoku.remove(candidate).at(row, column)\n success = true\n }\n }\n return success\n}\n\nexport function wouldRemoveCandidateFromCells(sudoku: PureSudoku, candidate: SudokuDigits, cells: Iterable) {\n for (const { row, column } of cells) {\n if (sudoku.data[row][column].includes(candidate)) {\n return true\n }\n }\n return false\n}\n","import { SudokuDigits, ROW_NAMES, COLUMN_NAMES, ALL_CANDIDATES, BOX_NAMES, INDICES_TO_NINE } from \"../../Types\"\nimport { convertArrayToEnglishList } from \"../../utils\"\nimport PureSudoku from \"../Spaces/PureSudoku\"\nimport { boxAt, algebraic, boxNameAt } from \"../Utils\"\n\ntype validityResult = {\n ok: true\n} | {\n ok: false,\n message: string\n}\n\nfunction listOfMissingCandidatesIn (group: Set) {\n return convertArrayToEnglishList(\n ALL_CANDIDATES.filter(\n candidate => !group.has(candidate)\n )\n )\n}\n\nexport default function checkValidity(sudoku: PureSudoku): validityResult {\n const solvedInColumns = [] as Array>\n const solvedInBoxes = [] as Array>\n const candidatesInColumns = [] as Array>\n const candidatesInBoxes = [] as Array>\n\n for (let i = 0; i < 9; i++) {\n solvedInColumns.push(new Set())\n solvedInBoxes.push(new Set())\n candidatesInColumns.push(new Set())\n candidatesInBoxes.push(new Set())\n }\n\n for (const i of INDICES_TO_NINE) {\n const solvedInRow = new Set()\n const candidatesInRow = new Set()\n\n for (const j of INDICES_TO_NINE) {\n const candidates = sudoku.data[i][j]\n const current = {\n column: j,\n box: boxAt(i, j)\n } as const\n\n // No possibilities\n if (candidates.length === 0) {\n return {\n ok: false,\n message: `Cell ${algebraic(i, j)} has 0 possible candidates!`\n }\n }\n\n if (candidates.length === 1) {\n const solvedCandidate = candidates[0]\n\n // Same in row\n if (solvedInRow.has(solvedCandidate)) {\n return {\n ok: false,\n message: `Two (or more) ${solvedCandidate}s in row ${ROW_NAMES[i]}!`\n }\n }\n\n // Same in column\n if (solvedInColumns[current.column].has(solvedCandidate)) {\n return {\n ok: false,\n message: `Two (or more) ${solvedCandidate}s in column ${COLUMN_NAMES[j]}!`\n }\n }\n\n // Same in box\n if (solvedInBoxes[current.box].has(solvedCandidate)) {\n return {\n ok: false,\n message: `Two (or more) ${solvedCandidate}s in box ${boxNameAt(i, j)}!`\n }\n }\n\n solvedInRow.add(solvedCandidate)\n solvedInColumns[current.column].add(solvedCandidate)\n solvedInBoxes[current.box].add(solvedCandidate)\n }\n\n for (const candidate of candidates) {\n candidatesInRow.add(candidate)\n candidatesInColumns[current.column].add(candidate)\n candidatesInBoxes[current.box].add(candidate)\n }\n }\n\n if (candidatesInRow.size !== 9) {\n return {\n ok: false,\n message: `Row ${ROW_NAMES[i]} has 0 possibilities for ${listOfMissingCandidatesIn(candidatesInRow)}!!!`\n }\n }\n }\n\n for (let i = 0; i < 9; i++) {\n if (candidatesInColumns[i].size !== 9) {\n return {\n ok: false,\n message: `Column ${COLUMN_NAMES[i]} has 0 possibilities for ${listOfMissingCandidatesIn(candidatesInColumns[i])}!!!`\n }\n }\n\n if (candidatesInBoxes[i].size !== 9) {\n return {\n ok: false,\n message: `Box ${BOX_NAMES[i]} has 0 possibilities for ${listOfMissingCandidatesIn(candidatesInBoxes[i])}!!!`\n }\n }\n }\n\n return {\n ok: true\n }\n}\n","\n\nimport { AlertType, SudokuDigits } from \"../../Types\";\nimport { convertArrayToEnglishList } from \"../../utils\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport Sudoku from \"../Spaces/Sudoku\";\nimport { CellGroup, SuccessError } from \"../Types\";\nimport { algebraic } from \"../Utils\";\n\n/**\n * Gets the unique combinations of an array\\\n * All elements are unmodified and assumed different\n *\n * \"combinations\" is in the mathematical sense:\n * if you give 7 elements, with min = 2 and max = 4,\n * you get (7 choose 2) + (7 choose 3) + (7 choose 4) elements.\n *\n * @example\n * combinations([1, 2, 3])\n * // [[3], [3, 2], [3, 2, 1], [3, 1], [2], [2, 1], [1]]\n *\n * @param {number} min - The minimum size of a combination\n * @param {number} max - The maximum size of a combination\n */\nexport function combinations(array: T[], min = 1, max = array.length, currentCount = 1) {\n const _combinations: T[][] = []\n const _arrayCopy = array.slice()\n while (_arrayCopy.length) {\n const element = _arrayCopy.pop() as T\n\n // For combinations shorter than max size, but also includes max size\n if (currentCount >= min) {\n _combinations.push([element])\n }\n\n // After the check\n if (currentCount === max) {\n continue\n }\n\n for (const combination of combinations(_arrayCopy, min, max, currentCount + 1)) {\n _combinations.push([element, ...combination])\n }\n }\n\n return _combinations\n}\n\n/**\n * Return a set of unique candidates in a conjugate\n */\nfunction getCandidatesOfConjugate(conjugate: CellGroup) {\n // Array from the values of a set\n // The set is the accumulated candidates\n return conjugate.reduce(\n (accum, currentCell) => {\n for (const candidate of currentCell.candidates) {\n accum.add(candidate)\n }\n return accum\n }, new Set()\n )\n}\n\n// Inner inner function to make things look nicer below\nfunction __errorHandling (conjugate: CellGroup, invalidGroupCandidates: Set) {\n const invalidGroupNames = convertArrayToEnglishList(\n conjugate.map(someCell => algebraic(someCell.position.row, someCell.position.column))\n )\n const invalidCandidateString = convertArrayToEnglishList(Array.from(invalidGroupCandidates).sort())\n\n if (conjugate.length === 1) {\n // Never happens since cells are filtered away\n window._custom.alert(`The cell ${invalidGroupNames} has 0 possibilities!`, AlertType.ERROR)\n } else if (invalidGroupCandidates.size === 1) {\n // Never happens\n window._custom.alert(`${invalidGroupNames}: ${conjugate.length} cells cannot share 1 candidate (${invalidCandidateString})!!!`, AlertType.ERROR)\n } else {\n window._custom.alert(`${invalidGroupNames}: ${conjugate.length} cells cannot share ${invalidGroupCandidates.size} candidates (${invalidCandidateString})!!!`, AlertType.ERROR)\n }\n}\n\n/**\n * Here, a \"group\" is a row, column, or box, but can be any group.\n *\n * Within that group, we're trying to find subgroups, aka conjugates\n * where such subgroup has n cells and n candidates.\n *\n * In Andrew Stuart's solver, this is equivalent to finding\n * naked pairs, triples, and quads.\n *\n * @param group - A group of cells. Generally a row, column, or box\n * @param maxSize - The maximum size of the conjugate. Default is 4.\n * (Not looking for conjugates of size 5 or more, since then there would be a\n * size 4 with the other cells by default. TODO better explanation)\n */\nfunction findConjugatesOfGroup(group: CellGroup, maxSize = 4 as 2 | 3 | 4) {\n // 1. Filter the possible cells\n // Each possible cell must have from 2 to maxSize candidates\n const possibleCells = group.filter(cell => 1 < cell.candidates.length && cell.candidates.length <= maxSize)\n\n // 2. Now that the cells are filtered actually find the conjugates\n const conjugates = [] as CellGroup[]\n for (const conjugate of combinations(possibleCells, 2, maxSize)) {\n const candidatesOfConjugate = getCandidatesOfConjugate(conjugate)\n\n // For example 3 cells needing 2 candidates = invalid.\n if (conjugate.length > candidatesOfConjugate.size) {\n __errorHandling(conjugate, candidatesOfConjugate)\n return \"ERROR!!!\" as const\n } else if (conjugate.length === candidatesOfConjugate.size) {\n // Found a conjugate!!!!!\n conjugates.push(conjugate)\n }\n }\n\n return conjugates\n}\n\n\n// Idea for hidden:\n// For each candidate find squares\n\n/**\n * Colors a conjugate, see Cell#highlight\n */\nexport function colorConjugate(sudoku: PureSudoku, conjugate: CellGroup, color = 'blue') {\n if (sudoku instanceof Sudoku) {\n for (const cell of conjugate) {\n const element = sudoku.cells[cell.position.row][cell.position.column]\n element?.highlight(cell.candidates, color)\n }\n }\n}\n\nfunction eliminateUsingConjugate(\n sudoku: PureSudoku,\n group: CellGroup,\n conjugate: CellGroup,\n) {\n let successcount = 0\n const conjugateCandidates = getCandidatesOfConjugate(conjugate)\n const eliminatedFrom = []\n\n for (const cell of group) {\n // If this cell is not in the conjugate\n if (!conjugate.some(jCell => jCell.position === cell.position)) {\n\n // The cell now cannot have any of the candidates in the conjugate!!!\n const nonConjugateCandidates = cell.candidates.filter(candidate => !conjugateCandidates.has(candidate))\n if (cell.candidates.length !== nonConjugateCandidates.length) { // If has any...\n successcount++ // Success!\n colorConjugate(sudoku, conjugate)\n sudoku.set(cell.position.row, cell.position.column).to(...nonConjugateCandidates)\n eliminatedFrom.push(algebraic(cell.position.row, cell.position.column))\n }\n }\n }\n\n return [\n successcount,\n `${[...conjugateCandidates].join(\"\")} ${conjugate.map(cell => algebraic(cell.position.row, cell.position.column))} ⇒ ${eliminatedFrom}`,\n ] as const\n}\n\nfunction eliminateUsingConjugates(sudoku: PureSudoku, groups: CellGroup[], conjugatesOfGroup: CellGroup[][]) {\n let successcount = 0\n const messages = []\n for (const [i, group] of groups.entries()) {\n for (const conjugate of conjugatesOfGroup[i]) {\n const [successes, message] = eliminateUsingConjugate(sudoku, group, conjugate)\n successcount += successes\n if (successes) {\n messages.push(message)\n }\n }\n }\n\n return [successcount, messages.join(\"\\n\")] as const\n}\n\n// Math.max(O(n^5), O(n^5))\nexport default function pairsTriplesAndQuads(sudoku: PureSudoku) {\n const groups = sudoku.getGroups()\n const conjugatesOfGroup = []\n for (const group of groups) {\n const conjugate = findConjugatesOfGroup(group)\n if (conjugate === \"ERROR!!!\") {\n return {\n success: false,\n successcount: SuccessError\n } as const\n }\n conjugatesOfGroup.push(conjugate)\n }\n\n const [successcount, message] = eliminateUsingConjugates(sudoku, groups, conjugatesOfGroup)\n\n if (successcount === 0) {\n return {\n success: false\n } as const\n }\n\n return {\n success: true,\n successcount,\n message,\n } as const\n}\n","import { AlertType, BOX_NAMES, COLUMN_NAMES, INDICES_TO_NINE, ROW_NAMES, SudokuDigits } from \"../../Types\";\nimport { convertArrayToEnglishList } from \"../../utils\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { CellGroup, CellInfo, SuccessError } from \"../Types\";\nimport { algebraic, removeFromArray } from \"../Utils\";\nimport { colorConjugate, combinations } from \"./pairsTriplesAndQuads\";\n\n/**\n * Returns an array of all the cells which contain at least one of the candidates\n */\nfunction getConjugateFromCandidates (cells: CellGroup, candidates: SudokuDigits[]) {\n return cells.filter(cell =>\n candidates.some(candidate => cell.candidates.includes(candidate))\n )\n}\n\nfunction __errorHandling (candidatesOfConjugate: SudokuDigits[], conjugate: CellInfo[]) {\n const invalidCandidateString = convertArrayToEnglishList(candidatesOfConjugate)\n\n // To prevent errors in convertArrayToEnglishList\n if (conjugate.length === 0) {\n // A previous elimination must've caused this!\n return ` has 0 possibilities for ${invalidCandidateString}!!!\\n`\n }\n const invalidGroupNames = convertArrayToEnglishList(\n conjugate.map(someCell => algebraic(someCell.position.row, someCell.position.column))\n )\n\n if (candidatesOfConjugate.length === 1) {\n return ` has 0 possibilities for ${invalidCandidateString}!!!`\n } else if (conjugate.length === 1) {\n return `: ${candidatesOfConjugate.length} candidates (${invalidCandidateString}) all want to be in ${invalidGroupNames} which is impossible!!!`\n } else {\n return `: ${candidatesOfConjugate.length} candidates (${invalidCandidateString}) all want to be in ${conjugate.length} cells (${invalidGroupNames}) which is impossible!!!`\n }\n}\n\nfunction __filterPossibleCandidates (group: CellGroup, maxSize: number, possibleCandidates: Set) {\n\n function removeCandidate (candidate: SudokuDigits) {\n possibleCandidates.delete(candidate)\n\n for (const cell of group) {\n removeFromArray(candidate, cell.candidates)\n }\n }\n\n // a. Remove candidates that occur > maxSize times\n const occurances = [-1, 0, 0, 0, 0, 0, 0, 0, 0, 0] as const as Record\n\n for (const cell of group) {\n for (const candidate of cell.candidates) {\n occurances[candidate]++\n }\n }\n\n for (const candidate of possibleCandidates) {\n if (occurances[candidate] > maxSize) {\n removeCandidate(candidate)\n } else if (occurances[candidate] === 0) {\n return `There is nowhere to put ${candidate}!` as const\n }\n }\n\n // b. Remove candidates that are alone in a cell\n // maxSize is now possibleCandidates.size\n let keepGoing = true\n while (keepGoing) {\n keepGoing = false\n for (const cell of group) {\n if (cell.candidates.length === 1) {\n removeCandidate(cell.candidates[0])\n keepGoing = true\n }\n }\n\n for (const candidate of possibleCandidates) {\n if (occurances[candidate] > possibleCandidates.size) {\n removeCandidate(candidate)\n keepGoing = true\n }\n }\n }\n}\n\n/**\n * Here, a \"group\" is a row, column, or box, but can be any group.\n *\n * Within that group, we're trying to find\n * n candidates which can only go in n cells.\n *\n * In Andrew Stuart's solver, this is equivalent to finding\n * hidden pairs, triples, and quads.\n *\n * See {@link hiddenPairsTriplesAndQuads}\n *\n * I think this is O(n^4)\n *\n * @param group - A group of cells. Generally a row, column, or box.\n * @param maxSize - The maximum size of the conjugate. Default is 4.\n * (Not looking for conjugates of size 5 or more, since then there would be a\n * size 4 with the other cells by default. TODO better explanation)\n */\nfunction findHiddenConjugatesOfGroup(group: CellGroup, maxSize = 4 as 2 | 3 | 4) {\n // copy cell objects and arrays before changing them\n group = group.map(cell => ({\n position: cell.position,\n candidates: cell.candidates.slice()\n }))\n\n // 1. Filter the possible candidates (return if error)\n const possibleCandidates = new Set([1, 2, 3, 4, 5, 6, 7, 8, 9] as const)\n const __result = __filterPossibleCandidates(group, maxSize, possibleCandidates)\n if (typeof __result === \"string\") {\n return __result\n }\n\n // c. Filter out cells that have too few candidates\n // (No limit on max candidates)\n const possibleCells = group.filter(cell => 1 < cell.candidates.length)\n\n if (possibleCandidates.size < 2 || possibleCells.length < 2) {\n return []\n }\n\n // 2. Do the regular pairsTriplesAndQuads function.\n // Time to find the conjugates, this time n candidates that must be in n cells\n\n maxSize = Math.min(maxSize, possibleCells.length, possibleCandidates.size) as 2 | 3 | 4\n\n const conjugates = []\n const conjugateCands = [] // Only used in one location\n\n for (const candidatesOfConjugate of combinations(Array.from(possibleCandidates), 2, maxSize)) {\n const conjugate = getConjugateFromCandidates(possibleCells, candidatesOfConjugate)\n\n // if (candidatesOfConjugate.some(candidate => conjugate.every(cell => !cell.candidates.includes(candidate)))) {\n // throw new TypeError(JSON.stringify([group, conjugate, candidatesOfConjugate]))\n // }\n\n // e.g.: 3 candidates must be in 2 cells\n if (candidatesOfConjugate.length > conjugate.length) {\n return __errorHandling(candidatesOfConjugate, conjugate)\n } else if (candidatesOfConjugate.length === conjugate.length) {\n // A conjugate was found!!\n\n // Check if this conjugate exactly overlaps a previous one\n // If so, error just like above\n for (const [i, prevConjugate] of conjugates.entries()) {\n if (prevConjugate.length === conjugate.length && prevConjugate.every(cell => conjugate.some(cell2 => cell.position === cell2.position))) {\n return __errorHandling([...new Set(candidatesOfConjugate.concat(conjugateCands[i]))], conjugate)\n }\n }\n\n // Filter extra candidates\n const filteredConjugate = conjugate.map(cell => ({\n candidates: cell.candidates.filter(\n candidate => candidatesOfConjugate.includes(candidate)\n ),\n position: cell.position,\n }))\n\n conjugates.push(filteredConjugate)\n conjugateCands.push(candidatesOfConjugate)\n }\n }\n\n return conjugates\n}\n\n\nfunction findHiddenConjugatesOfSudoku(sudoku: PureSudoku, maxSize = 4 as 2 | 3 | 4) {\n const conjugates = [] as CellGroup[]\n const groups = sudoku.getGroups()\n for (const i of INDICES_TO_NINE) {\n const resultRow = findHiddenConjugatesOfGroup(groups[i * 3], maxSize)\n if (typeof resultRow === \"string\") {\n return `Row ${ROW_NAMES[i]}${resultRow}`\n }\n\n const resultColumn = findHiddenConjugatesOfGroup(groups[i * 3 + 1], maxSize)\n if (typeof resultColumn === \"string\") {\n return `Column ${COLUMN_NAMES[i]}${resultColumn}`\n }\n\n const resultBox = findHiddenConjugatesOfGroup(groups[i * 3 + 2], maxSize)\n if (typeof resultBox === \"string\") {\n return `Box ${BOX_NAMES[i]}${resultBox}`\n }\n\n conjugates.push(...resultRow, ...resultColumn, ...resultBox)\n }\n\n return conjugates\n}\n\n/**\n * You should probably look at {@link findHiddenConjugatesOfGroup}\n *\n * Consider the following:\n *\n * ```\n * ...45..89 .234567.9 ..34567.9\n * ....5..8. .23.5.... ..3.5....\n * 1........ .23.5...9 ..3.5...9\n * ```\n *\n * The hidden pair is `67`... but how is this detected?\n *\n * ## How to find hidden conjugates programmatically\n *\n * 1. First remove candidates that occur > 4 times\\\n * (`3` `5` and `9` are removed)\n *\n * ```\n * ...4...8. .2.4.67.. ...4.67..\n * .......8. .2....... .........\n * 1........ .2....... .........\n * ```\n *\n * 2. Remove candidates that are alone in a cell\\\n * (`2` `4` and `8` are removed)\n *\n * ```\n * ......... .....67.. .....67..\n * ......... ......... .........\n * ......... ......... .........\n * ```\n *\n * 3. Tada!!!!! Found them! (Use naked conjugate function)\n *\n * ### Footnote about step 2:\n *\n * If that lone candidate *was* part of a hidden pair/triple/quad\n * `123...... 123...... ..3......`\n *\n * then there would be a simpler hidden pair/triple/quad:\n * `12....... 12....... .........`\n */\nexport default function hiddenPairsTriplesAndQuads(sudoku: PureSudoku) {\n let successcount = 0\n\n const result = findHiddenConjugatesOfSudoku(sudoku)\n if (typeof result === \"string\") {\n window._custom.alert(result, AlertType.ERROR)\n return {\n success: false,\n successcount: SuccessError\n } as const\n }\n\n for (const conjugate of result) {\n let success = false\n\n for (const conjugateCell of conjugate) {\n const actualCell = sudoku.data[conjugateCell.position.row][conjugateCell.position.column]\n\n // If different, replace\n // Note: Candidates are only ever removed so just check lengths\n if (actualCell.length > conjugateCell.candidates.length) {\n sudoku.set(conjugateCell.position.row, conjugateCell.position.column).to(...conjugateCell.candidates)\n colorConjugate(sudoku, conjugate, 'solved')\n success = true\n }\n }\n\n // 1 success per conjugate\n if (success) {\n successcount++\n }\n }\n\n if (successcount === 0) {\n return {\n success: false\n } as const\n }\n\n return {\n success: true,\n successcount\n } as const\n}\n","import { ALL_CANDIDATES, IndexToNine, INDICES_TO_NINE, SudokuDigits } from \"../../Types\"\nimport PureSudoku from \"../Spaces/PureSudoku\"\nimport { boxAt } from \"../Utils\"\nimport { colorCandidateF } from \"../Utils.dependent\"\n\n/**\n * The state for a candidate in a group\n * At the beginning, the state isn't set. This is practically undefined\n * undefined = 0 found\n * [...] = 1 found, position of hidden single\n * false = 2+ found, hidden single not possible anymore\n */\ntype PossibleState = false | {\n row: IndexToNine\n column: IndexToNine\n}\n\n/**\n * For each group (row, column, or box) there are 9 candidates.\n * Digits go from 1 to 9\n *\n * So each group tracks the possibilities of each digit.\n */\ntype PossibleGroup = Partial>\n\n/** A group of rows or columns or boxes. */\ntype PossibleGroups = [PossibleGroup, PossibleGroup, PossibleGroup, PossibleGroup, PossibleGroup, PossibleGroup, PossibleGroup, PossibleGroup, PossibleGroup]\n\nfunction _CreateArrayOf9Groups (): PossibleGroups {\n return [\n {}, {}, {},\n {}, {}, {},\n {}, {}, {},\n ] as PossibleGroups\n}\n\n\nfunction _nextState (currentState: PossibleState | undefined, row: IndexToNine, column: IndexToNine) {\n if (currentState === undefined) {\n return { row, column }\n }\n\n return false\n}\n\n\nexport default function hiddenSingles(sudoku: PureSudoku) {\n const possible = {\n rows: _CreateArrayOf9Groups(),\n columns: _CreateArrayOf9Groups(),\n boxes: _CreateArrayOf9Groups(),\n }\n\n for (const row of INDICES_TO_NINE) {\n for (const column of INDICES_TO_NINE) {\n // Prevent hidden single when already solved\n if (sudoku.data[row][column].length === 1) {\n const candidate = sudoku.data[row][column][0]\n possible.rows[row][candidate] = false\n possible.columns[column][candidate] = false\n possible.boxes[boxAt(row, column)][candidate] = false\n } else {\n const box = boxAt(row, column)\n for (const candidate of sudoku.data[row][column]) {\n possible.rows[row][candidate] = _nextState(possible.rows[row][candidate], row, column)\n possible.columns[column][candidate] = _nextState(possible.columns[column][candidate], row, column)\n possible.boxes[box][candidate] = _nextState(possible.boxes[box][candidate], row, column)\n }\n }\n }\n }\n\n let successcount = 0\n for (const candidate of ALL_CANDIDATES) {\n for (let i = 0; i < 9; i++) {\n const currentPossible = [\n possible.rows[i][candidate],\n possible.columns[i][candidate],\n possible.boxes[i][candidate],\n ]\n\n for (const cell of currentPossible) {\n if (cell !== false && cell !== undefined) {\n successcount++\n sudoku.set(cell.row, cell.column).to(candidate)\n colorCandidateF(sudoku, cell.row, cell.column, candidate, 'solved')\n }\n }\n }\n }\n\n if (successcount !== 0) {\n return {\n success: true,\n successcount\n } as const\n }\n\n return {\n success: false\n } as const\n}\n","import { ALL_CANDIDATES, IndexToNine, INDICES_TO_NINE, SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { CellID } from \"../Utils\";\nimport { colorGroup } from \"../Utils.dependent\";\n\ntype PossibleNth = Readonly<{\n lines: Set[]\n sumLines: Set // Set of all cells in the lines\n}>\n\nfunction _innerWingLogic(\n candidate: SudokuDigits,\n candidateLocations: ReturnType,\n sudoku: PureSudoku,\n sumLines: Set,\n isRow: boolean,\n wingSize: number\n) {\n const patternRows = new Set()\n const patternColumns = new Set()\n for (const cell of sumLines) {\n patternRows.add(cell.row)\n patternColumns.add(cell.column)\n }\n\n // Undifferentiate rows and columns\n const [patternLines, patternPendLines, lineProp, pendLineProp] =\n isRow\n ? [patternRows, patternColumns, \"row\", \"column\"] as const\n : [patternColumns, patternRows, \"column\", \"row\"] as const\n\n if (patternPendLines.size <= wingSize) {\n // Pattern finally identified!\n let success = false\n\n for (const eliminationPendLine of patternPendLines) {\n for (const cell of candidateLocations[candidate][pendLineProp][eliminationPendLine]) {\n if (patternLines.has(cell[lineProp]) === false) {\n sudoku.remove(candidate).at(cell.row, cell.column)\n colorGroup(sudoku, sumLines, candidate)\n success = true\n }\n }\n }\n\n if (success) {\n return 1\n }\n }\n\n return 0\n}\n\n/**\n * Updates the accumulated arrays and sets.\n * If an accumulated array reaches the required amount of lines, callback()\n */\nfunction _accum (line: Set, possibleNLines: PossibleNth[][], index: IndexToNine, size: 2 | 3 | 4, callback: (sumIthLines: Set) => void) {\n const optimization = Math.max(index + size - 9, 0) // Example: If index = 8, size = 2,\n for (let i = possibleNLines.length - 1; i >= optimization; i--) {\n for (const ithLines of possibleNLines[i]) {\n const sumIthLines = new Set(ithLines.sumLines)\n line.forEach(cell => sumIthLines.add(cell))\n\n if (sumIthLines.size <= size * size) {\n if (i + 1 === size) {\n callback(sumIthLines)\n } else {\n possibleNLines[i + 1].push({\n lines: [...ithLines.lines, line],\n sumLines: sumIthLines\n })\n }\n }\n }\n }\n}\n\n/**\n * See Strategies.md#fish\n */\nexport default function fish (size: 2 | 3 | 4, sudoku: PureSudoku) {\n let successcount = 0\n\n const candidateLocations = sudoku.getCandidateLocations()\n for (const candidate of ALL_CANDIDATES) {\n const possibleNRows = [] as PossibleNth[][]\n const possibleNColumns = [] as PossibleNth[][]\n for (let i = 0; i < size; i++) {\n possibleNRows.push([])\n possibleNColumns.push([])\n }\n\n for (const index of INDICES_TO_NINE) {\n const row = candidateLocations[candidate].row[index]\n const column = candidateLocations[candidate].column[index]\n\n for (const [line, isRow, possibleNLines] of [[row, true, possibleNRows] as const, [column, false, possibleNColumns] as const]) {\n if (line.size <= size && line.size > 1) {\n // Optimization:\n // Say index = 6\n // Jellyfish (size 4) can't be made with only 6 7 8\n // But it can be with 5 6 7 8\n if (9 - index >= size) {\n possibleNLines[0].push({\n lines: [line],\n sumLines: new Set(line)\n })\n }\n\n _accum(line, possibleNLines, index, size, sumLines => { // eslint-disable-line no-loop-func\n successcount += _innerWingLogic(candidate, candidateLocations, sudoku, sumLines, isRow, size)\n })\n }\n }\n }\n }\n\n if (successcount === 0) {\n return {\n success: false\n } as const\n }\n\n return {\n success: true,\n successcount\n } as const\n}\n","import { GRP_TYPS, INDICES_TO_NINE, SudokuDigits } from \"../../Types\";\nimport PureSudoku, { CandidateLocations } from \"../Spaces/PureSudoku\";\nimport { SuccessError } from \"../Types\";\nimport { affects, assertGet, CellID, id, sharedInArrays } from \"../Utils\";\nimport { colorGroup, removeCandidateFromCells } from \"../Utils.dependent\";\n\nfunction _checkPair(\n cellA: CellID,\n cellB: CellID,\n candidateA: SudokuDigits,\n candidateB: SudokuDigits,\n affectsCW2C: Map,\n candidateLocations: CandidateLocations[],\n sudoku: PureSudoku,\n) {\n const affectsA = assertGet(affectsCW2C, cellA)\n const affectsB = assertGet(affectsCW2C, cellB)\n const affectsAB = sharedInArrays(affectsA, affectsB)\n\n // check for a row, column, or box\n // which has two cells: an affectsA and an affectsB\n for (const prop of GRP_TYPS) {\n for (const group of candidateLocations[candidateA][prop]) {\n if (group.size === 2) {\n const cellxA = affectsA.find(cell => group.has(cell))\n const cellxB = affectsB.find(cell => group.has(cell))\n\n // eslint-disable-next-line sonarjs/no-collapsible-if -- line too long\n if (cellxA !== cellxB && cellxA !== undefined && cellxB !== undefined) {\n // check for shared cells containing the other candidate\n if (removeCandidateFromCells(sudoku, candidateB, affectsAB)) {\n colorGroup(sudoku, [cellA, cellB, cellxA, cellxB], candidateA, \"green\")\n colorGroup(sudoku, [cellA, cellB], candidateB)\n return 1\n }\n }\n }\n }\n }\n return 0\n}\n\nfunction checkPair(\n cellA: CellID,\n cellB: CellID,\n candidateA: SudokuDigits,\n candidateB: SudokuDigits,\n affectsCW2C: Map,\n candidateLocations: CandidateLocations[],\n sudoku: PureSudoku,\n): number | string {\n return _checkPair(cellA, cellB, candidateA, candidateB, affectsCW2C, candidateLocations, sudoku) +\n _checkPair(cellA, cellB, candidateB, candidateA, affectsCW2C, candidateLocations, sudoku)\n}\n\n/**\n * Calls {@param callback} with all pairs of cells\n * whose two candidates are equal\n *\n * NOTE: You must call another function with (candidateA, candidateB)\n * and (candidateB, candidateA)\n */\nexport function wWingBase (sudoku: PureSudoku, callback: typeof checkPair) {\n const found = new Map()\n // Delay calculations\n const affectsCW2C = new Map()\n let candidateLocations\n\n for (const row of INDICES_TO_NINE) {\n for (const column of INDICES_TO_NINE) {\n const cell = sudoku.data[row][column]\n if (cell.length === 2) {\n const numericID = cell[0] * 10 + cell[1]\n const equivs = found.get(numericID)\n const cid = id(row, column)\n affectsCW2C.set(cid, affects(row, column))\n if (equivs === undefined) {\n found.set(numericID, [cid])\n } else {\n candidateLocations ??= sudoku.getCandidateLocations()\n for (const cell2 of equivs) {\n const [candidateA, candidateB] = cell\n const successcount =\n callback(cid, cell2, candidateA, candidateB, affectsCW2C, candidateLocations, sudoku)\n if (typeof successcount === \"string\") {\n return {\n success: false,\n successcount: SuccessError,\n message: successcount\n }\n } else if (successcount) {\n return {\n success: true,\n successcount\n } as const\n }\n }\n equivs.push(cid)\n }\n }\n }\n }\n\n return {\n success: false\n } as const\n}\n\n/**\n * http://sudopedia.enjoysudoku.com/W-Wing.html\n *\n * @example\n * If the As are strongly linked (and ABs see As)\n * then A can be eliminated in the shared AB cells\n * ```\n * x x x | AB | A\n * AB | x x x | A\n * ```\n */\nexport default function wWing (sudoku: PureSudoku) {\n return wWingBase(sudoku, checkPair)\n}\n","import { GRP_TYPS, SudokuDigits, GRP_NAMES } from \"../../Types\";\nimport PureSudoku, { CandidateLocations } from \"../Spaces/PureSudoku\";\nimport { affects, assertGet, CellID, groupInfo, isSubarray, isSubset, setDifference, sharedInArrays, sharedInSets } from \"../Utils\";\nimport { colorGroup, highlightGroup, removeCandidateFromCells } from \"../Utils.dependent\";\nimport { wWingBase } from \"./wWing\";\n\n// TODO: Return information on what kind of success types there are\nfunction _checkPair(\n cellA: CellID,\n cellB: CellID,\n candidateA: SudokuDigits,\n candidateB: SudokuDigits,\n affectsCW2C: Map,\n candidateLocations: CandidateLocations[],\n sudoku: PureSudoku,\n) {\n const affectsA = assertGet(affectsCW2C, cellA)\n const affectsB = assertGet(affectsCW2C, cellB)\n const affectsAandB = sharedInArrays(affectsA, affectsB)\n const affectsEitherAorB = new Set(affectsA.concat(affectsB))\n\n let success = 0\n for (const prop of GRP_TYPS) {\n for (const group_A of candidateLocations[candidateA][prop]) {\n // ^ group [sees A or B] [has A or B]\n // Avoid eliminating everything\n if (group_A.size === 0) {\n const [groupIndex] = groupInfo(prop, group_A)\n window._custom.alert(`${prop} ${GRP_NAMES[prop][groupIndex]} has no possibilities for ${candidateA} !`)\n return \"error\"\n }\n\n // Condition 2: The two cells see all cells in the group that have A.\n if (group_A.isSubsetOf(affectsEitherAorB)) {\n // Elimination type A\n const [, allOfGroup, remaining] = groupInfo(prop, group_A)\n\n // Strategy does not work when X or Y is in the group\n if (allOfGroup.includes(cellA) || allOfGroup.includes(cellB)) {\n continue\n }\n\n // check for shared cells containing the other candidate\n if (removeCandidateFromCells(sudoku, candidateB, affectsAandB)) {\n highlightGroup(sudoku, remaining, \"orange\")\n colorGroup(sudoku, [cellA, cellB], candidateA, \"green\")\n colorGroup(sudoku, [cellA, cellB], candidateB)\n success++\n }\n\n // Group + sees A or B\n // const groupA = sharedInSets(affectsA, group_A) // <\n // const groupB = sharedInSets(affectsB, group_A)\n\n // Group + sees A or B + has A or B\n const group_B = [...allOfGroup].filter(cell => sudoku.data[cell.row][cell.column].includes(candidateB))\n const groupAA = sharedInSets(affectsA, group_A)\n const groupAB = sharedInArrays(affectsA, group_B)\n // const groupBA = [...groupB].filter(cell => sudoku.data[cell.row][cell.column].includes(candidateA))\n // const groupBB = [...groupB].filter(cell => sudoku.data[cell.row][cell.column].includes(candidateB))\n\n // Affects A or B + Not in G\n const aAnG = setDifference(affectsA, allOfGroup)\n\n const xSeesY = affectsA.includes(cellB)\n const cond3 = isSubset(group_B, affectsEitherAorB)\n\n // We have\n // Z sees X, Z not in GXA (groupAA), X = A or B, Y = A or B\n //\n // Elim B0\n // Z = B --> X = A --> GX != A\n //\n // Elim B and C (Z sees GXA) + cond 2\n // Z = A --> GXA != A --> GY = A --> Y != A\n //\n // Elim B cond 3\n // Z = A --> X = B --> GX != B --> GY = B --> Y != B\n //\n // Elim C (X sees Y)\n // Z = A --> X = B --> Y = A\n if (xSeesY || cond3) {\n for (const z of aAnG) {\n if (z === cellA || z === cellB) {\n continue\n }\n\n let affectsZ = affectsCW2C.get(z)\n if (affectsZ === undefined) {\n affectsZ = affects(z.row, z.column)\n affectsCW2C.set(z, affectsZ)\n }\n\n if (isSubarray(groupAA, affectsZ) && removeCandidateFromCells(sudoku, candidateA, [z])) {\n highlightGroup(sudoku, remaining, \"orange\")\n colorGroup(sudoku, [cellA, cellB], candidateA, \"green\")\n colorGroup(sudoku, [cellA, cellB], candidateB)\n success++\n }\n }\n }\n\n // Elim B2 Z sees X + Z sees GXB + cond 2 + cond 3\n // Z = B --> X = A --> GX != A --> GY = A --> Y != A\n // Z = B --> GXB != B --> GY = B --> Y != B\n if (cond3) {\n for (const z of aAnG) {\n if (z === cellA || z === cellB) {\n continue\n }\n\n let affectsZ = affectsCW2C.get(z)\n if (affectsZ === undefined) {\n affectsZ = affects(z.row, z.column)\n affectsCW2C.set(z, affectsZ)\n }\n\n if (isSubarray(groupAB, affectsZ) && removeCandidateFromCells(sudoku, candidateB, [z])) {\n highlightGroup(sudoku, remaining, \"salmon\")\n colorGroup(sudoku, [cellA, cellB], candidateA, \"green\")\n colorGroup(sudoku, [cellA, cellB], candidateB)\n success++\n }\n }\n }\n\n if (success) {\n return success\n }\n }\n }\n }\n return success\n}\n\nfunction checkPair(\n cellA: CellID,\n cellB: CellID,\n candidateA: SudokuDigits,\n candidateB: SudokuDigits,\n affectsCW2C: Map,\n candidateLocations: CandidateLocations[],\n sudoku: PureSudoku,\n) {\n const result1 = _checkPair(cellA, cellB, candidateA, candidateB, affectsCW2C, candidateLocations, sudoku)\n const result2 = _checkPair(cellA, cellB, candidateB, candidateA, affectsCW2C, candidateLocations, sudoku)\n const result3 = _checkPair(cellB, cellA, candidateA, candidateB, affectsCW2C, candidateLocations, sudoku)\n const result4 = _checkPair(cellB, cellA, candidateB, candidateA, affectsCW2C, candidateLocations, sudoku)\n\n if (result1 === \"error\" || result2 === \"error\" || result3 === \"error\" || result4 === \"error\") {\n return \"error\"\n }\n\n return result1 + result2 + result3 + result4\n}\n\n/**\n * http://sudopedia.enjoysudoku.com/2-String_Kite.html\n * https://github.com/icecream17/solver/wiki/Pair-covers-group\n *\n * Simultaneously more general than both W-Wing and 2-string kite\n *\n * This was implemented by pure accident.\n *\n * @example\n *\n * ## Elimination A\n *\n * ```\n * AB\n * CC CC\n * CC CC\n *\n * AB xx\n * ```\n *\n * It's extended because this is also applied to other groups,\n * e.g. rows and columns\n * ```\n * xx AB\n * CC\n * CC\n *\n * AB xx\n *\n * CC\n * CC\n * CC\n * ```\n *\n * Example 3\n *\n * ```\n * AB xx xx xx\n * xx xx xx AB\n * CC CC CC\n * ```\n *\n * Eliminations are in all shared cells not in the group\n *\n * ## Elimination B\n *\n * If A and B are not in the C cells,\n * A abd B can be eliminated from x\n * ```\n * | | C C C\n * x x x | x x AB |\n * x x AB | x x x |\n * ```\n *\n * Example 2\n *\n * ```\n * x x x | AB x x |\n * | | C C\n * | | C C\n * ---------+----------+----------\n * | | x\n * | | x\n * | | AB\n * ---------+----------+----------\n * | | x\n * | | x\n * | | x\n * ```\n */\nexport default function pairCoversGroup(sudoku: PureSudoku) {\n return wWingBase(sudoku, checkPair)\n}\n","import { ALL_CANDIDATES, IndexToNine, INDICES_TO_NINE, SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { affects, CellID, sharedInArrays } from \"../Utils\";\nimport { colorGroup, removeCandidateFromCells } from \"../Utils.dependent\";\n\nexport function __incrementMapValue, K>(map: T, key: K) {\n const value = map.get(key)\n if (value === undefined) {\n map.set(key, 1)\n } else {\n map.set(key, value + 1)\n }\n}\n\nfunction _innerInnerSkyscraperLogic(\n candidate: SudokuDigits,\n sudoku: PureSudoku,\n sumLines: Set,\n perpendProp: \"row\" | \"column\",\n wingSize: number\n) {\n const patternPendLines = new Map()\n for (const cell of sumLines) {\n __incrementMapValue(patternPendLines, cell[perpendProp])\n }\n\n if (patternPendLines.size === wingSize + 1) {\n for (const [eliminationPendLine, count] of patternPendLines) {\n if (count > 1) {\n const cellsNotInLine = [] as CellID[]\n const affectsNotInLine = [] as CellID[][]\n for (const cell of sumLines) {\n if (cell[perpendProp] !== eliminationPendLine) {\n cellsNotInLine.push(cell)\n affectsNotInLine.push(affects(cell.row, cell.column))\n }\n }\n\n // shared = all extra see\n const shared = sharedInArrays(...affectsNotInLine)\n\n if (removeCandidateFromCells(sudoku, candidate, shared)) {\n colorGroup(sudoku, sumLines, candidate)\n colorGroup(sudoku, cellsNotInLine, candidate, \"orange\")\n return {\n success: true,\n successcount: 1\n } as const\n }\n }\n }\n }\n\n return null\n}\n\nfunction _innerSkyscraperLogic(line1: Set, possibleLines: Set[], candidate: SudokuDigits, sudoku: PureSudoku, perpendProp: \"row\" | \"column\") {\n // line = row/column\n // pendLine = column/row\n for (const line2 of possibleLines) {\n const sumLines = new Set()\n line1.forEach(cell => sumLines.add(cell))\n line2.forEach(cell => sumLines.add(cell))\n\n if (sumLines.size < 5) {\n const result = _innerInnerSkyscraperLogic(candidate, sudoku, sumLines, perpendProp, 2)\n if (result !== null) {\n return result\n }\n }\n }\n\n return null\n}\n\n/**\n * Disjointed x wing - see Strategies.md\n *\n * Two lines - 1 cross line = extra\n * If all extra see n, n is eliminated (since extra must have at least 1)\n */\nexport default function skyscraper(sudoku: PureSudoku) {\n const candidateLocations = sudoku.getCandidateLocations()\n for (const candidate of ALL_CANDIDATES) {\n const possibleRows = [] as Set[]\n const possibleColumns = [] as Set[]\n for (const index of INDICES_TO_NINE) {\n const row = candidateLocations[candidate].row[index]\n const column = candidateLocations[candidate].column[index]\n\n if (row.size < 3) {\n const result = _innerSkyscraperLogic(row, possibleRows, candidate, sudoku, \"column\")\n if (result !== null) {\n return result\n }\n\n possibleRows.push(row)\n }\n\n if (column.size < 3) {\n const result = _innerSkyscraperLogic(column, possibleColumns, candidate, sudoku, \"row\")\n if (result !== null) {\n return result\n }\n\n possibleColumns.push(column)\n }\n }\n }\n\n return {\n success: false\n } as const\n}\n","import { ALL_CANDIDATES, IndexToNine, INDICES_TO_NINE, SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { affects, CellID, sharedInArrays } from \"../Utils\";\nimport { colorGroup } from \"../Utils.dependent\";\nimport { __incrementMapValue } from \"./skyscraper\";\n\nfunction __inLine(sumLines: Set, eliminationPendLine: IndexToNine, pendLineProp: \"column\" | \"row\") {\n const inLine = new Set()\n for (const cell of sumLines) {\n if (cell[pendLineProp] === eliminationPendLine) {\n inLine.add(cell)\n }\n }\n\n return inLine\n}\n\nfunction __updatePatternPendLineElims(\n sudoku: PureSudoku,\n cell: CellID,\n candidate: SudokuDigits,\n sumLines: Set,\n patternPendLineElims: Map,\n eliminationPendLine: IndexToNine\n) {\n const complexCondition =\n sudoku.data[cell.row][cell.column].includes(candidate) &&\n !sumLines.has(cell)\n\n if (complexCondition) {\n const recieved = patternPendLineElims.get(cell)\n if (recieved === undefined) {\n patternPendLineElims.set(cell, [eliminationPendLine])\n } else {\n patternPendLineElims.set(cell, [...recieved, eliminationPendLine])\n }\n }\n}\n\nexport function _innerGroupSubtractionLogic(\n candidate: SudokuDigits,\n sudoku: PureSudoku,\n sumLines: Set,\n isRow: boolean,\n wingSize: number\n) {\n const patternRows = new Map()\n const patternColumns = new Map()\n for (const cell of sumLines) {\n __incrementMapValue(patternRows, cell.row)\n __incrementMapValue(patternColumns, cell.column)\n }\n\n // How many lines see a candidate\n const patternPendLineElims = new Map()\n const patternPendLines = isRow ? patternColumns : patternRows\n const pendLineProp = isRow ? \"column\" : \"row\"\n\n /**\n * A B C | D\n * ! |\n * ! |\n * --------+--\n * C | D\n *\n * A B C | D | D\n * A B C | D | D\n * ab ab ! | |\n * --------+-----+---\n * a b C | D | D\n * Take each column and ask what they collectively see\n * Anything seen by > total - wingSize columns can be eliminated\n */\n for (const [eliminationPendLine] of patternPendLines) {\n const inLine = [] as CellID[][]\n for (const cell of sumLines) {\n if (cell[pendLineProp] === eliminationPendLine) {\n inLine.push(affects(cell.row, cell.column))\n }\n }\n\n // shared = all extra see\n const shared = sharedInArrays(...inLine)\n for (const cell of shared) {\n __updatePatternPendLineElims(sudoku, cell, candidate, sumLines, patternPendLineElims, eliminationPendLine)\n }\n }\n\n let successcount = 0\n let nonExtraLine = null\n for (const [cell, linesWhichSee] of patternPendLineElims) {\n if (patternPendLines.size - linesWhichSee.length < wingSize) {\n const currentNonExtraLine = [...patternPendLines.keys()].find(line => !linesWhichSee.includes(line))\n nonExtraLine ??= currentNonExtraLine\n if (nonExtraLine === currentNonExtraLine) {\n successcount++\n sudoku.remove(candidate).at(cell.row, cell.column)\n }\n }\n }\n\n if (successcount) {\n const nonExtraLineCells = __inLine(sumLines, nonExtraLine as IndexToNine, pendLineProp)\n const extraCells = sumLines.difference(nonExtraLineCells)\n colorGroup(sudoku, extraCells, candidate, \"orange\")\n colorGroup(sudoku, nonExtraLineCells, candidate)\n return {\n success: true,\n successcount\n }\n }\n\n return null\n}\n\n/**\n * Disjointed x wing - see Strategies.md\n *\n * Two lines - 1 cross line = extra\n * If all extra see n, n is eliminated (since extra must have at least 1)\n */\nexport default function twoMinusOneLines(sudoku: PureSudoku) {\n const candidateLocations = sudoku.getCandidateLocations()\n for (const candidate of ALL_CANDIDATES) {\n const possibleRows = [] as Set[]\n const possibleColumns = [] as Set[]\n for (const index of INDICES_TO_NINE) {\n const row = candidateLocations[candidate].row[index]\n const column = candidateLocations[candidate].column[index]\n\n const check = []\n if (row.size <= 4) { // 4 cells of a row cannot share anything affects other than the row\n check.push([row, possibleRows] as const)\n possibleRows.push(row) // Marker 1\n }\n\n if (column.size <= 4) {\n check.push([column, possibleColumns] as const)\n possibleColumns.push(column) // Marker 1\n }\n\n // line = row/column\n // pendLine = column/row\n for (const [line1, possibleLines] of check) {\n for (const line2 of possibleLines) {\n // Necessary because `Marker 1` happens before this\n if (line1 === line2) {\n continue\n }\n\n const sumLines = new Set()\n line1.forEach(cell => sumLines.add(cell))\n line2.forEach(cell => sumLines.add(cell))\n\n const result = _innerGroupSubtractionLogic(candidate, sudoku, sumLines, line1 === row, 2)\n if (result !== null) {\n return result\n }\n }\n }\n }\n }\n\n return {\n success: false\n } as const\n}\n","import { ALL_CANDIDATES, SudokuDigits } from \"../../Types\";\nimport PureSudoku, { CandidateLocations } from \"../Spaces/PureSudoku\";\nimport { CellID } from \"../Utils\";\nimport { colorCandidateF } from \"../Utils.dependent\";\n\n/**\n * Checks if two cells create a two string kite\n * Maybe these checkers could be symbolized as matchers\n */\nfunction check(cell1: CellID, cell2: CellID, candidate: SudokuDigits, candLocations: CandidateLocations, sudoku: PureSudoku) {\n /* eslint-disable sonarjs/no-collapsible-if -- It's clearer */\n if (cell1.row === cell2.row || cell1.column === cell2.column) {\n return 0\n }\n\n const sameRowAsCell1 = candLocations.row[cell1.row]\n const sameColAsCell2 = candLocations.column[cell2.column]\n if (sameRowAsCell1.size === 2 && sameColAsCell2.size === 2) {\n for (const cell1B of sameRowAsCell1) {\n if (cell1B !== cell1) {\n for (const cell2B of sameColAsCell2) {\n if (cell2B !== cell2) {\n // All this does is get 1b and 2b\n // 1 1b\n // 2\n // 2b\n if (sudoku.data[cell2B.row][cell1B.column].includes(candidate)) {\n colorCandidateF(sudoku, cell1.row, cell1.column, candidate)\n colorCandidateF(sudoku, cell2.row, cell2.column, candidate, 'green')\n colorCandidateF(sudoku, cell1B.row, cell1B.column, candidate)\n colorCandidateF(sudoku, cell2B.row, cell2B.column, candidate, 'green')\n sudoku.remove(candidate).at(cell2B.row, cell1B.column)\n return 1\n }\n }\n }\n }\n }\n }\n\n return 0\n}\n\nexport default function twoStringKite(sudoku: PureSudoku) {\n const candidateLocations = sudoku.getCandidateLocations()\n\n for (const candidate of ALL_CANDIDATES) {\n for (const box of candidateLocations[candidate].box) {\n if (box.size === 2) {\n const [cell1, cell2] = box\n const successcount =\n check(cell1, cell2, candidate, candidateLocations[candidate], sudoku) +\n check(cell2, cell1, candidate, candidateLocations[candidate], sudoku)\n if (successcount) {\n return {\n success: true,\n successcount\n } as const\n }\n }\n }\n }\n\n return { success: false } as const\n}\n","import { SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { affects, algebraic, assertGet, CandidateID, CellID, id, removeFromArray, sharedInSets } from \"../Utils\";\nimport { colorCandidate, getCellsWithNCandidates } from \"../Utils.dependent\";\n\n/**\n * next = has\n *\n * See {@link findLoop}\n */\nexport function cellIsValidLoop (sudoku: PureSudoku, sees: CellID, has: SudokuDigits, loop: CellID[]) {\n const cell = sudoku.data[sees.row][sees.column]\n return cell.includes(has) && !loop.includes(sees)\n}\n\nfunction seenByColor (sudoku: PureSudoku, color: CandidateID[]) {\n const seen = new Set()\n for (const { row, column, digit } of color) {\n for (const cell of affects(row, column)) {\n if (sudoku.data[cell.row][cell.column].includes(digit)) {\n seen.add(id(cell.row, cell.column, digit))\n }\n }\n }\n\n return seen\n}\n\n/**\n * Checks if a loop actually eliminates anything\n *\n * @param endsConnect If ends don't connect, only eliminate from the ends\n */\nfunction checkLoop (sudoku: PureSudoku, color1: CandidateID[], color2: CandidateID[]) {\n const seenByColor1 = seenByColor(sudoku, color1)\n const seenByColor2 = seenByColor(sudoku, color2)\n const seenByBoth = sharedInSets(seenByColor1, seenByColor2)\n\n if (seenByBoth.size > 0) {\n for (const candidate of color1) {\n colorCandidate(sudoku, candidate)\n }\n for (const candidate of color2) {\n colorCandidate(sudoku, candidate, \"green\")\n }\n for (const { row, column, digit } of seenByBoth) {\n sudoku.remove(digit).at(row, column)\n }\n\n return {\n success: true,\n successcount: 1,\n message: `${color2.map(cand => algebraic(cand.row, cand.column)).join(\"\\u200B<>\\u200B\")}\\u200B<>\\u200B`,\n } as const\n }\n\n return false\n}\n\n/**\n * Looking for a loop of cells\n *\n * AB, BC, CD, DE, EF, ... and so on, until you reach the end ZA,\n * which loops back to AB\n *\n * In an xyLoop you can be certain that the loop will either be:\n *\n * ABCDEF...Z\n * or\n * BCDEF....A\n */\nexport default function xyLoop (sudoku: PureSudoku) {\n /**\n * The most important util\n *\n * @param cell The cell just added to the loop\n * @param start The first cell in the loop\n * @param next The next cell in the loop needs to have *this* candidate\n * @param end The last cell in the loop needs to have *this* candidate\n * @param color1 Used for coloring the candidate for display\n * @param color2 Used for coloring the candidate for display\n * @param loop The current built up loop\n * @returns false if failed, CellID[] is loop was found\n */\n function findLoop (\n cell: CellID,\n start: CellID,\n next: SudokuDigits,\n end: SudokuDigits,\n color1: CandidateID[],\n color2: CandidateID[],\n loop: CellID[] = [cell]\n ): ReturnType | false {\n // All cells AB sees with 2 candidates\n const validAffectsCell = __getFellowCWTC(cell).filter(fellow => cellIsValidLoop(sudoku, fellow, next, loop))\n\n for (const possibleNext of validAffectsCell) {\n const nextNext = sudoku.data[possibleNext.row][possibleNext.column].find(\n candidate => candidate !== next) as SudokuDigits\n\n loop.push(possibleNext)\n\n // No parity check needed, AB BC CD --> ABC BCD\n color2.push(id(possibleNext.row, possibleNext.column, next))\n color1.push(id(possibleNext.row, possibleNext.column, nextNext))\n\n const endsConnect = affects(start.row, start.column).includes(possibleNext)\n if (nextNext === end && endsConnect) {\n const isLoopResult = checkLoop(sudoku, color1, color2)\n\n if (isLoopResult) {\n return isLoopResult\n }\n }\n\n const result = findLoop(possibleNext, start, nextNext, end, color1, color2, loop)\n if (result) {\n return result\n }\n\n loop.pop()\n color1.pop()\n color2.pop()\n }\n\n return false // Fail\n }\n\n const __getFellowCWTC = (cell: CellID) =>\n assertGet(affectsCWTC, cell).filter(sees => cellsWithTwoCandidates.includes(sees))\n\n\n const cellsWithTwoCandidates = getCellsWithNCandidates(sudoku, 2)\n\n // CWTC acronym for cellsWithTwoCandidates\n const affectsCWTC = new Map(\n cellsWithTwoCandidates.map(cell => [cell, affects(cell.row, cell.column)])\n )\n\n for (const cell of cellsWithTwoCandidates) {\n const [candA, candB] = sudoku.data[cell.row][cell.column]\n\n // Candidate coloring\n const color1 = [id(cell.row, cell.column, candA)] as CandidateID[]\n const color2 = [id(cell.row, cell.column, candB)] as CandidateID[]\n\n // Start with candA\n // With a recursive function, add to the list until it fails or succeeds\n const result = findLoop(cell, cell, candA, candB, color1, color2)\n if (result) {\n return result\n }\n\n // Failed, so that cell must not be in any loop, it can be removed\n removeFromArray(cell, cellsWithTwoCandidates)\n }\n\n return {\n success: false\n } as const\n}\n","import { SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { affects, algebraic, assertGet, CandidateID, CellID, id, sharedInSets } from \"../Utils\";\nimport { colorCandidate, getCellsWithNCandidates, highlightCell } from \"../Utils.dependent\";\nimport { cellIsValidLoop } from \"./xyLoop\";\n\n// Very similar to seenByColor in xyLoop\nfunction seenByEnd (sudoku: PureSudoku, { row, column, digit }: CandidateID) {\n const seen = new Set()\n for (const cell of affects(row, column)) {\n if (sudoku.data[cell.row][cell.column].includes(digit)) {\n seen.add(id(cell.row, cell.column, digit))\n }\n }\n\n return seen\n}\n\n/**\n * Checks if a loop (or here, a chain) actually eliminates anything\n */\nfunction checkLoop (sudoku: PureSudoku, color1: CandidateID[], color2: CandidateID[]) {\n const color1End = color1[color1.length - 1]\n const color2End = color2[0]\n const seenByColor1 = seenByEnd(sudoku, color1End)\n const seenByColor2 = seenByEnd(sudoku, color2End)\n const seenByBoth = sharedInSets(seenByColor1, seenByColor2)\n\n if (seenByBoth.size > 0) {\n highlightCell(sudoku, id(color1End.row, color1End.column), \"orange\")\n highlightCell(sudoku, id(color2End.row, color2End.column), \"orange\")\n\n for (const candidate of color1) {\n colorCandidate(sudoku, candidate)\n }\n for (const candidate of color2) {\n colorCandidate(sudoku, candidate, \"green\")\n }\n for (const { row, column, digit } of seenByBoth) {\n sudoku.remove(digit).at(row, column)\n }\n\n return {\n success: true,\n successcount: 1,\n message: `${color2End.digit} ${color2.map(cand => algebraic(cand.row, cand.column)).join(\"\\u200B<>\\u200B\")} ${color1End.digit}`,\n } as const\n }\n\n return false\n}\n\n/**\n * Looking for a chain of cells\n * The code used is extremely similar to xyLoop\n *\n * However, the loop doesn't have to be completed.\n * There's still the restriction that the first and last cells of the chain must share a candidate\n *\n * The logic in this case is either:\n * first cell = candidate --> not last cell\n * last cell = candidate --> not first cell\n *\n * Basically no matter what, one of the ends has the candidate.\n */\nexport default function xyChain(sudoku: PureSudoku) {\n /**\n * The most important util\n * Extremely similar to \"findLoop\" in xyLoop\n *\n * @param cell The cell just added to the loop\n * @param next The next cell in the loop needs to have *this* candidate\n * @param end The last cell in the loop needs to have *this* candidate\n * @param color1 Used for coloring the candidate for display\n * @param color2 Used for coloring the candidate for display\n * @param loop The current built up loop\n * @returns false if failed, CellID[] is loop was found\n */\n function findLoop(cell: CellID, next: SudokuDigits, end: SudokuDigits, color1: CandidateID[], color2: CandidateID[], loop: CellID[] = [cell]): ReturnType | false {\n // All cells AB sees with 2 candidates\n const validAffectsCell = __getFellowCWTC(cell).filter(fellow => cellIsValidLoop(sudoku, fellow, next, loop))\n\n for (const possibleNext of validAffectsCell) {\n const nextNext = sudoku.data[possibleNext.row][possibleNext.column].find(\n candidate => candidate !== next) as SudokuDigits\n\n loop.push(possibleNext)\n\n // No parity check needed, AB BC CD --> ABC BCD\n color2.push(id(possibleNext.row, possibleNext.column, next))\n color1.push(id(possibleNext.row, possibleNext.column, nextNext))\n\n if (nextNext === end) {\n // Don't care if ends connect\n const isLoopResult = checkLoop(sudoku, color1, color2)\n\n if (isLoopResult) {\n return isLoopResult\n }\n }\n\n const result = findLoop(possibleNext, nextNext, end, color1, color2, loop)\n if (result) {\n return result\n }\n\n loop.pop()\n color1.pop()\n color2.pop()\n }\n\n return false // Fail\n }\n\n const __getFellowCWTC = (cell: CellID) =>\n assertGet(affectsCWTC, cell).filter(sees => cellsWithTwoCandidates.includes(sees))\n\n\n const cellsWithTwoCandidates = getCellsWithNCandidates(sudoku, 2)\n\n // CWTC acronym for cellsWithTwoCandidates\n const affectsCWTC = new Map(\n cellsWithTwoCandidates.map(cell => [cell, affects(cell.row, cell.column)])\n )\n\n for (const cell of cellsWithTwoCandidates) {\n const [candA, candB] = sudoku.data[cell.row][cell.column]\n\n // Candidate coloring\n const color1 = [id(cell.row, cell.column, candA)] as CandidateID[]\n const color2 = [id(cell.row, cell.column, candB)] as CandidateID[]\n\n // Start with candA\n // With a recursive function, add to the list until it fails or succeeds\n const resultA = findLoop(cell, candA, candB, color1, color2)\n if (resultA) {\n return resultA\n }\n\n const resultB = findLoop(cell, candB, candA, color2, color1)\n if (resultB) {\n return resultB\n }\n }\n\n return {\n success: false\n } as const\n}\n","import { SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { affects, assertGet, CellID, sharedInArrays } from \"../Utils\";\nimport { getCellsWithNCandidates, colorGroup, removeCandidateFromCells } from \"../Utils.dependent\";\n\n/**\n * The AC cell has A but not B\n * The BC cell has B but not A\n */\nfunction cellIsValidWing(sudoku: PureSudoku, sees: CellID, has: SudokuDigits, notHas: SudokuDigits) {\n const cell = sudoku.data[sees.row][sees.column]\n return cell.includes(has) && !cell.includes(notHas)\n}\n\nexport default function yWing (sudoku: PureSudoku) {\n // AB BC\n // AC\n\n // Looking for a cell AB\n // Where A sees another A in AC\n // Where B sees another B in BC\n\n const cellsWithTwoCandidates = getCellsWithNCandidates(sudoku, 2)\n\n // CWTC acronym for cellsWithTwoCandidates\n const affectsCWTC = new Map(\n cellsWithTwoCandidates.map(cell => [cell, affects(cell.row, cell.column)])\n )\n\n for (const cell of cellsWithTwoCandidates) {\n const [ candA, candB ] = sudoku.data[cell.row][cell.column]\n\n // All cells AB sees with 2 candidates\n const validAffectsCell = assertGet(affectsCWTC, cell).filter(sees => cellsWithTwoCandidates.includes(sees))\n\n const possibleAC = validAffectsCell.filter(sees => cellIsValidWing(sudoku, sees, candA, candB))\n const possibleBC = validAffectsCell.filter(sees => cellIsValidWing(sudoku, sees, candB, candA))\n\n for (const AC of possibleAC) {\n for (const BC of possibleBC) {\n if (AC === BC) {\n continue;\n }\n\n const cellAC = sudoku.data[AC.row][AC.column]\n const cellBC = sudoku.data[BC.row][BC.column]\n const candC = cellAC.find(candidate => candidate !== candA) as SudokuDigits\n\n if (cellBC.includes(candC)) {\n // Found a strong link with C in AC and BC!\n const sharedEffects = sharedInArrays(\n assertGet(affectsCWTC, AC), assertGet(affectsCWTC, BC)\n )\n\n if (removeCandidateFromCells(sudoku, candC, sharedEffects)) {\n colorGroup(sudoku, [cell, AC], candA)\n colorGroup(sudoku, [cell, BC], candB, \"green\")\n colorGroup(sudoku, [AC, BC], candC, \"orange\")\n return {\n success: true,\n successcount: 1\n } as const\n }\n }\n }\n }\n }\n\n return {\n success: false\n } as const\n}\n","\nimport { Strategy } from \"../Types\";\nimport checkForSolved from \"./checkForSolved\";\nimport hiddenPairsTriplesAndQuads from \"./hiddenPairsTriplesAndQuads\";\nimport hiddenSingles from \"./hiddenSingles\";\nimport intersectionRemoval from \"./intersectionRemoval\";\nimport jellyfish from \"./jellyfish\";\nimport pairCoversGroup from \"./pairCoversGroup\";\nimport pairsTriplesAndQuads from \"./pairsTriplesAndQuads\";\nimport skyscraper from \"./skyscraper\";\nimport swordfish from \"./swordfish\";\nimport twoMinusOneLines from \"./twoMinusOneLines\";\nimport twoStringKite from \"./twoStringKite\";\nimport updateCandidates from \"./updateCandidates\";\nimport wWing from \"./wWing\";\nimport xWing from \"./xWing\";\nimport xyChain from \"./xyChain\";\nimport xyLoop from \"./xyLoop\";\nimport xyzWing from \"./xyzWing\";\nimport yWing from \"./yWing\";\n\n/**\n * If testing a particular strategy, import that strategy instead of this array.\n * This array is meant to be general not specific.\n */\nexport default [\n checkForSolved,\n updateCandidates,\n hiddenSingles,\n intersectionRemoval,\n pairsTriplesAndQuads,\n hiddenPairsTriplesAndQuads,\n xWing,\n swordfish,\n jellyfish,\n skyscraper,\n twoStringKite,\n yWing,\n twoMinusOneLines,\n wWing,\n xyzWing,\n pairCoversGroup,\n xyLoop,\n xyChain,\n] as Readonly\n","import { AlertType, NUMBER_OF_CELLS } from \"../../Types\"\nimport PureSudoku from \"../Spaces/PureSudoku\"\nimport { StrategyMemory, SuccessError } from \"../Types\"\nimport { numberOfCellsWithNCandidates } from \"../Utils.dependent\"\nimport checkValidity from \"./checkValidity\"\n\nexport default function checkForSolved(sudoku: PureSudoku, memory: StrategyMemory[0]) {\n const validity = checkValidity(sudoku)\n if (!validity.ok) {\n window._custom.alert(validity.message, AlertType.ERROR)\n return {\n success: false,\n message: validity.message,\n successcount: SuccessError,\n } as const\n }\n\n // Should this be before checkValidity?\n if (typeof memory.solved !== \"number\") {\n throw TypeError(`memory.solved is not a number - got ${String(memory.solved)}`)\n } else if (!Number.isInteger(memory.solved)) {\n throw TypeError(`memory.solved is not an integer - got ${memory.solved}`)\n } else if (0 > memory.solved || memory.solved > NUMBER_OF_CELLS) {\n throw TypeError(`impossible amount of memory.solved - got ${memory.solved}`)\n }\n\n const totalSolved = numberOfCellsWithNCandidates(sudoku, 1)\n if (totalSolved === NUMBER_OF_CELLS) {\n memory.solved = NUMBER_OF_CELLS\n return {\n success: true,\n successcount: NUMBER_OF_CELLS\n } as const\n }\n\n if (totalSolved !== memory.solved) {\n const difference = totalSolved - memory.solved\n memory.solved = totalSolved\n\n return {\n success: true,\n successcount: difference\n } as const\n }\n\n return {\n success: false\n } as const\n}\n","import { INDICES_TO_NINE } from \"../../Types\"\nimport PureSudoku from \"../Spaces/PureSudoku\"\nimport { SuccessError } from \"../Types\"\nimport { affects, algebraic, CellID } from \"../Utils\"\n\n// O(n^5)\nexport default function updateCandidates(sudoku: PureSudoku) {\n let updated = 0\n const newResults = new Set()\n\n for (const i of INDICES_TO_NINE) {\n for (const j of INDICES_TO_NINE) {\n // Cell\n if (sudoku.data[i][j].length === 1) {\n\n // Cell > Candidate\n const solvedCandidate = sudoku.data[i][j][0]\n\n // Cell > Affects\n for (const id of affects(i, j)) {\n\n // Cell > Affects > Cell\n const datacell = sudoku.data[id.row][id.column]\n const tempIndex = datacell.indexOf(solvedCandidate)\n\n // If has candidate\n if (tempIndex !== -1) {\n // If last candidate of that cell\n if (datacell.length === 1) {\n return {\n success: false,\n successcount: SuccessError,\n message: `Both ${algebraic(i, j)} and ${algebraic(id.row, id.column)} must be ${solvedCandidate}`\n }\n }\n\n datacell.splice(tempIndex, 1) // Deletes the candidate\n newResults.add(id)\n updated++\n }\n }\n }\n }\n }\n\n if (updated > 0) {\n for (const {row, column} of newResults) {\n sudoku.set(row, column).to(...sudoku.data[row][column]) // Don't run Cell#setState on every single candidate removal\n }\n\n return {\n success: true,\n successcount: updated\n } as const\n }\n\n return {\n success: false\n } as const\n}\n","import { ALL_CANDIDATES, INDICES_TO_NINE, SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { boxAt, CellID, removeFromArray } from \"../Utils\";\nimport { colorGroup } from \"../Utils.dependent\";\n\n/**\n * If all candidate N in group G attacks range R,\n * you can remove N from R\n */\nexport default function intersectionRemoval(sudoku: PureSudoku) {\n const messages = new Set()\n\n function _removeCandidate(candidate: SudokuDigits, cellID: CellID) {\n const newCandidates = sudoku.data[cellID.row][cellID.column]\n removeFromArray(candidate, newCandidates)\n sudoku.set(cellID.row, cellID.column).to(...newCandidates)\n\n candidateLocations[candidate].row[cellID.row].delete(cellID)\n candidateLocations[candidate].column[cellID.column].delete(cellID)\n candidateLocations[candidate].box[boxAt(cellID.row, cellID.column)].delete(cellID)\n }\n\n\n function _innerTwoGroupLogic(\n candidate: SudokuDigits,\n boxLocations: Set,\n group2Locations: Set,\n boxId: string,\n group2Id: string,\n ) {\n // boxDiff is a copy of boxLocations\n const boxDiff = new Set(boxLocations) // Box locations not in line\n const group2Diff = new Set() // Row locations not in box\n\n for (const group2Location of group2Locations) {\n if (boxDiff.has(group2Location)) {\n boxDiff.delete(group2Location)\n } else {\n group2Diff.add(group2Location)\n }\n }\n\n // In (line and box), but not (rest of box)\n if (boxDiff.size === 0 && group2Diff.size !== 0) {\n successcount++\n colorGroup(sudoku, boxLocations, candidate)\n for (const extraCell of group2Diff) {\n _removeCandidate(candidate, extraCell)\n messages.add(`${candidate}: ${boxId} ⇒ ${group2Id}`)\n }\n }\n\n // In (line and box), but not (rest of line)\n if (group2Diff.size === 0 && boxDiff.size !== 0) {\n successcount++\n colorGroup(sudoku, group2Locations, candidate)\n for (const extraCell of boxDiff) {\n _removeCandidate(candidate, extraCell)\n messages.add(`${candidate}: ${group2Id} ⇒ ${boxId}`)\n }\n }\n }\n\n\n let successcount = 0\n const candidateLocations = sudoku.getCandidateLocations()\n for (const candidate of ALL_CANDIDATES) {\n // Boxes vs Rows (and) Boxes vs Columns\n for (const boxIndex of INDICES_TO_NINE) {\n const boxLocations = candidateLocations[candidate].box[boxIndex]\n\n for (const groupIndex of INDICES_TO_NINE) {\n const rowLocations = candidateLocations[candidate].row[groupIndex]\n const columnLocations = candidateLocations[candidate].column[groupIndex]\n\n if (boxLocations.size < 4 || rowLocations.size < 4) {\n _innerTwoGroupLogic(candidate, boxLocations, rowLocations, `Box ${boxIndex + 1}`, `Row ${groupIndex + 1}`)\n }\n\n if (boxLocations.size < 4 || columnLocations.size < 4) {\n _innerTwoGroupLogic(candidate, boxLocations, columnLocations, `Box ${boxIndex + 1}`, `Column ${groupIndex + 1}`)\n }\n }\n }\n }\n\n if (successcount === 0) {\n return {\n success: false\n } as const\n }\n\n return {\n success: true,\n successcount,\n message: [...messages].join(\"\\n\")\n } as const\n}\n","import PureSudoku from \"../Spaces/PureSudoku\"\nimport fish from \"./fish\";\n\n/**\n * 2 candidates in 2 rows, which align on 2 columns\n *\n * or\n *\n * 2 candidates in 2 columns, which align on 2 rows\n */\nexport default function xWing (sudoku: PureSudoku) {\n return fish(2, sudoku)\n}\n","import PureSudoku from \"../Spaces/PureSudoku\"\nimport fish from \"./fish\";\n\n/**\n * Same as xWing, but with 3 lines\n */\nexport default function swordfish(sudoku: PureSudoku) {\n return fish(3, sudoku)\n}\n","import PureSudoku from \"../Spaces/PureSudoku\";\nimport fish from \"./fish\";\n\n/**\n * Same as xWing and swordfish, but with 4 lines\n */\nexport default function jellyfish(sudoku: PureSudoku) {\n return fish(4, sudoku)\n}\n","import { SudokuDigits } from \"../../Types\";\nimport PureSudoku from \"../Spaces/PureSudoku\";\nimport { affects, assertGet, boxAt, CellID, sharedInArrays } from \"../Utils\";\nimport { getCellsWithNCandidates, highlightCell, removeCandidateFromCells } from \"../Utils.dependent\";\n\n/* TODO: When finished move to Utils */\nfunction inSameBox(cellA: CellID, cellB: CellID) {\n return boxAt(cellA.row, cellA.column) === boxAt(cellB.row, cellB.column)\n}\n\n\n// /**\n// * The AC cell has A but not B\n// * The BC cell has B but not A\n// */\n// function cellIsValidWing (sudoku: PureSudoku, sees: CellID, has: SudokuDigits, notHas: SudokuDigits) {\n// const cell = sudoku.data[sees.row][sees.column]\n// return cell.includes(has) && !cell.includes(notHas)\n// }\n\nexport default function xyzWing (sudoku: PureSudoku) {\n // ABC eliminations | [ABC]{2}\n // AC |\n\n // Looking for ABC\n // which is in the same box as an AB\n // and also sees another two candidate cell with C (AC or BC)\n const cellsWithThreeCandidates = getCellsWithNCandidates(sudoku, 3)\n const cellsWithTwoCandidates = getCellsWithNCandidates(sudoku, 2)\n\n // CW3C acronym for cellsWithTwoCandidates\n const affectsCW3C = new Map(\n cellsWithThreeCandidates.map(cell => [cell, affects(cell.row, cell.column)])\n )\n const affectsCW2C = new Map(\n cellsWithTwoCandidates.map(cell => [cell, affects(cell.row, cell.column)])\n )\n\n for (const basecell of cellsWithThreeCandidates) {\n // AC BC --> ABC\n // AC AB --> ABC\n const sudokubasecell = sudoku.data[basecell.row][basecell.column]\n\n // All cells AB sees with 2 candidates\n const affectsBaseCell = assertGet(affectsCW3C, basecell)\n const validAffectsCell = affectsBaseCell.filter(\n sees => cellsWithTwoCandidates.includes(sees) && sudoku.data[sees.row][sees.column].every(candidate => sudokubasecell.includes(candidate))\n )\n const [valid1stWing, valid2ndWing] = validAffectsCell.reduce<[CellID[], CellID[]]>((accum, sees) => {\n if (inSameBox(sees, basecell)) {\n accum[0].push(sees)\n } else {\n accum[1].push(sees)\n }\n return accum\n }, [[], []])\n\n if (valid1stWing.length === 0 || valid2ndWing.length === 0) {\n continue\n }\n\n for (const wing1 of valid1stWing) {\n const wing1Cell = sudoku.data[wing1.row][wing1.column]\n const extraCandidate = sudokubasecell.find(candidate => !wing1Cell.includes(candidate)) as SudokuDigits\n\n for (const wing2 of valid2ndWing) {\n const wing2Cell = sudoku.data[wing2.row][wing2.column]\n if (wing2Cell.includes(extraCandidate) && wing2Cell.every(candidate => sudokubasecell.includes(candidate))) {\n const sharedCandidate = sudokubasecell.find(candidate =>\n wing1Cell.includes(candidate) && wing2Cell.includes(candidate)\n ) as SudokuDigits\n\n const affectsAll = sharedInArrays(\n affectsBaseCell,\n assertGet(affectsCW2C, wing1),\n assertGet(affectsCW2C, wing2)\n )\n\n if (removeCandidateFromCells(sudoku, sharedCandidate, affectsAll)) {\n highlightCell(sudoku, wing1)\n highlightCell(sudoku, wing2)\n highlightCell(sudoku, basecell, 'orange')\n return {\n success: true,\n successcount: 1\n } as const\n }\n }\n }\n }\n }\n\n return {\n success: false\n } as const\n}\n","import asyncPrompt from \"../asyncPrompt\"\nimport EventRegistry from \"../eventRegistry\"\nimport { AlertType } from \"../Types\"\nimport { forComponentsToUpdate } from \"../utils\"\nimport Sudoku from \"./Spaces/Sudoku\"\nimport STRATEGIES from \"./Strategies/Strategies\"\nimport { SuccessError, StrategyMemory } from \"./Types\"\nimport { numberOfCellsWithNCandidates } from \"./Utils.dependent\"\n\ntype SolverEvents = 'new turn' | 'step finish'\n\n/**\n * Keeps track of the solving strategies, and also implements a few commands\n * (Step, Go, Import, Export, Clear).\n */\nexport default class Solver {\n strategyIndex = 0\n\n /** Later steps wait for earlier steps to finish. Implemented using callback and promises */\n whenStepHasFinished: ((stop: boolean) => void)[] = []\n isDoingStep = false\n\n /** Information strategies keep across calls */\n memory = new StrategyMemory()\n\n /**\n * Whether a strategy is logically skippable (disabled does not apply).\n * Right now, all strategies are skippable if they're retried with no changes to the sudoku.\n */\n skippable = [] as boolean[]\n\n /** Which strategies are disabled (used by StrategyItems) */\n disabled = [] as boolean[]\n\n /** Used to reset + update the StrategyItems */\n eventRegistry = new EventRegistry()\n\n constructor(public sudoku: Sudoku) {\n // Bind the StrategyControl handlers which have capitzalized names\n this.Go = this.Go.bind(this)\n this.Step = this.Step.bind(this)\n this.Undo = this.Undo.bind(this)\n this.Import = this.Import.bind(this)\n this.Export = this.Export.bind(this)\n this.Clear = this.Clear.bind(this)\n }\n\n updateCounters (success: boolean, errored: boolean, solved: boolean) {\n if ((success && this.strategyIndex !== 0) || errored || solved) {\n // Go back to the start when a strategy succeeds, errors,\n // (or the sudoku is solved, because the user edited it or smth idk)\n this.strategyIndex = 0\n\n this.skippable.fill(false)\n\n // errored or solved\n if (!success) {\n return\n }\n }\n\n // Usually what happens here is that the strategy fails,\n // and is skipped. Which is practically the same as doing one step.\n\n // Exception: \"check for solved\" isn't really a strategy\n else if (this.strategyIndex === 0) {\n this.strategyIndex = 1 // sneaky, next if statement will skip \"update candidates\" iff \"check for solved\" failed\n this.skippable[0] = true // shouldn't matter, but set for clarity\n }\n\n if (!success) {\n this.skippable[this.strategyIndex] = true\n }\n\n while ((this.skippable[this.strategyIndex] || this.disabled[this.strategyIndex]) && this.strategyIndex !== 0) {\n this.__step()\n }\n }\n\n private __step () {\n this.strategyIndex++\n if (this.strategyIndex === STRATEGIES.length) {\n this.strategyIndex = 0\n }\n }\n\n private __promisifyCellMethod (methodName: T & (\"setExplainingToTrue\" | \"setExplainingToFalse\")) {\n const promises = [] as Promise[]\n\n for (const row of this.sudoku.cells) {\n for (const cell of row) {\n if (cell != null) {\n promises.push(new Promise(resolve => {\n cell[methodName](resolve)\n }))\n }\n }\n }\n\n return Promise.allSettled(promises)\n }\n\n /**\n * !async\n */\n setupCells () {\n return this.__promisifyCellMethod(\"setExplainingToTrue\")\n }\n\n /**\n * !async\n *\n * !misnomer\n *\n * For each cell, run {@link Cell#setExplainingToFalse}\n */\n resetCells () {\n return this.__promisifyCellMethod(\"setExplainingToFalse\")\n }\n\n /**\n * Returns a boolean: \"success\" as in went to next strategy\n */\n private goToNextStrategyIfDisabled () {\n if (this.disabled[this.strategyIndex]) {\n this.updateCounters(false, false, false)\n return true\n }\n\n return false\n }\n\n private async StartStep (): Promise {\n await forComponentsToUpdate()\n\n this.isDoingStep = true\n\n // This could theoretically go on forever, but right now the first\n // strategy cannot be disabled. TODO: Better solution\n\n // If current strat is disabled, go to first non-disabled strat.\n do {\n if (this.strategyIndex === 0) {\n this.eventRegistry.notify('new turn')\n await this.resetCells()\n }\n } while (this.goToNextStrategyIfDisabled())\n\n // Set cells to strategy mode\n await this.setupCells()\n }\n\n private async FinishStep (strategyResult: {\n success: boolean\n successcount: number | null\n message: string | null\n }) {\n // Set cells to non-strategy mode if failed\n if (strategyResult.success === false) {\n await this.resetCells()\n }\n\n // notify the strategyItem UI\n this.eventRegistry.notify('step finish', strategyResult, this.strategyIndex)\n await forComponentsToUpdate()\n\n // \"check for solved\" can return -1 without being an error\n // if the user edits the sudoku\n const errored = !strategyResult.success && strategyResult.successcount === SuccessError\n const solved = numberOfCellsWithNCandidates(this.sudoku, 1) === 81\n if (solved) {\n window._custom.alert(\"Finished! :D\", AlertType.SUCCESS)\n }\n\n this.updateCounters(strategyResult.success, errored, solved)\n this.isDoingStep = false\n\n return errored || solved\n }\n\n // This is a big function.\n // Each comment labels a group of code that does something\n\n // Originally Promise\n async Step (): Promise {\n if (this.isDoingStep) {\n // Don't do this step yet\n // Wait for any previous steps to finish\n // After that, continue to the main code\n const stop = await new Promise(resolve => {\n this.whenStepHasFinished.push(resolve)\n })\n\n this.whenStepHasFinished.shift()\n\n if (stop) {\n this.whenStepHasFinished[0]?.(stop)\n return\n }\n }\n\n // Main code\n await this.StartStep()\n\n // Run strategy\n const _strategyResult = STRATEGIES[this.strategyIndex](this.sudoku, this.memory[this.strategyIndex])\n const strategyResult = {\n success: _strategyResult.success,\n successcount: _strategyResult.successcount ?? null,\n message: _strategyResult.message ?? null,\n }\n\n const stop = await this.FinishStep(strategyResult)\n\n // Do the next step if it's waiting for this one\n this.whenStepHasFinished[0]?.(stop)\n }\n\n /** Does \"Step\" until it reaches the end or a strategy succeeds */\n async Go () {\n do {\n await this.Step()\n } while (this.strategyIndex !== 0)\n }\n\n async Undo () {\n if (this.sudoku === null) return;\n for (const row of this.sudoku.cells) {\n for (const cell of row) {\n cell?.undo(() => {\n void this.sudoku\n .set(cell.props.row, cell.props.column)\n .to(...cell.state.candidates)\n })\n }\n }\n this.skippable[this.strategyIndex] = false\n await forComponentsToUpdate()\n }\n\n async Import () {\n const result = await asyncPrompt(\"Import\", \"Enter digits or candidates\")\n if (result === null || result === \"\") {\n return; // Don't import on cancel\n }\n\n await this.reset()\n this.sudoku.import(result)\n }\n\n Export () {\n window._custom.alert(this.sudoku.to81(), undefined, \"monospace\")\n window._custom.alert(this.sudoku.to729(), undefined, \"monospace\")\n }\n\n async Clear () {\n this.sudoku.clear()\n await this.reset()\n }\n\n async reset () {\n // BUG: Doesn't wait for steps to finish\n await this.resetCells()\n this.eventRegistry.notify('new turn')\n this.memory = new StrategyMemory()\n this.whenStepHasFinished = []\n this.strategyIndex = 0\n this.skippable = []\n }\n}\n","import { PromptCallback } from \"./Types\"\n\nexport default function asyncPrompt(title: string, message: string, defaultResult?: string, cssCls?: string): Promise {\n return new Promise(resolve => {\n window._custom.prompt(title, message, defaultResult, resolve as PromptCallback, cssCls)\n })\n}\n","\nimport './AlertNotice.css'\nimport React from 'react'\nimport Control from '../Control'\nimport { AlertType } from '../../Types'\n\ntype AlertProps = {\n whenFinish: () => void\n message: string\n type: AlertType\n cssCls?: string\n}\n\nexport default class AlertNotice extends React.Component {\n render () {\n const text = [...this.props.message].map((character, index) => (\n character === '\\n'\n ?

    \n : character\n ))\n return (\n
    \n

    {text}

    \n Ok\n
    \n )\n }\n}\n","\nimport './PromptWindow.css'\nimport React from 'react'\nimport { PromptCallback } from '../../Types'\nimport Control from '../Control'\n\n// Could also have other props (e.g. \"type\")\ntype PromptWindowProps = {\n whenFinish: () => void,\n\n title: string,\n message: string,\n defaultResponse: string,\n callback?: PromptCallback,\n cssCls?: string,\n}\n\nexport default class PromptWindow extends React.Component {\n inputElement: HTMLTextAreaElement | null = null\n setInputElement: (element: HTMLTextAreaElement | null) => HTMLTextAreaElement | null\n constructor (props: PromptWindowProps) {\n super(props)\n this.cancel = this.cancel.bind(this)\n this.submit = this.submit.bind(this)\n this.setInputElement = (element: HTMLTextAreaElement | null) => this.inputElement = element\n }\n\n render() {\n return (\n
    \n
    \n

    {this.props.title}

    \n