Last active
December 5, 2020 00:40
-
-
Save Johann150/ac2372f8f79a11a0602562664ecdd4fa to your computer and use it in GitHub Desktop.
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
<html> | |
<head> | |
<style> | |
body{ | |
background:#333; | |
} | |
.toggle{ | |
display:none; | |
} | |
#fold{ | |
display:none; | |
padding:5px; | |
padding-left:20px; | |
background-color:#f8f888; | |
margin-bottom:10px; | |
} | |
#fold *{ | |
vertical-align: top; | |
} | |
#fold input,#fold label{ | |
float:right; | |
} | |
.toggle ~ label{ | |
/*poisiton:absolute;*/ | |
float:left; | |
cursor:zoom-in; | |
} | |
.toggle ~ label:before{ | |
content:'\25B6'; | |
width:20px; | |
display:inline-block; | |
} | |
.toggle:checked ~ #fold{ | |
display:block; | |
} | |
.toggle:checked ~ label{ | |
cursor:zoom-out; | |
} | |
.toggle:checked ~ label:before{ | |
content: '\25BC'; | |
} | |
</style> | |
</head> | |
<body> | |
<input type="checkbox" checked="true" class="toggle" id="t"/> | |
<label for="t"></label> | |
<table id="fold"> | |
<tr> | |
<td> | |
<textarea id="in" cols="120" rows="5">NICE JOB IF YOU FOUND THIS CONGRATS UVE DONE A THING GOOD WELL DONE U SILLY</textarea><br> | |
<input type="button" onclick="draw()" value="draw!"/> | |
<label for="spiral">spiral</label><input type="checkbox" id="spiral"/> | |
<label for="cipher">ciphered</label><input type="checkbox" checked id="cipher"/> | |
</td> | |
<td> | |
<ul> | |
<li>Unknown characters are displayed as purple.</li> | |
<li>Can be rendered as a spiral instead of stepped like a floppy.</li> | |
<li>Unciphered is ASCII with the leading zero cut off.</li> | |
</ul> | |
<a href="https://gist.github.com/Johann150/ac2372f8f79a11a0602562664ecdd4fa">this file as a gist</a> | |
<a href="https://www.reddit.com/r/codes/comments/k6o0oo/graphic_containing_a_secret_message/">from this reddit post by u/Avieasolia</a> | |
<p xmlns:dct="http://purl.org/dc/terms/" xmlns:cc="http://creativecommons.org/ns#" class="license-text">This work is licensed under <a rel="license" href="https://creativecommons.org/licenses/by-sa/4.0">CC BY-SA 4.0</a>. 🅭🛊⭵</p> | |
</td> | |
</tr> | |
</table> | |
<canvas id="c"></canvas> | |
<script> | |
draw(); | |
function draw(){ | |
let spiral = document.getElementById("spiral").checked; | |
let cipher = document.getElementById("cipher").checked; | |
let c = document.getElementById("c"); | |
c.width = c.height = 512; | |
let ctx = c.getContext("2d"); | |
let input = document.getElementById("in").value; | |
input=input.split("").map(e=>{ | |
if(cipher){ | |
switch(e.toUpperCase()){ | |
case " ": return "X"; | |
case "A": return "100000"; | |
case "B": return "101000"; | |
case "C": return "110000"; | |
case "D": return "110100"; | |
case "E": return "100100"; | |
case "F": return "111000"; | |
case "G": return "111100"; | |
case "H": return "101100"; | |
case "I": return "011000"; | |
case "J": return "011100"; | |
case "K": return "??????"; | |
case "L": return "101010"; | |
case "M": return "??????"; | |
case "N": return "110110"; | |
case "O": return "100110"; | |
case "P": return "??????"; | |
case "Q": return "??????"; | |
case "R": return "101110"; | |
case "S": return "011010"; | |
case "T": return "011110"; | |
case "U": return "100011"; | |
case "V": return "101011"; | |
case "W": return "011101"; | |
case "X": return "??????"; | |
case "Y": return "110111"; | |
case "Z": return "??????"; | |
default: return e; | |
} | |
}else{ | |
if(e==" "){ | |
return "X"; | |
} | |
return (e.charCodeAt(0)&0x3f).toString(2).padStart(6,"0"); | |
} | |
}).join(""); | |
ctx.lineWidth=17.6; | |
const step = 9.474/180*Math.PI; | |
let theta=-Math.PI/2; | |
for(let i=0;i<input.length;i++){ | |
let t_start = theta; | |
theta+=step; | |
let t_end = theta; | |
if(input[i]=="0"){ | |
ctx.strokeStyle="black"; | |
}else if(input[i]=="1"){ | |
ctx.strokeStyle="white"; | |
}else if(input[i]=="X"){ | |
ctx.strokeStyle="red"; | |
}else{ | |
ctx.strokeStyle="purple"; | |
} | |
let x=i/38; | |
if(!spiral){ | |
x=Math.floor(x); | |
} | |
ctx.beginPath(); | |
ctx.arc(256,256,256-(17/2)-(17*x),t_start,t_end); | |
ctx.stroke(); | |
} | |
} | |
</script> | |
</body> | |
</html> |
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
<html> | |
<body style="color:#fff;background:#333;"> | |
You will have to edit the source code of this file to change the image.<br> | |
<script> | |
// These configuration values are hard coded here. | |
// If you change the image, you might have to change these too. | |
const spiral = false; | |
const cipher = true; | |
const sectors = 38; | |
const track_width = 17; | |
</script> | |
<!-- | |
If you want to decode a different image, you have to change the value for src. | |
The image should be cropped to the right size. | |
You might have to change the configuration values above. | |
--> | |
<img id="img" style="display:none;" src=""/> | |
<canvas id="c"></canvas><br> | |
<pre id="out"><!-- This is where the decoded text will go. --></pre> | |
<script> | |
let c = document.getElementById("c"); | |
// automatically resize the canvas to the image size | |
c.width=document.getElementById("img").width; | |
c.height=document.getElementById("img").height; | |
let ctx = c.getContext("2d"); | |
let step = 0; // keeps track of the currently read sector and track | |
let sector = 0; // for putting the spaces in the right places | |
draw(); | |
function draw(){ | |
// reset with image | |
ctx.clearRect(0,0,c.width,c.height); | |
ctx.drawImage(document.getElementById("img"),0,0,c.width,c.height); | |
// calculate angle for next sector | |
let angle = ((step%sectors)+0.5)*(360/sectors); | |
angle = angle/180*Math.PI; | |
let s = step/sectors; | |
if(!spiral){ | |
s=Math.floor(s); | |
} | |
let scale_x = c.width/2-(0.5+s)*track_width; | |
let scale_y = c.height/2-(0.5+s)*track_width; | |
let x = (c.width/2)+Math.sin(angle)*scale_x; | |
let y = (c.height/2)-Math.cos(angle)*scale_y; | |
let pixel = ctx.getImageData(x,y,1,1).data; | |
// detect end by transparent pixel | |
if(pixel[3]==0){ | |
// decipher the message | |
after(); | |
return; | |
} | |
// black and white must be exactly right for this to work | |
if(255==pixel[0]&&255==pixel[1]&&255==pixel[2]){ | |
pixel="1"; | |
sector++; | |
if(sector==6){ | |
pixel+=" "; | |
sector=0; | |
} | |
}else if(0==pixel[0]&&0==pixel[1]&&0==pixel[2]){ | |
pixel="0"; | |
sector++; | |
if(sector==6){ | |
pixel+=" "; | |
sector=0; | |
} | |
}else{ | |
// any other color is taken to be a space | |
sector = 0; | |
pixel="X "; | |
} | |
// add this letter to the output | |
document.getElementById("out").innerText+=pixel; | |
// highlight what was just read | |
ctx.strokeStyle="green"; | |
ctx.lineWidth=2; | |
ctx.beginPath(); | |
ctx.arc(x,y,track_width/2-ctx.lineWidth,0,Math.PI*2); | |
ctx.stroke(); | |
step++; | |
setTimeout(draw, 5); | |
} | |
function after(){ | |
let input=document.getElementById("out").innerText; | |
input=input.split(" ").map(e=>{ | |
if(cipher){ | |
switch(e){ | |
case "X": return " "; | |
case "100000": return "A"; | |
case "101000": return "B"; | |
case "110000": return "C"; | |
case "110100": return "D"; | |
case "100100": return "E"; | |
case "111000": return "F"; | |
case "111100": return "G"; | |
case "101100": return "H"; | |
case "011000": return "I"; | |
case "011100": return "J"; | |
// case "??????": return "K"; | |
case "101010": return "L"; | |
// case "??????": return "M"; | |
case "110110": return "N"; | |
case "100110": return "O"; | |
// case "??????": return "P"; | |
// case "??????": return "Q"; | |
case "101110": return "R"; | |
case "011010": return "S"; | |
case "011110": return "T"; | |
case "100011": return "U"; | |
case "101011": return "V"; | |
case "011101": return "W"; | |
// case "??????": return "X"; | |
case "110111": return "Y"; | |
// case "??????": return "Z"; | |
default: return e; | |
} | |
}else{ | |
if(e=="X"){ | |
return " "; | |
} | |
return String.fromCharCode(parseInt("1"+e,2)); | |
} | |
}).join(""); | |
// append deciphered text to output | |
document.getElementById("out").innerHTML+="\n\n"+input; | |
} | |
</script> | |
</body> | |
</html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment