racket/collects/htdch/draw/support.scm
2006-04-24 14:19:31 +00:00

191 lines
7.4 KiB
Scheme

#cs
(module support mzscheme
(require (lib "draw.ss" "htdp")
(lib "posn.ss" "lang")
(lib "class.ss")
(lib "unit.ss") (lib "unitsig.ss")
(lib "String.ss" "profj" "libs" "java" "lang")
(lib "Throwable.ss" "profj" "libs""java""lang")
(lib "RuntimeException.ss" "profj" "libs" "java" "lang"))
(provide world-native@ world-native^ canvas-native@ canvas-native^ support^)
(define-signature world-native^
(endOfTime-native
endOfWorld-native
bigBangO-double-native))
(define-signature canvas-native^
(show-native
close-native
stop-native
copy-native
drawCircle-draw.Posn-int-draw.Color-native
drawDisk-draw.Posn-int-draw.Color-native
drawRect-draw.Posn-int-int-draw.Color-native
drawLine-draw.Posn-draw.Posn-draw.Color-native
drawString-draw.Posn-java.lang.String-native
clearCircle-draw.Posn-int-draw.Color-native
clearDisk-draw.Posn-int-draw.Color-native
clearRect-draw.Posn-int-int-draw.Color-native
clearLine-draw.Posn-draw.Posn-draw.Color-native))
(define-signature support^ (void-or-true imperative))
(define canvas-native@
(unit/sig canvas-native^
(import support^)
;raises a Java exception with the specified error message
;raise-error: String -> void
(define (raise-error message)
(raise
(create-java-exception RuntimeException message
(lambda (exn str)
(send exn RuntimeException-constructor-java.lang.String str))
(current-continuation-marks))))
(define-syntax (wrap-start-check stx)
(syntax-case stx ()
[(_ body ...)
#'(with-handlers
((exn:fail?
(lambda (e)
(raise-error
(format
(string-append
"The method show() must be called on the canvas"
"before using any drawing methods [~s]")
(exn-message e))))))
(begin (begin body ...) void-or-true))]))
(define Posn-x-get (dynamic-require '(lib "Posn.ss" "htdch" "draw") 'Posn-x-get))
(define Posn-y-get (dynamic-require '(lib "Posn.ss" "htdch" "draw") 'Posn-y-get))
(define (build-posn posnO) (make-posn (Posn-x-get posnO) (Posn-y-get posnO)))
(define (color->symbol colorO) (string->symbol (to-lower-case (send colorO my-name))))
;Raises an error if value less than or equal to 0
;check-arg: num string string -> boolean
(define (check-arg value method argument)
(or (> value 0)
(raise-error
(format "Method ~a expects an int >= 0 for ~a argument, given ~a" method argument value))))
(define (to-lower-case s)
(letrec ((lower
(lambda (s)
(cond
((null? s) s)
(else (cons (char-downcase (car s))
(lower (cdr s))))))))
(list->string (lower (string->list s)))))
(define (show-native this accs gets privates)
;; Kathy: it looks like I am working around a bug here.
;; I really wanted to write ([hash-table-get privates 'width] this)
;; but that didn't work at all. 'width is not a key for privates,
;; even though it is a private field. Then I wanted to write
;; ([hash-table-get privates 'width] this), just like in World-native-methods.
;; That failed, too, with an arity error.
(define x (with-method ([g (this Canvas-width-get)]) (g '___)))
(define y (with-method ([g (this Canvas-height-get)]) (g '___)))
(start-and-export x y privates)
void-or-true)
(define (close-native this accs gets privates)
(wrap-start-check ([hash-table-get privates '%stop])))
(define (stop-native this accs gets privates)
(wrap-start-check ([hash-table-get privates '%end-of-time])))
(define (copy-native this accs gets privates)
(wrap-start-check ([hash-table-get privates 'copy])))
(define (drawCircle-draw.Posn-int-draw.Color-native this accs gets privates posn r c)
(wrap-start-check
(check-arg r "drawCircle(Posn, int, Color)" "second")
([hash-table-get privates '%draw-circle] (build-posn posn) r (color->symbol c))))
(define (drawDisk-draw.Posn-int-draw.Color-native this accs gets privates posn r c)
(wrap-start-check
(check-arg r "drawDisk(Posn, int, Color)" "second")
([hash-table-get privates '%draw-solid-disk] (build-posn posn) r (color->symbol c))))
(define (drawRect-draw.Posn-int-int-draw.Color-native this accs gets privates posn w h c)
(wrap-start-check
(check-arg w "drawRect(Posn, int, int, Color)" "second")
(check-arg h "drawRect(Posn, int, int, Color)" "third")
([hash-table-get privates '%draw-solid-rect] (build-posn posn) w h (color->symbol c))))
(define (drawLine-draw.Posn-draw.Posn-draw.Color-native this accs gets privates p0 p1 c)
(wrap-start-check
([hash-table-get privates '%draw-solid-line] (build-posn p0) (build-posn p1) (color->symbol c))))
(define (drawString-draw.Posn-java.lang.String-native this accs gets privates p s)
(define s* (send s get-mzscheme-string))
(wrap-start-check
([hash-table-get privates '%draw-string] (build-posn p) s*)))
(define (clearCircle-draw.Posn-int-draw.Color-native this accs gets privates p r c)
(wrap-start-check
(check-arg r "clearCircle(Posn, int, Color)" "second")
([hash-table-get privates '%clear-circle] (build-posn p) r (color->symbol c))))
(define (clearDisk-draw.Posn-int-draw.Color-native this accs gets privates p r c)
(wrap-start-check
(check-arg r "clearDisk(Posn, int, Color)" "second")
([hash-table-get privates '%clear-solid-disk] (build-posn p) r (color->symbol c))))
(define (clearRect-draw.Posn-int-int-draw.Color-native this accs gets privates p w h c)
(wrap-start-check
(check-arg w "clearRect(Posn, int, int, Color)" "second")
(check-arg h "clearRect(Posn, int, int, Color)" "third")
([hash-table-get privates '%clear-solid-rect] (build-posn p) w h (color->symbol c))))
(define (clearLine-draw.Posn-draw.Posn-draw.Color-native this accs gets privates p0 p1 c)
(wrap-start-check
([hash-table-get privates '%clear-solid-line] (build-posn p0) (build-posn p1) (color->symbol c))))
))
(define world-native@
(unit/sig world-native^
(import support^)
(define (bigBangO-double-native this accs gets privates i)
(define theCanvas ((hash-table-get accs 'theCanvas) this))
(define width (with-method ([g (theCanvas Canvas-width-get)]) (g '___)))
(define height (with-method ([g (theCanvas Canvas-height-get)]) (g '___)))
;; call only *after* start
(define (on-event world th)
(begin-draw-sequence)
(send theCanvas copy)
(send world erase)
(let ([world (imperative (th) world)])
(send world putCanvas-draw.Canvas theCanvas)
(send world draw)
(end-draw-sequence)
world))
(send theCanvas show)
(big-bang i this)
(on-tick-event
(lambda (world)
(on-event world (lambda () (send world onTick)))))
(on-key-event
(lambda (ke world)
(define ke* (make-java-string (if (char? ke) (string ke) (symbol->string ke))))
(on-event world (lambda () (send world onKeyEvent-java.lang.String ke*)))))
void-or-true)
(define (endOfTime-native this accs gets privates)
(define theCanvas ((hash-table-get accs 'theCanvas) this))
(send theCanvas stop)
#t)
(define (endOfWorld-native this accs gets privates)
(define theCanvas ((hash-table-get accs 'theCanvas) this))
(send theCanvas stop)
this)))
)