|
4 | 4 | <meta charset="UTF-8" />
|
5 | 5 | <meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
6 | 6 | <title>Generative Art with Canvas</title>
|
| 7 | + <style> |
| 8 | + html, |
| 9 | + body { |
| 10 | + height: 100%; |
| 11 | + margin: 0; |
| 12 | + overflow: hidden; /* Prevent scrollbars */ |
| 13 | + } |
| 14 | + canvas { |
| 15 | + display: block; /* Remove the default margin of the canvas */ |
| 16 | + } |
| 17 | + </style> |
7 | 18 | </head>
|
8 | 19 | <body>
|
9 | 20 | <canvas id="destinationCanvas" width="400" height="400"></canvas>
|
10 |
| - <button onclick="generateArt()">Generate Art</button> |
11 | 21 | <script>
|
12 | 22 | const words = [
|
13 | 23 | "austin",
|
|
24 | 34 | const ctx = canvas.getContext("2d");
|
25 | 35 |
|
26 | 36 | async function generateRandomSeed() {
|
| 37 | + // read query string search param to get seed from URL?id=123 |
| 38 | + const urlParams = new URLSearchParams(window.location.search); |
| 39 | + const seed = urlParams.get("id"); |
| 40 | + if (seed) { |
| 41 | + return seed; |
| 42 | + } |
27 | 43 | const array = window.crypto.getRandomValues(new Uint8Array(32));
|
28 | 44 | const hashBuffer = await crypto.subtle.digest("SHA-256", array);
|
29 | 45 | const hashArray = Array.from(new Uint8Array(hashBuffer));
|
|
36 | 52 | return [color1, color2];
|
37 | 53 | }
|
38 | 54 |
|
39 |
| - function drawZigZag(ctx, width, height) { |
40 |
| - const zigZagWidth = width / 10; |
41 |
| - ctx.beginPath(); |
42 |
| - ctx.moveTo(0, 0); |
43 |
| - let x = 0, |
44 |
| - y = 0; |
45 |
| - while (x < width && y < height) { |
46 |
| - x += zigZagWidth; |
47 |
| - y = y === 0 ? height / 2 : 0; |
48 |
| - ctx.lineTo(x, y); |
49 |
| - } |
50 |
| - ctx.strokeStyle = "black"; |
51 |
| - ctx.lineWidth = 2; |
52 |
| - ctx.stroke(); |
53 |
| - ctx.closePath(); |
54 |
| - } |
55 |
| - |
56 |
| - function fillZigZag(ctx, width, height, colors) { |
| 55 | + function fillRects(ctx, width, height, colors) { |
57 | 56 | const [color1, color2] = colors;
|
58 | 57 | ctx.fillStyle = color1;
|
59 | 58 | ctx.fillRect(0, 0, width / 2, height);
|
60 | 59 | ctx.fillStyle = color2;
|
61 | 60 | ctx.fillRect(width / 2, 0, width / 2, height);
|
62 | 61 | }
|
63 | 62 |
|
64 |
| - function getRandomPosition(seed, width, height) { |
65 |
| - const x = parseInt(seed.slice(0, 2), 16) % width; |
66 |
| - const y = parseInt(seed.slice(2, 4), 16) % height; |
67 |
| - return [x, y]; |
68 |
| - } |
69 |
| - |
70 |
| - function drawText(ctx, width, height, words, seed) { |
71 |
| - ctx.font = "20px Arial"; |
72 |
| - ctx.fillStyle = "black"; |
73 |
| - const positions = []; |
74 |
| - const maxAttempts = 100; // Maximum attempts to find a non-overlapping position |
75 |
| - const cornerDeadZone = 50; // Distance from corners where no text is placed |
76 |
| - |
77 |
| - function shuffleArray(array) { |
78 |
| - for (let i = array.length - 1; i > 0; i--) { |
79 |
| - const j = Math.floor(Math.random() * (i + 1)); |
80 |
| - [array[i], array[j]] = [array[j], array[i]]; // Swap elements |
81 |
| - } |
82 |
| - return array; |
83 |
| - } |
84 |
| - shuffleArray([...words]) |
85 |
| - .slice(0, Math.random() < 0.5 ? 2 : 3) |
86 |
| - .forEach((word) => { |
87 |
| - let attempts = 0; |
88 |
| - let [x, y] = getRandomPosition(seed + word, width, height); |
89 |
| - |
90 |
| - // Adjust position if too close to a corner |
91 |
| - x = Math.max(x, cornerDeadZone); |
92 |
| - y = Math.max(y, cornerDeadZone); |
93 |
| - x = Math.min(x, width - cornerDeadZone); |
94 |
| - y = Math.min(y, height - cornerDeadZone); |
95 |
| - |
96 |
| - while ( |
97 |
| - positions.some( |
98 |
| - ([px, py]) => Math.abs(px - x) < 50 && Math.abs(py - y) < 20 |
99 |
| - ) && |
100 |
| - attempts < maxAttempts |
101 |
| - ) { |
102 |
| - [x, y] = getRandomPosition( |
103 |
| - seed + word + Math.random().toString(16).slice(2), |
104 |
| - width, |
105 |
| - height |
106 |
| - ); |
107 |
| - // Adjust for corner dead zone again after repositioning |
108 |
| - x = Math.max(x, cornerDeadZone); |
109 |
| - y = Math.max(y, cornerDeadZone); |
110 |
| - x = Math.min(x, width - cornerDeadZone); |
111 |
| - y = Math.min(y, height - cornerDeadZone); |
112 |
| - attempts++; |
113 |
| - } |
114 |
| - |
115 |
| - if (attempts < maxAttempts) { |
116 |
| - positions.push([x, y]); |
117 |
| - |
118 |
| - if (Math.random() > 0.5) { |
119 |
| - ctx.fillText(word, x, y); |
120 |
| - } else { |
121 |
| - ctx.save(); |
122 |
| - ctx.translate(x, y); |
123 |
| - ctx.rotate(Math.PI / 2); |
124 |
| - ctx.fillText(word, 0, 0); |
125 |
| - ctx.restore(); |
126 |
| - } |
127 |
| - } else { |
128 |
| - // Handle the case where a suitable position wasn't found |
129 |
| - console.log( |
130 |
| - `Could not place '${word}' after ${maxAttempts} attempts.` |
131 |
| - ); |
132 |
| - } |
133 |
| - }); |
| 63 | + let seedPromise = generateRandomSeed(); |
| 64 | + function resizeCanvas() { |
| 65 | + canvas.width = window.innerWidth; |
| 66 | + canvas.height = window.innerHeight; |
| 67 | + generateArt(); // Regenerate the art to fit the new canvas size |
134 | 68 | }
|
135 | 69 |
|
136 | 70 | async function generateArt() {
|
137 |
| - const seed = await generateRandomSeed(); |
| 71 | + const seed = await seedPromise; |
138 | 72 | const [color1, color2] = getRandomColorPair(seed);
|
139 | 73 | ctx.clearRect(0, 0, canvas.width, canvas.height);
|
140 | 74 |
|
141 | 75 | // Draw zigzag and fill with colors
|
142 |
| - drawZigZag(ctx, canvas.width, canvas.height); |
143 |
| - fillZigZag(ctx, canvas.width, canvas.height, [color1, color2]); |
144 |
| - |
145 |
| - // Draw text |
146 |
| - drawText(ctx, canvas.width, canvas.height, words, seed); |
147 |
| - |
148 |
| - // Export canvas to PNG |
149 |
| - const dataURL = canvas.toDataURL("image/png"); |
150 |
| - const link = document.createElement("a"); |
151 |
| - link.download = "generated_art.png"; |
152 |
| - link.href = dataURL; |
153 |
| - link.click(); |
| 76 | + fillRects(ctx, canvas.width, canvas.height, [color1, color2]); |
154 | 77 | }
|
| 78 | + window.addEventListener("load", resizeCanvas); |
| 79 | + window.addEventListener("resize", resizeCanvas); |
155 | 80 | </script>
|
156 | 81 | </body>
|
157 | 82 | </html>
|
0 commit comments