Home My Page Projects Code Snippets Project Openings SML/NJ
Summary Activity Forums Tracker Lists Tasks Docs Surveys News SCM Files

SCM Repository

[smlnj] View of /eXene/releases/release-110.63/graph-util/bitmap-io.sml
ViewVC logotype

View of /eXene/releases/release-110.63/graph-util/bitmap-io.sml

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1368 - (download) (annotate)
Sat Sep 13 04:44:43 2003 UTC (16 years ago) by mblume
Original Path: sml/trunk/src/eXene/graph-util/bitmap-io.sml
File size: 5787 byte(s)
type of inputLine changed; minor bugfixes; minor installer improvements
(* bitmap-io.sml
 *
 * COPYRIGHT (c) 1993 by AT&T Bell Laboratories.  See COPYRIGHT file for details.
 *
 * This module provides code to read and write depth-1 images
 * stored in X11 bitmap file format (see XReadBitmapFile(3X).
 * It does not use any CML features, and thus can be compiled
 * as part of a sequential SML program.
 *)

structure BitmapIO : BITMAP_IO =
  struct

    structure EXB = EXeneBase
    structure G = Geometry
    structure SS = Substring
    structure W8V = Word8Vector

    exception BitmapFileInvalid

    local
      fun scan f s = valOf(f s) handle _ => []
      val scanDefine = scan (Scan.sscanf "#define %s %d")
      val scanUChar = scan (Scan.sscanf "static unsigned char %s = {")
      val scanChar = scan (Scan.sscanf "static char %s = {")
    in
    datatype line = SKIP | DEFINE of (string * int) | BEGIN of string
    fun scanString s = (case (scanDefine s)
	   of [Format.STR s, Format.INT n] => DEFINE(s, n)
	    | _ => (case (scanUChar s)
		 of [Format.STR s] => BEGIN s
		  | _ => (case (scanChar s)
		       of [Format.STR s] => BEGIN s
			| _ => SKIP
		      (* end case *))
		(* end case *))
	  (* end case *))
    end

    val isDelim = Char.contains " \t\n,}"

  (* return true if s1 is a suffix of s2 *)
    fun isSuffix (s1, s2) = let
	  val n1 = size s1 and n2 = size s2
	  in
	    (n1 <= n2) andalso SS.isPrefix s1 (SS.substring(s2, n2 - n1,n1))
	  end

    fun readBitmap inStrm = let
	  fun inputLine () = (case TextIO.inputLine inStrm
		 of NONE => raise BitmapFileInvalid
		  | SOME s => s
		(* end case *))
          val inputSS = SS.all o inputLine
	  fun setWid ({wid, ht, x_hot, y_hot}, w) =
		{wid=SOME w, ht=ht, x_hot=x_hot, y_hot=y_hot}
	  fun setHt ({wid, ht, x_hot, y_hot}, h) =
		{wid=wid, ht=SOME h, x_hot=x_hot, y_hot=y_hot}
	  fun setXHot ({wid, ht, x_hot, y_hot}, x) =
		{wid=wid, ht=ht, x_hot=SOME x, y_hot=y_hot}
	  fun setYHot ({wid, ht, x_hot, y_hot}, y) =
		{wid=wid, ht=ht, x_hot=x_hot, y_hot=SOME y}
	  fun scanHdr (arg as {wid, ht, x_hot, y_hot}) = (
		case (scanString (inputLine ()))
		 of SKIP => scanHdr arg
		  | (DEFINE("width", n)) => scanHdr(setWid(arg, n))
		  | (DEFINE("height", n)) => scanHdr(setHt(arg, n))
		  | (DEFINE("x_hot", n)) => scanHdr(setXHot(arg, n))
		  | (DEFINE("y_hot", n)) => scanHdr(setYHot(arg, n))
		  | (DEFINE(s, n)) =>
		      if isSuffix("_width", s)
			then scanHdr(setWid(arg, n))
		      else if isSuffix("_height", s)
			then scanHdr(setHt(arg, n))
		      else if isSuffix("_x_hot", s)
			then scanHdr(setXHot(arg, n))
		      else if isSuffix("_y_hot", s)
			then scanHdr(setYHot(arg, n))
			else scanHdr arg
		  | (BEGIN s) => arg
		(* end case *))
	  fun getNextInt ss = let
                val ss' = SS.dropl isDelim ss
		in
                  if SS.isEmpty ss' then getNextInt (inputSS())
                  else case Int.scan StringCvt.HEX (SS.getc) ss' of
                    NONE => raise BitmapFileInvalid
                  | SOME v => v
		end
	  val (wid, ht, hot) = (
		case (scanHdr{wid=NONE, ht=NONE, x_hot=NONE, y_hot=NONE})
		 of {wid=NONE, ...} => raise BitmapFileInvalid
		  | {ht=NONE, ...} => raise BitmapFileInvalid
		  | {wid=SOME w, ht=SOME h, x_hot=SOME x, y_hot=SOME y} =>
		      (w, h, SOME(G.PT{x=x, y=y}))
		  | {wid=SOME w, ht=SOME h, ...} => (w, h, NONE)
		(* end case *))
	  val bytesPerLine = (wid+7) div 8
	  fun getScanLine ss = let
		val scanLn = Unsafe.CharVector.create bytesPerLine
		fun get (ss, k) = if (k < bytesPerLine)
		      then let
			val (byte, ss) = getNextInt ss
			in
			  Unsafe.CharVector.update (scanLn, k, Char.chr byte);
			  get(ss, k+1)
			end
		      else (Byte.stringToBytes scanLn, ss)
		in
		  get (ss, 0)
		end (* getScanLine *)
	  fun getData (_, 0, l) = [rev l]
	    | getData (ss, n, l) = let
		val (scanLn, ss) = getScanLine ss
		in
		  getData(ss, n-1, scanLn::l)
		end
	  in
	    { image = EXB.IMAGE{
		  sz = G.SIZE{wid=wid, ht=ht},
		  data = getData(inputSS(), ht, [])
		},
	      hot_spot = hot
	    }
	  end

    val formatDefine = Format.format "#define %s%s %d\n"
    val formatUChar = Format.format "static unsigned char %sbits[] = {\n"
    val formatByte = Format.format "%#04x"

    exception NotBitmap
    exception BadImageData = EXB.BadImageData

    fun writeBitmap (outStrm, name, {image, hot_spot}) = let
	  val name = (case name of "" => "" | _ => name ^ "_")
	  fun pr s = TextIO.output (outStrm, s)
	  fun writeDefine (s, n) =
		pr(formatDefine [Format.STR name, Format.STR s, Format.INT n])
	  val (wid, ht, data) = (case image
		 of (EXB.IMAGE{sz=G.SIZE{wid, ht}, data=[data]}) => (wid, ht, data)
		  | _ => raise NotBitmap
		(* end case *))
	  fun prData () = let
		val bytesPerLine = (wid + 7) div 8
		fun nextByte (s, r, i) = if (i < bytesPerLine)
		      then (W8V.sub(s, i), (s, r, i+1))
		      else nextLine r
		and nextLine [] = raise BadImageData
		  | nextLine (s::r) = if (W8V.length s = bytesPerLine)
		      then nextByte(s, r, 0)
		      else raise BadImageData
		fun prLine (0, _, _) = ()
		  | prLine (n, 12, data) = (pr ",\n"; prLine(n, 0, data))
		  | prLine (n, k, data) = let
		      val (byte, data) = nextByte data
		      in
			if (k = 0) then pr "    " else pr ", ";
			pr(formatByte [Format.WORD8 byte]);
			prLine (n-1, k+1, data)
		      end
		in
		  if (length data = ht)
		    then prLine(ht*bytesPerLine, 0, (W8V.fromList[], data, bytesPerLine))
		    else raise BadImageData
		end
	  in
	    writeDefine ("height", ht);
	    writeDefine ("width", wid);
	    case hot_spot
	     of (SOME(G.PT{x, y})) => (
		  writeDefine ("x_hot", x);
		  writeDefine ("y_hot", y))
	      | _ => ()
	    (* end case *);
	    pr(formatUChar [Format.STR name]);
	    prData ();
	    pr "\n};\n";
	    TextIO.flushOut outStrm
	  end

  end; (* structure BitmapIO *)


root@smlnj-gforge.cs.uchicago.edu
ViewVC Help
Powered by ViewVC 1.0.0