Skip to content

Instantly share code, notes, and snippets.

@Johann150
Last active December 5, 2020 00:40
Show Gist options
  • Save Johann150/ac2372f8f79a11a0602562664ecdd4fa to your computer and use it in GitHub Desktop.
Save Johann150/ac2372f8f79a11a0602562664ecdd4fa to your computer and use it in GitHub Desktop.
<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>. &#x1F16D;&#x1f6ca;&#x2b75;</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>
<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