
/g,">").replace(/"/g,""").replace(/'/g,"'")}let s={},c=[];function d(e,n,t){s={...s,dynamic:e,core:n,leinProject:t}}function p(){c=["wh","with-style"]}window.onload=()=>{let{dynamic:n,core:t,leinProject:r}=s;if(n&&(document.getElementById("code-dynamic").innerHTML=o(n)),t&&(document.getElementById("code-core").innerHTML=o(t)),r&&(document.getElementById("code-lein-project").innerHTML=o(r)),c){let e={wh:';; Copyright (c) 2016 Tyler Hobbs\n;; MIT License\n;; https://github.com/thobbs/genartlib/\n(defn h\n "Returns a given percentage of the height Quil-specific."\n ([] (h 1.0))\n ([percentage] (* (height) percentage)))\n\n;; Copyright (c) 2016 Tyler Hobbs\n;; MIT License\n;; https://github.com/thobbs/genartlib/\n(defn w\n "Returns a given percentage of the width. Quil-specific."\n ([] (w 1.0))\n ([percentage] (* (width) percentage)))\n',"with-style":"(defmacro with-style [& body]\n `(do\n (q/push-style)\n (try\n ~@body\n (finally (q/pop-style)))))\n"},n=c.map(n=>e[n]).join("\n");document.getElementById("code-helpers").innerHTML=o(n)}e(a).addPlugin((0,l.number)()),e(a).highlightAll()}}),l("3PmhV",function(e,t){n(e.exports,"number",()=>r);let r=e=>({"after:highlightElement":({el:e,result:n,text:t})=>{let r=n.value.split("\n");r.pop();let a=r.length.toString().length,i=r.map((e,n)=>{let t=(n+1).toString().padStart(a);return`${e}`}).join("\n");e.innerHTML=i}})}),i("d3dEA"),i("7Y8VX");var o=i("dZ8Um");(0,o.make)('(ns sketch.dynamic\n (:require ; [clojure.java.shell :refer [sh]]\n [genartlib.algebra :refer :all]\n ; [genartlib.curves :refer :all]\n ; [genartlib.geometry :refer :all]\n ; [genartlib.random :refer :all]\n [artlib.color.dictionary :as colors]\n [sketch.kdtree :as kd]\n [clojure.core.matrix :as mat]\n [artlib.quil.global :refer :all]\n [genartlib.util :refer [w h]]\n [quil.core :as q]))\n\n(def color-dict (colors/init))\n(defn rand-color []\n (let [palette (colors/get-combination-hsb color-dict :205)]\n (first (shuffle palette))))\n\n(defrecord Particle [pos vel acc col])\n\n(defn make-particles \n ([] (make-particles 0.01))\n ([step]\n (let [half-step (* step 0.5)\n xs (range half-step (+ 1 half-step) step)\n ys (range half-step (+ 1 half-step) step)\n positions (mapcat (fn [x] (map (fn [y] [x y]) ys)) ys)]\n (map (fn [pos] (Particle. pos [0 0] [0 0] (rand-color))) positions))))\n\n(defn wind-at \n "Queries wind force at the specified position at time t using \'perlin\' noise."\n [[x y] t]\n (let [rez 2.1\n n (q/noise (* rez x) (* rez y) (* t 0.001))\n theta (mod (* n 15) Math/TAU)\n strength 0.001]\n [(* strength (Math/cos theta))\n (* strength (Math/sin theta))]))\n\n(defn em-at\n "Queries an electromagnetic foce at the specified position."\n [[x y] kdtree]\n (let [strength -0.0001\n result (kd/nearest-neighbor kdtree [x y] 20)\n total (->> \n (kd/nearest-neighbor kdtree [x y] 20)\n (map :point)\n (filter (fn [[x2 y2]] (> (q/dist x y x2 y2) 1e-8))) ;; filter close by points (appx machine epsilon)\n (map (fn [[x2 y2]] \n (let [d (q/dist x y x2 y2) \n f (/ 0.01 (max Float/MIN_VALUE (* d d)))\n f (min 1.5 f)\n a (angle x y x2 y2)]\n [(* strength f (Math/cos a))\n (* strength f (Math/sin a))])))\n (apply mat/add))]\n total))\n\n(defn color-at\n "Queries an electromagnetic foce at the specified position."\n [[x y] kdtree]\n (let [strength 0.001\n result (kd/nearest-neighbor kdtree [x y] 50)\n total (->> \n (kd/nearest-neighbor kdtree [x y] 20)\n (map :point)\n (filter (fn [[x2 y2]] (> (q/dist x y x2 y2) 1e-8))) ;; filter close by points (appx machine epsilon)\n (map (fn [[x2 y2]] \n (let [d (q/dist x y x2 y2) \n f (/ 0.01 (max Float/MIN_VALUE (* d d)))\n f (min 0.5 f)\n a (angle x y x2 y2)]\n [(* strength f (Math/cos a))\n (* strength f (Math/sin a))])))\n (apply mat/add))]\n total))\n\n(defn in-bounds?\n "Returns whether a particle is in bounds"\n [{[x y] :pos}]\n (and \n (< (max x y) 1)\n (> (min x y) 0)))\n\n(defn update-physics [{pos :pos vel :vel acc :acc col :col} kdtree color-trees frame]\n (let [wind-force (wind-at pos frame)\n repulsive-force (mat/mul (em-at pos kdtree) 0.01)\n color-tree (last (first (filter #(= col (first %)) color-trees)))\n color-force (mat/mul (color-at pos color-tree) 0.01)\n time-step 0.1\n friction (mat/mul vel -0.1)\n total-acc (mat/add acc friction wind-force repulsive-force color-force)\n delta-vel (mat/mul total-acc time-step) ;; second term is time step\n new-vel (mat/add vel delta-vel)\n delta-pos (mat/mul new-vel time-step)\n new-pos (mat/add pos delta-pos)]\n (Particle. new-pos new-vel 0 col)))\n\n(def particles (atom [] #_(make-particles 0.1)))\n\n(defn gauss\n "Samples a single value from a Gaussian distribution with the given mean\n and variance"\n [mean variance]\n (+ mean (* variance (q/random-gaussian))))\n\n(defn draw [state]\n #_(q/background 0 0 0)\n\n (let [frame (:frame state)\n f (/ frame 300)]\n \n (let [x (gauss 0.5 0.003)\n y (gauss 0.5 0.003)]\n (count @particles)\n (swap! particles concat [(Particle. [x y] [0 0] [0 0] (rand-color))]))\n\n ;; render\n #_(with-style\n (q/no-stroke)\n (q/fill 0 0 100 0.01)\n (q/rect 0 0 (w 1) (h 1)))\n (with-style\n (q/stroke-weight (w 0.001))\n (doseq [{[x y] :pos [hue s b] :col} @particles]\n (q/stroke hue s b 0.1)\n (q/point (w x) (h y))))\n\n ;; update\n (let [points-for-tree (map #(with-meta (:pos %) {:original %}) @particles)\n kdtree (kd/build-tree points-for-tree)\n\n color-trees (->>\n @particles\n (map :col)\n distinct\n (map (fn [color]\n (let [points (filter #(mat/eq color (:col %)) @particles)\n points-for-tree (map #(with-meta (:pos %) {:original %}) points)\n kdtree (kd/build-tree points-for-tree)]\n [color kdtree]))))\n updated (->> \n @particles\n (map #(update-physics % kdtree color-trees frame))\n (filter in-bounds?))] \n (reset! particles updated))\n ))\n','(ns sketch.core\n (:require [artlib.quil.middleware :refer [animation-mode]]\n [clojure.core.matrix :refer [set-current-implementation]]\n [quil.middleware :as qm]\n [sketch.dynamic :as dynamic]\n [quil.core :as q]))\n\n(set-current-implementation :vectorz)\n(println)\n\n(q/defsketch example\n :title "genuary-2024-01"\n :setup (fn [] (do (q/background 0 0 0) {}))\n :draw dynamic/draw\n :update identity\n :size [1800 1800]\n :animation {:render? true :dirname "01/" :length 6000 :framerate 600}\n :middleware [qm/fun-mode animation-mode])\n\n','(defproject genuary-2024-01 "0.1.0-SNAPSHOT"\n :description "FIXME: write description"\n :url "https://example.com/FIXME"\n :license {:name "Apache License, Version 2.0"\n :url "https://www.apache.org/licenses/LICENSE-2.0.html"}\n :dependencies [[org.clojure/clojure "1.11.1"]\n [com.dedovic/artlib-core "0.0.8"]\n [genartlib/genartlib "0.1.21"]] ; utility functions\n :jvm-opts ["-Xms4000m" "-Xmx4000M" ; 4GB heap size\n "-server"\n "-Dsun.java2d.uiScale=1.0"] ; adjust scaling for high DPI displays\n :source-paths ["src/clj"]\n :java-source-paths ["src/java"]\n :resource-paths ["resources"]\n :aot :all)\n'),(0,o.withDefaultHelpers)();
dynamic.clj
core.clj
helpers
project.clj