diff src/laser/rasterize.clj @ 15:8ad629298649

major refactoring
author Robert McIntyre <rlm@mit.edu>
date Sun, 29 Aug 2010 19:02:26 -0400
parents db745c95aabd
children 52f544d05414
line wrap: on
line diff
     1.1 --- a/src/laser/rasterize.clj	Sun Aug 29 00:09:28 2010 -0400
     1.2 +++ b/src/laser/rasterize.clj	Sun Aug 29 19:02:26 2010 -0400
     1.3 @@ -1,107 +1,50 @@
     1.4 -(ns laser.rasterize)
     1.5 +(ns laser.rasterize
     1.6 +  (:use [rlm
     1.7 +	 image-utils
     1.8 +	 map-utils]
     1.9 +	[clojure.contrib
    1.10 +	 [str-utils :only [str-join re-gsub]]
    1.11 +	 [seq :only [indexed]]
    1.12 +	 [math]])
    1.13 +  (:import [ij ImagePlus IJ]))
    1.14  
    1.15 -(import '(java.io File))
    1.16 -(import '(org.apache.commons.io FileUtils))
    1.17 -(import '(javax.imageio ImageIO) )
    1.18 -(import '(javax.swing JFrame))
    1.19 -(import '(java.awt Color BorderLayout))
    1.20 -(import '(ij ImagePlus IJ))
    1.21 -(import '(java.lang Math))
    1.22 -(import '(java.awt Graphics2D Panel))
    1.23 -(import '(ij Macro))
    1.24 +;(import '(java.io File))
    1.25 +;(import '(org.apache.commons.io FileUtils))
    1.26 +;(import '(javax.imageio ImageIO) )
    1.27  
    1.28 -(import '(java.io BufferedReader InputStreamReader))
    1.29 -(import '(java.awt.image BufferedImage))
    1.30  
    1.31 -;(use 'clojure.contrib.str-utils)
    1.32 -;(use 'clojure.contrib.seq-utils)
    1.33 -;(use 'clojure.contrib.combinatorics)
    1.34 -;(use 'clojure.contrib.duck-streams)
    1.35 -
    1.36 -;(use 'clojure.contrib.repl-utils)
    1.37 -
    1.38 -;(set! *print-length* 20)
    1.39 -
    1.40 -
    1.41 -
    1.42 -
    1.43 +(set! *print-length* 20)
    1.44  (def feed 120)
    1.45  (def  dpi [500, 500])
    1.46  
    1.47  
    1.48 +;;; this process is divided into two tasks,
    1.49 +;;; creating the raster g-code, which sweeps back and forth
    1.50 +;;; and creating the gmask, which turns the laser on and off.
    1.51  
    1.52  
    1.53 -(defn preserve-meta [f]
    1.54 -  (fn [& x] (with-meta
    1.55 -	      (apply f x)
    1.56 -	      (meta (last x)))))
    1.57 +;;; we'll be using frame-hashes, which represent picutres as
    1.58 +;;; a 3D vector field over 2D space, with the vectors representing
    1.59 +;;; the rgb values at that particulat point. 
    1.60  
    1.61 -(defmulti frame-hash-multi  class)
    1.62 +(defn select-row
    1.63 +  "returns a frame hash that is just a single line at the chosen y"
    1.64 +  [y window]
    1.65 +  (filter-keys (comp (partial = y) last) window))
    1.66  
    1.67 +(defn make-rows [pic]
    1.68 +  (map (partial sort #(< (first %1) (first %2)))
    1.69 +       (partition-by last
    1.70 +		     (sort (fn [[x1 y1][x2 y2]] (> y2 y1))
    1.71 +			   (map first (filter-vals (partial = black) pic))))))
    1.72  
    1.73 -(defmethod frame-hash-multi ImagePlus
    1.74 -  [image+]
    1.75 -    (with-meta
    1.76 -      (let [buf (.. image+ getBufferedImage)
    1.77 -	    color (.getColorModel buf)]
    1.78 -	(apply hash-map 
    1.79 -	       (interleave 
    1.80 -		(doall (for [x (range (.getWidth image+)) y (range (.getHeight image+))]
    1.81 -			 (vector x y)))
    1.82 -		(doall (for [x (range (.getWidth image+)) y (range (.getHeight image+))]
    1.83 -			 (let [data (.getRGB buf x y)] 
    1.84 -			   (hash-map :r (bit-shift-right (bit-and 0xff0000 data) 16)
    1.85 -				     :g (bit-shift-right (bit-and 0x00ff00 data) 8)
    1.86 -				     :b (bit-and 0x0000ff data))))))))
    1.87 -      {:width  (.getWidth image+) :height (.getHeight image+)}))
    1.88  
    1.89 -
    1.90 -(defmethod frame-hash-multi String
    1.91 -  [image-name]
    1.92 -  (let [image+ (ImagePlus. image-name)]
    1.93 -    (frame-hash-multi image+)))
    1.94 -
    1.95 -
    1.96 -(defn frame-hash
    1.97 -  "yields a convienent representation for the pixles in an image.
    1.98 -   Because of the size of the structvre generated, this must only be used
    1.99 -   in a transient way so that java can do it's garbage collection."
   1.100 -  [something]
   1.101 -  (frame-hash-multi something))
   1.102 -
   1.103 -;(def frame-hash (preserve-meta frame-hash))
   1.104 - 
   1.105 -
   1.106 -
   1.107 -
   1.108 -(def white {:r 255, :g 255, :b 255})
   1.109 -(def black {:r 0,   :g 0,   :b 0})
   1.110 -
   1.111 -
   1.112 -
   1.113 -(defn rgb-euclidian
   1.114 -  [{r1 :r g1 :g b1 :b} {r2 :r g2 :g b2 :b} ]
   1.115 -  (expt (+ (expt (- r1 r2) 2)
   1.116 -	  (expt (- g1 g2) 2)
   1.117 -	  (expt (- b1 b2) 2)) 0.5))
   1.118 -
   1.119 -(defn b&w
   1.120 -  "turn everything strictly black or white"
   1.121 -  [window]
   1.122 -  (with-meta
   1.123 -  (zipmap
   1.124 -   (keys window)
   1.125 -   (map (fn  [rgb] 
   1.126 -	  (if (> (rgb-euclidian rgb white) (rgb-euclidian rgb black))
   1.127 -	    black white))
   1.128 -	(vals window))) (meta window)))
   1.129 -
   1.130 -
   1.131 +;;; generate rastering g-code
   1.132  
   1.133  (defn raster-preamble []
   1.134    (str-join \newline
   1.135  	    ["M63 P0\nG61"
   1.136 -	    (str \F feed)
   1.137 +	    (str "F" feed)
   1.138  	    "M101"
   1.139  	    "M3 S1\n"]))
   1.140  
   1.141 @@ -111,68 +54,9 @@
   1.142  	     "M5"
   1.143  	     "M2\n"]))
   1.144  
   1.145 -
   1.146  (defn raster-comment [string]
   1.147    (str "(" (re-gsub #"[()]" "" string) ")"))
   1.148  
   1.149 -(defn filter-keys [fun m]
   1.150 -  (select-keys m (filter fun (keys m))))
   1.151 -
   1.152 -(def filter-keys (preserve-meta filter-keys))
   1.153 -
   1.154 -(defn filter-vals [fun m]
   1.155 -    (into {} (filter (comp fun val) m)))
   1.156 -
   1.157 -(def filter-vals (preserve-meta filter-vals))
   1.158 -
   1.159 -(defn frame-hash->bufferedImage
   1.160 -  [frame-hash]
   1.161 -    (let [data  (meta frame-hash)
   1.162 -	image (BufferedImage. (:width data) (:height data) BufferedImage/TYPE_INT_BGR)]
   1.163 -	
   1.164 -	(doall (for [element frame-hash] 
   1.165 -	  (let [coord  (key element) 
   1.166 -		rgb   (val element)
   1.167 -		packed-RGB 
   1.168 -		(+ (bit-shift-left (:r rgb) 16)
   1.169 -		   (bit-shift-left (:g rgb) 8)
   1.170 -		                   (:b rgb))]
   1.171 -	    (.setRGB image (first coord) (last coord) packed-RGB))))
   1.172 -	image))
   1.173 -  
   1.174 -(defmulti  display "Creates a JFrame and displays a buffered image"  class)
   1.175 -
   1.176 -(defn- makePanel [image] (proxy [Panel] [] (paint [g]  (.drawImage g image 0 0 nil))))
   1.177 -
   1.178 -(defn select-row [x window]
   1.179 -  (filter-keys (comp (partial = x) first) window))
   1.180 -
   1.181 -
   1.182 -
   1.183 -(defmethod display 
   1.184 -  BufferedImage  [image] 
   1.185 -  (let [panel (makePanel image)
   1.186 -	frame (JFrame. "Oh Yeah!")]
   1.187 -    (.add frame panel) 
   1.188 -    (.pack frame) 
   1.189 -    (.setVisible frame true ) 
   1.190 -    (.setSize frame(.getWidth image) (.getHeight image))))
   1.191 - 
   1.192 -(defmethod display
   1.193 -  ImagePlus [image]
   1.194 -  (display (.getBufferedImage image)))
   1.195 -
   1.196 -(defmethod display
   1.197 -  clojure.lang.PersistentHashMap [frame-hash]
   1.198 -  (display (frame-hash->bufferedImage frame-hash)))
   1.199 -
   1.200 -(defmethod display
   1.201 -    clojure.lang.PersistentArrayMap [frame-hash]
   1.202 -    (display (frame-hash->bufferedImage frame-hash)))
   1.203 -
   1.204 -
   1.205 -
   1.206 -
   1.207  
   1.208  ;this is a sequence of rows
   1.209  
   1.210 @@ -202,6 +86,34 @@
   1.211  		 (float (* x2 (/ x-dpi)))
   1.212  		 (float (* y2 (/ y-dpi)))))))
   1.213  
   1.214 +
   1.215 +(defn generate-gcode [pic]
   1.216 +  (str (raster-preamble)
   1.217 +       (str-join ""
   1.218 +		 (map
   1.219 +		  (fn [[index row]]
   1.220 +		    (row->gcode dpi (even? index) row))
   1.221 +		    (indexed (make-rows pic))))
   1.222 +		 (raster-epilogue)))
   1.223 +
   1.224 +
   1.225 +
   1.226 +
   1.227 +
   1.228 +
   1.229 +
   1.230 +
   1.231 +
   1.232 +
   1.233 +
   1.234 +
   1.235 +
   1.236 +
   1.237 +
   1.238 +
   1.239 +
   1.240 +
   1.241 +
   1.242  (defn gather-row [row]
   1.243    (let [base    [[(first (first row)) (first (first row))]]]
   1.244  					;  (println base)
   1.245 @@ -228,11 +140,9 @@
   1.246  
   1.247  
   1.248  (defn row->gmask [[x-dpi y-dpi] forward? row]
   1.249 -;  (println forward?)
   1.250    (let [start (float (* (/ x-dpi) (first (first
   1.251  					  (if forward?
   1.252  					  (reverse row) row)))))]
   1.253 -
   1.254      (let [preamble (if-not forward?
   1.255  		     (format "0 0 0 %.3f\n" start)
   1.256  		     (format "0 0 1 %.3f\n" start))
   1.257 @@ -242,7 +152,7 @@
   1.258  		  (reverse (gather-row row))
   1.259  		  (gather-row row))]
   1.260  	    (let [x (float (* x (/ x-dpi)))
   1.261 -		  y (float (* y (/ x-dpi)))
   1.262 +		  y (float (* y (/ x-dpi)))]
   1.263  ;;		  x (+ x 0.159)];; shift by a small margin.
   1.264  	      (if-not forward?
   1.265  		(str (format "0 0 1 %.3f\n" x)
   1.266 @@ -254,144 +164,45 @@
   1.267        (str preamble (str-join "" body)))))
   1.268  
   1.269  
   1.270 -
   1.271 -(defn make-rows [pic]
   1.272 -  (map (partial sort #(< (first %1) (first %2)))
   1.273 -       (partition-by last
   1.274 -		     (sort (fn [[x1 y1][x2 y2]] (> y2 y1))
   1.275 -			   (map first (filter-vals (partial = black) pic))))))
   1.276 -
   1.277 -
   1.278 -
   1.279  (defn generate-gmask [pic]
   1.280 -
   1.281    (str "1 0 0 0\n"
   1.282         (str-join "" (map (fn [[index row]]
   1.283  			   (row->gmask dpi (even? index) row))
   1.284  			 (indexed (make-rows pic))))))
   1.285  
   1.286  
   1.287 -;; 1 0 0 0
   1.288 -;; 0 0 1 2.881
   1.289 -;; 0 0 0 2.881
   1.290 -;; 0 1 0 2.863
   1.291 -;; 0 0 0 2.769
   1.292 -;; 0 1 0 2.751
   1.293 -;; 0 0 0 2.729
   1.294 -;; 0 1 0 2.617
   1.295 -;; 0 0 0 2.593
   1.296 -;; 0 1 0 2.561
   1.297 -;; 0 0 0 2.463
   1.298 -;; 0 1 0 2.445
   1.299 -;; 0 0 0 2.385
   1.300 -;; 0 1 0 2.317
   1.301 -;; 0 0 0 2.253
   1.302 -;; 0 1 0 2.233
   1.303 -;; 0 0 0 2.177
   1.304 -
   1.305 -
   1.306 -
   1.307 -(defn generate-gcode [pic]
   1.308 -  (str (raster-preamble)
   1.309 -
   1.310 -       (str-join ""
   1.311 -		 (map
   1.312 -		  (fn [index row]
   1.313 -		    (partial row->gcode dpi (even? index)) (indexed (make-rows pic))))
   1.314 -		 (raster-epilogue))))
   1.315 -
   1.316 -
   1.317 -       
   1.318 -;       (str-join "" (map (partial row->gcode dpi) (make-rows pic)))
   1.319 - ;      (raster-epilogue)))
   1.320 -  
   1.321 -
   1.322 -
   1.323 -(defn rotate [degrees #^ImagePlus image]
   1.324 -  (.rotate (.getChannelProcessor image) degrees)
   1.325 -  image)
   1.326 -
   1.327 -(defn map-keys [f m]
   1.328 -  (into {} (map (fn [[key val]] [(f key) val]) m)))
   1.329 -
   1.330 -
   1.331 -
   1.332 -(defn invert-frame-hash [pic]
   1.333 -  (map-keys (fn [[x y]] [x (- (:height (meta pic)) y 1)]) pic ))
   1.334 -
   1.335 -
   1.336 -(defn generate-files [pic]
   1.337 -  (let [image (invert-frame-hash (b&w (frame-hash  (rotate 180 (ImagePlus. pic)))))]
   1.338 -    (spit "/home/r/kevin/out.ngc" (generate-gcode image))
   1.339 -    (spit "/home/r/kevin/out.gmask" (generate-gmask image))
   1.340 -    image))
   1.341 -
   1.342 -
   1.343 -
   1.344 -(defn update-state []
   1.345 -(def sing "/home/r/lasercutter/graster/signer4laser2x1.png")
   1.346 -
   1.347 -(def pic (frame-hash (let [image  (ImagePlus. sing)]
   1.348 -		       (.rotate (.getChannelProcessor image) 180)
   1.349 -		       image)))
   1.350 -
   1.351 -(def pic (b&w pic)))
   1.352 -
   1.353 -
   1.354 -
   1.355 -
   1.356 -
   1.357 -
   1.358 -
   1.359 -
   1.360 -
   1.361  
   1.362  
   1.363  ;;;; testing
   1.364  
   1.365 -(defn init []
   1.366 -  (let [stuff
   1.367 -    
   1.368 -      (bound-fn []
   1.369 +(defn generate-files [pic]
   1.370 +    (println "made-image")
   1.371 +    (spit "/home/r/kevin/out.ngc" (generate-gcode pic))
   1.372 +    (println "/home/r/kevin/out.ngc")
   1.373 +    (spit "/home/r/kevin/out.gmask" (generate-gmask pic))
   1.374 +    (println "/home/r/kevin/out.gmask")
   1.375 +    pic)
   1.376  
   1.377 -	(do 
   1.378 -	  (println "hi everyone")
   1.379 -	  (def img "/home/r/kevin/sing.png")
   1.380 -	  (def pic (frame-hash (let [image  (ImagePlus. img)]
   1.381 -				 (.rotate (.getChannelProcessor image) 180)
   1.382 -				 image)))
   1.383 +(defn update-state []
   1.384 +  (def sing "/home/r/kevin/sing.png")
   1.385 +  (def pic (frame-hash (ImagePlus. sing)))  
   1.386 +  (def pic (b&w pic)))
   1.387  
   1.388 +(defn compare-gen-fn [n f cmp]
   1.389 +  (let [theirs (re-split #"\n" (slurp cmp))
   1.390 +	ours   (re-split #"\n" (f pic))]
   1.391 +    (println (format "%1$-25s%2$s" "OURS" "THEIRS"))
   1.392 +    (println "_______________________________________")
   1.393 +    (dorun (map (fn [[us them]] (println
   1.394 +			     (format "%1$-25s%2$s" us them)))
   1.395 +			     (take n (partition 2 (interleave  ours theirs)))))))
   1.396  
   1.397 -	  (def test-image
   1.398 -	       (invert-frame-hash (b&w (frame-hash  (rotate 180 (ImagePlus. img))))))
   1.399 -	  
   1.400 -	  (defn test-gmask []
   1.401 -	    (println (str-join "" (take 170 (generate-gmask test-image)))))
   1.402 +(defn compare-gcode [n]
   1.403 +  (compare-gen-fn n generate-gcode "/home/r/kevin/reference.ngc"))
   1.404 +(defn compare-gmask [n]
   1.405 +  (compare-gen-fn n generate-gmask "/home/r/kevin/reference.gmask"))
   1.406 +  
   1.407  
   1.408 -	  (println "ALL variables initialized!")
   1.409  
   1.410 -	  ))]
   1.411 -  (.start
   1.412 -     (Thread. stuff))))
   1.413  
   1.414  
   1.415 -
   1.416 -(defn thread-test []
   1.417 -
   1.418 -  (let [temp *out*]
   1.419 -    (.start
   1.420 -     (Thread.
   1.421 -      (fn []
   1.422 -	(with-bindings  {#'*out* temp}
   1.423 -	  (Thread/sleep 5000)
   1.424 -	  (println "hi")))))))
   1.425 -
   1.426 -
   1.427 -(comment
   1.428 -  (do
   1.429 -    (require 'rlm.quick)
   1.430 -    (ns laser.rasterize)
   1.431 -    (rlm.quick/dirty)
   1.432 -    (use :reload-all 'laser.rasterize)
   1.433 -    (undef map-keys)
   1.434 -    (use :reload-all 'laser.rasterize)))