Forked from dclamage/fpuzzles-solution.user.js
Last active
November 20, 2022 08:00
-
-
Save killroy42/7c86b25da1c76f7173c51714c2bd615f to your computer and use it in GitHub Desktop.
Allows setters to specify a solution to the puzzle, which is checked in solve mode when the final digit is entered.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name Fpuzzles-Solution | |
// @namespace http://tampermonkey.net/ | |
// @version 1.6 | |
// @description Can input a solution to a puzzle | |
// @author Rangsk | |
// @match https://*.f-puzzles.com/* | |
// @match https://f-puzzles.com/* | |
// @icon  | |
// @grant none | |
// @run-at document-end | |
// ==/UserScript== | |
(function() { | |
'use strict'; | |
const numSolutionToolsButtons = 3; | |
const doShim = function() { | |
// Additional import/export data | |
const origExportPuzzle = window.exportPuzzle; | |
window.exportPuzzle = function(includeCandidates) { | |
const compressed = origExportPuzzle(includeCandidates); | |
const puzzle = JSON.parse(compressor.decompressFromBase64(compressed)); | |
if (window.grid.solution) { | |
puzzle.solution = window.grid.solution; | |
} | |
return compressor.compressToBase64(JSON.stringify(puzzle)); | |
} | |
const origImportPuzzle = window.importPuzzle; | |
window.importPuzzle = function(string, clearHistory) { | |
origImportPuzzle(string, clearHistory); | |
const puzzle = JSON.parse(compressor.decompressFromBase64(string)); | |
if (puzzle.solution) { | |
window.grid.solution = puzzle.solution; | |
} else if (window.grid.solution !== undefined) { | |
delete window.grid.solution; | |
} | |
} | |
const origCreateSidebars = window.createSidebars; | |
window.createSidebars = function() { | |
origCreateSidebars(); | |
const sidebar = window.sidebars[0]; | |
sidebar.show = function() { | |
if (this.modes.includes(mode)) { | |
ctx.lineWidth = lineWW; | |
ctx.fillStyle = boolSettings['Dark Mode'] ? '#404040' : '#D0D0D0'; | |
ctx.strokeStyle = boolSettings['Dark Mode'] ? '#202020' : '#808080'; | |
ctx.fillRect(this.x - sidebarW / 2, gridY, sidebarW, gridSL + 35); | |
ctx.strokeRect(this.x - sidebarW / 2, gridY, sidebarW, gridSL + 35); | |
for (var a = 0; a < this.sections.length; a++) | |
this.sections[a].show(); | |
for (var a = 0; a < this.buttons.length; a++) | |
this.buttons[a].show(); | |
} | |
} | |
const cosmeticToolsButton = sidebars[0].buttons.find(b => b.id === 'CosmeticTools'); | |
const solutionToolsX = cosmeticToolsButton.x; | |
const solutionToolsY = cosmeticToolsButton.y + buttonSH + buttonGap; | |
const solutionToolsButton = new button(solutionToolsX, solutionToolsY, buttonW, buttonSH, ['Setting'], 'SolutionTools', 'Solution Tools', true, true); | |
sidebar.buttons.push(solutionToolsButton); | |
solutionToolsButton.click = function() { | |
if (!this.hovering() || !this.modes.includes(window.mode)) { | |
return; | |
} | |
togglePopup('SolutionTools'); | |
} | |
const baseX = gridX - (sidebarDist + sidebarW / 2) + sidebarW; | |
const baseY = gridY + gridSL + 35 - ((buttonSH * numSolutionToolsButtons) + (buttonGap * (numSolutionToolsButtons - 1)) + buttonMargin); | |
const saveSolutionButton = new button(baseX, baseY + (buttonSH + buttonGap) * 0, buttonW, buttonSH, ['SolutionTools'], 'SaveSolution', 'Save Solution'); | |
const viewSolutionButton = new button(baseX, baseY + (buttonSH + buttonGap) * 1, buttonW, buttonSH, ['SolutionTools'], 'ViewSolution', 'View Solution'); | |
const clearSolutionButton = new button(baseX, baseY + (buttonSH + buttonGap) * 2, buttonW, buttonSH, ['SolutionTools'], 'ClearSolution', 'Clear Solution'); | |
saveSolutionButton.click = function() { | |
if (!this.hovering()) { | |
return; | |
} | |
let solution = []; | |
for (let i = 0; i < size; i++) { | |
for (let j = 0; j < size; j++) { | |
let value = window.grid[i][j].value; | |
if (value == 0) { | |
value = '.'; | |
} | |
solution.push(value); | |
} | |
} | |
if(solution.every(value => value === '.')) { | |
// Solution is entirely blank | |
delete window.grid.solution; | |
} | |
else { | |
window.grid.solution = solution; | |
} | |
return true; | |
} | |
viewSolutionButton.click = function() { | |
if (!this.hovering()) { | |
return; | |
} | |
if (window.grid.solution) { | |
const solution = window.grid.solution; | |
for (let i = 0; i < size; i++) { | |
for (let j = 0; j < size; j++) { | |
let value = solution[i * size + j]; | |
if (value == '.') { | |
value = 0; | |
} | |
window.grid[i][j].value = value; | |
} | |
} | |
} | |
return true; | |
} | |
clearSolutionButton.click = function() { | |
if (!this.hovering()) { | |
return; | |
} | |
delete window.grid.solution; | |
return true; | |
} | |
sidebar.buttons.push(saveSolutionButton); | |
sidebar.buttons.push(viewSolutionButton); | |
sidebar.buttons.push(clearSolutionButton); | |
} | |
const origDrawPopups = window.drawPopups; | |
window.drawPopups = function(overlapSidebars) { | |
origDrawPopups(overlapSidebars); | |
if (overlapSidebars) { | |
return; | |
} | |
if (popup === "SolutionTools") { | |
ctx.lineWidth = lineWW; | |
ctx.fillStyle = boolSettings['Dark Mode'] ? '#404040' : '#D0D0D0'; | |
ctx.strokeStyle = boolSettings['Dark Mode'] ? '#202020' : '#808080'; | |
ctx.fillRect(gridX - sidebarDist, gridY + gridSL + 35, sidebarW, -((buttonSH * numSolutionToolsButtons) + (buttonGap * (numSolutionToolsButtons - 1)) + (buttonMargin * 2))); | |
ctx.strokeRect(gridX - sidebarDist, gridY + gridSL + 35, sidebarW, -((buttonSH * numSolutionToolsButtons) + (buttonGap * (numSolutionToolsButtons - 1)) + (buttonMargin * 2))); | |
} | |
} | |
const origMouseMove = document.onmousemove; | |
document.onmousemove = function(e) { | |
origMouseMove(e); | |
if (!testPaused() && !disableInputs) { | |
if (sidebars.length) { | |
var hoveredButton = sidebars[sidebars.findIndex(a => a.title === 'Constraints')].buttons[sidebars[sidebars.findIndex(a => a.title === 'Constraints')].buttons.findIndex(a => a.id === 'SolutionTools')]; | |
if (popup === 'SolutionTools' && (mouseX < hoveredButton.x - hoveredButton.w / 2 - buttonMargin || mouseX > hoveredButton.x + hoveredButton.w / 2 + buttonMargin || mouseY < hoveredButton.y - buttonMargin || mouseY > hoveredButton.y + buttonSH + buttonMargin) && | |
(mouseX < gridX - sidebarDist || mouseX > gridX - sidebarDist + sidebarW + (buttonGap + buttonSH) * 2 || mouseY < gridY + gridSL + 35 - ((buttonSH * numSolutionToolsButtons) + (buttonGap * (numSolutionToolsButtons - 1)) + (buttonMargin * 2)) || mouseY > gridY + gridSL + 35)) { | |
closePopups(); | |
} | |
} | |
} | |
} | |
const origCandidatePossibleInCell = window.candidatePossibleInCell; | |
window.candidatePossibleInCell = function(n, cell, options) { | |
if (n < 1 || n > size) | |
return false; | |
if (!options) | |
options = {}; | |
if (!options.bruteForce && cell.value) | |
return cell.value === n; | |
// If there is no solution, then all candidates are possible | |
if (!window.grid.solution) { | |
return origCandidatePossibleInCell(n, cell, options); | |
} | |
// Cells that are set to 0 can be ignored | |
const solutionVal = window.grid.solution[cell.i * size + cell.j]; | |
if (solutionVal === 0) { | |
return origCandidatePossibleInCell(n, cell, options); | |
} | |
// Only use the solution if all cells are filled | |
var allFilled = true; | |
for (var i = 0; i < size; i++) { | |
for (var j = 0; j < size; j++) { | |
if (grid[i][j].value === 0) { | |
allFilled = false; | |
break; | |
} | |
} | |
} | |
if (!allFilled) { | |
return origCandidatePossibleInCell(n, cell, options); | |
} | |
// Check against the solution | |
return solutionVal === n; | |
} | |
if (window.sidebars && window.sidebars.length && window.boolConstraints) { | |
createSidebars(); | |
onInputEnd(); | |
} | |
if (window.boolConstraints) { | |
let prevButtons = buttons.splice(0, buttons.length); | |
window.onload(); | |
buttons.splice(0, buttons.length); | |
for (let i = 0; i < prevButtons.length; i++) { | |
buttons.push(prevButtons[i]); | |
} | |
} | |
} | |
let intervalId = setInterval(() => { | |
if (typeof exportPuzzle === 'undefined' || | |
typeof importPuzzle === 'undefined' || | |
typeof createSidebars === 'undefined' || | |
typeof drawPopups === 'undefined' || | |
typeof candidatePossibleInCell === 'undefined') { | |
return; | |
} | |
clearInterval(intervalId); | |
doShim(); | |
}, 16); | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment