mirror of
https://github.com/osmarks/random-stuff
synced 2024-11-17 01:14:48 +00:00
fractalart things
This commit is contained in:
parent
c65a55be67
commit
d8ddcc6c3e
@ -6,7 +6,7 @@ edition = "2018"
|
||||
|
||||
[dependencies]
|
||||
image = { version = "0.23", features = ["png"], default-features = false }
|
||||
nanorand = { version = "0.5", features = ["std", "rdseed", "wyrand", "tls"], default-features = false }
|
||||
nanorand = { version = "0.5", features = ["std", "rdseed", "wyrand"], default-features = false }
|
||||
hsl = "0.1.1"
|
||||
gumdrop = "0.8"
|
||||
smallvec = "1.4.1"
|
@ -45,7 +45,7 @@ fn mod_channel(rng: &mut WyRand, range: u16, n: u16) -> u16 {
|
||||
let rand: u16 = rng.generate::<u16>() % (range * 2 + 2);
|
||||
let o = ((n as u32) + (rand as u32)).saturating_sub(range as u32);
|
||||
// avoid weird artifacts - just directly using `as` truncates it, i.e. drops the high bytes, which leads to integer-overflow-like issues
|
||||
min(o, 65535) as u16
|
||||
max(min(o, 65535), 1) as u16
|
||||
}
|
||||
|
||||
// randomly adjust all the channels in a color by `range`
|
||||
|
157
fractalart.html
Normal file
157
fractalart.html
Normal file
@ -0,0 +1,157 @@
|
||||
<!DOCTYPE html>
|
||||
<title>FractalArt</title>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<style>
|
||||
.controls input, .controls button { border: 1px solid black; padding: 0.2em; }
|
||||
#fractalart { display: flex; align-items: flex-start; flex-wrap: wrap; }
|
||||
#status { font-style: italic; }
|
||||
#info { padding-top: 0.5em; font-size: 0.8em; }
|
||||
#out { image-rendering: crisp-edges; }
|
||||
</style>
|
||||
<div id="fractalart">
|
||||
<canvas id="out" width=512 height=512></canvas>
|
||||
<table class="controls">
|
||||
<tr><td>Lightness:</td><td><input id="lightness" type="number" max="1" min="0" step="0.01" value="0.6"></td></tr>
|
||||
<tr><td>Saturation:</td><td><input id="saturation" type="number" max="1" min="0" step="0.01" value="1.0"></td></tr>
|
||||
<tr><td>Hue:</td><td><input id="hue" type="number" max="360" min="0" step="1" value="0"></td><td>(or random: <input type="checkbox" checked id="hue-random">)</td></tr>
|
||||
<tr><td>Variance:</td><td><input id="variance" type="number" min="0" step="0.25" value="4"></td></tr>
|
||||
<tr><td>Bias:</td><td><input id="bias" type="number" step="0.25" value="0"></td></td></tr>
|
||||
<tr><td>Render Size:</td><td><input id="width" type="number" step="16" value="512"></td></td></tr>
|
||||
<tr><td><button id="generate">Generate</button></td></tr>
|
||||
<tr><td colspan="4" id="status"></td></tr>
|
||||
<tr><td colspan="4"><div id="info">
|
||||
<a href="https://github.com/TomSmeets/FractalArt/">FractalArt</a> for JS.<br>
|
||||
If this is too slow try my <a href="https://github.com/osmarks/random-stuff/tree/master/fractalart-rs">Rust port</a>.<br>
|
||||
This is all contained in one HTML file, so you can save it locally if you want.
|
||||
</div></td></tr>
|
||||
</table>
|
||||
</div>
|
||||
<script id="fractalart-code">
|
||||
// harvested from github somewhere
|
||||
function hslToRgb(h, s, l) {
|
||||
var r, g, b;
|
||||
|
||||
if (s == 0) {
|
||||
r = g = b = l; // achromatic
|
||||
} else {
|
||||
function hue2rgb(p, q, t) {
|
||||
if (t < 0) t += 1;
|
||||
if (t > 1) t -= 1;
|
||||
if (t < 1/6) return p + (q - p) * 6 * t;
|
||||
if (t < 1/2) return q;
|
||||
if (t < 2/3) return p + (q - p) * (2/3 - t) * 6;
|
||||
return p;
|
||||
}
|
||||
|
||||
var q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
||||
var p = 2 * l - q;
|
||||
|
||||
r = hue2rgb(p, q, h + 1/3);
|
||||
g = hue2rgb(p, q, h);
|
||||
b = hue2rgb(p, q, h - 1/3);
|
||||
}
|
||||
|
||||
return [ r * 255, g * 255, b * 255 ];
|
||||
}
|
||||
const randRange = x => Math.floor(Math.random() * x)
|
||||
|
||||
// main fractalart logic
|
||||
// todo: WASM (there is already a perfectly good Rust version...) or just actually optimize this
|
||||
const run = (im, { hue, saturation, lightness, variance, bias }) => {
|
||||
const set = (x, y, offset, value) => { im.data[(y * im.width + x) * 4 + offset] = value }
|
||||
const get = (x, y, offset) => im.data[(y * im.width + x) * 4 + offset]
|
||||
const boundcheck = (x, y) => x >= 0 && y >= 0 && x < im.width && y < im.height
|
||||
const ringAt = (x, y, l) => {
|
||||
let out = []
|
||||
for (let n = -l; n <= l; n++) {
|
||||
out.push([n + x, l + y])
|
||||
out.push([n + x, -l + y])
|
||||
}
|
||||
for (let n = 1 - l; n <= l; n++) {
|
||||
out.push([l + x, n + y])
|
||||
out.push([-l + x, n + y])
|
||||
}
|
||||
return out.filter(([x, y]) => boundcheck(x, y))
|
||||
}
|
||||
const modColorChannel = x => Math.min(Math.max(Math.floor(x + (Math.random() * (variance * 2 + 1)) - variance + bias), 0), 255)
|
||||
|
||||
const startX = randRange(im.width)
|
||||
const startY = randRange(im.height)
|
||||
const iterations = Math.max(startX, im.width - 1 - startX, startY, im.height - 1 - startY)
|
||||
const [sR, sG, sB] = hslToRgb(hue, saturation, lightness)
|
||||
set(startX, startY, 0, Math.floor(sR))
|
||||
set(startX, startY, 1, Math.floor(sG))
|
||||
set(startX, startY, 2, Math.floor(sB))
|
||||
set(startX, startY, 3, 255)
|
||||
for (let iteration = 1; iteration <= iterations; iteration++) {
|
||||
for (const [x, y] of ringAt(startX, startY, iteration)) {
|
||||
const possibleColors = []
|
||||
for (const [ax, ay] of ringAt(x, y, 1)) {
|
||||
if (get(ax, ay, 3)) {
|
||||
possibleColors.push([get(ax, ay, 0), get(ax, ay, 1), get(ax, ay, 2)])
|
||||
}
|
||||
}
|
||||
|
||||
const [chosenR, chosenG, chosenB] = possibleColors[randRange(possibleColors.length)]
|
||||
set(x, y, 0, modColorChannel(chosenR))
|
||||
set(x, y, 1, modColorChannel(chosenG))
|
||||
set(x, y, 2, modColorChannel(chosenB))
|
||||
set(x, y, 3, 255)
|
||||
}
|
||||
}
|
||||
|
||||
postMessage(im)
|
||||
}
|
||||
|
||||
const readInput = name => document.getElementById(name).value
|
||||
|
||||
// detect if not running on webworker
|
||||
if ("document" in self) {
|
||||
const statusDisplay = document.querySelector("#status")
|
||||
let status = "Idle"
|
||||
let startTime = null
|
||||
const updateStatus = s => {
|
||||
if (statusDisplay.firstChild) { statusDisplay.removeChild(statusDisplay.firstChild) }
|
||||
statusDisplay.appendChild(document.createTextNode(s))
|
||||
status = s
|
||||
}
|
||||
const canv = document.querySelector("#out")
|
||||
// TODO things
|
||||
canv.style.width = canv.style.height = Math.min(window.innerHeight - 16, window.innerWidth - 16) + "px"
|
||||
const ctx = canv.getContext("2d")
|
||||
const source = `
|
||||
${document.querySelector("#fractalart-code").innerHTML}
|
||||
onmessage = e => run(e.data[0], e.data[1])
|
||||
`
|
||||
// be nice to foolish single threaded browsers
|
||||
// offload to worker thread
|
||||
const worker = new Worker(URL.createObjectURL(new Blob([source], {type: 'application/javascript'})))
|
||||
worker.onmessage = e => {
|
||||
const im = e.data
|
||||
canv.width = canv.height = im.width
|
||||
ctx.putImageData(im, 0, 0)
|
||||
updateStatus(`Idle (${Date.now() - startTime}ms)`)
|
||||
}
|
||||
worker.onerror = e => {
|
||||
console.warn(e)
|
||||
updateStatus(e.message)
|
||||
}
|
||||
const generate = () => {
|
||||
if (status !== "Working") {
|
||||
updateStatus("Working")
|
||||
startTime = Date.now()
|
||||
const size = parseInt(readInput("width"))
|
||||
worker.postMessage([ctx.createImageData(size, size), {
|
||||
hue: document.getElementById("hue-random").checked ? Math.random() : parseFloat(readInput("hue")),
|
||||
saturation: parseFloat(readInput("saturation")),
|
||||
lightness: parseFloat(readInput("lightness")),
|
||||
variance: parseFloat(readInput("variance")),
|
||||
bias: parseFloat(readInput("bias"))
|
||||
}])
|
||||
}
|
||||
}
|
||||
document.querySelector("#generate").addEventListener("click", generate)
|
||||
generate()
|
||||
}
|
||||
</script>
|
14
spudnet-http.py
Normal file
14
spudnet-http.py
Normal file
@ -0,0 +1,14 @@
|
||||
import urllib3, json
|
||||
|
||||
http = urllib3.PoolManager()
|
||||
def send(x):
|
||||
http.request("POST", "https://spudnet.osmarks.net/httponly", body=json.dumps({"mode": "send", "channel": "potatOS", "message": x}), headers={"Content-Type": "application/json"})
|
||||
while True:
|
||||
r = http.request("POST", "https://spudnet.osmarks.net/httponly", body=json.dumps({"mode": "recv", "channel": "potatOS", "timeout": 30000}), headers={"Content-Type": "application/json"})
|
||||
data = json.loads(r.data)
|
||||
if data["result"] != None:
|
||||
res = data["result"]["data"]
|
||||
try:
|
||||
send(repr(eval(res)))
|
||||
except Exception as e:
|
||||
send(repr(e))
|
Loading…
Reference in New Issue
Block a user