2014-11-19 22:48:05 +00:00
|
|
|
#lang racket/base
|
|
|
|
(require racket/list
|
|
|
|
racket/match
|
|
|
|
racket/contract/base
|
|
|
|
racket/flonum
|
|
|
|
racket/format
|
|
|
|
racket/generic
|
|
|
|
lux/chaos)
|
|
|
|
|
|
|
|
(define-generics word
|
2014-11-21 18:26:01 +00:00
|
|
|
(word-fps word)
|
2014-11-19 22:48:05 +00:00
|
|
|
(word-label word frame-time)
|
2015-04-14 18:00:59 +00:00
|
|
|
(word-evt word)
|
2014-11-20 15:53:15 +00:00
|
|
|
(word-event word evt)
|
|
|
|
(word-tick word)
|
|
|
|
(word-output word)
|
|
|
|
(word-return word)
|
2014-11-19 22:48:05 +00:00
|
|
|
#:fallbacks
|
2014-11-21 18:26:01 +00:00
|
|
|
[(define (word-fps w)
|
|
|
|
60.0)
|
|
|
|
(define (word-label w frame-time)
|
2014-11-19 22:48:05 +00:00
|
|
|
(lux-standard-label "Lux" frame-time))
|
2015-04-14 18:00:59 +00:00
|
|
|
(define (word-evt w)
|
|
|
|
never-evt)
|
2014-11-20 15:53:15 +00:00
|
|
|
(define (word-event w e) w)
|
|
|
|
(define (word-tick w) w)
|
2014-11-22 18:54:08 +00:00
|
|
|
(define (word-output w) #f)
|
2014-11-20 15:53:15 +00:00
|
|
|
(define (word-return w) w)])
|
2014-11-19 22:48:05 +00:00
|
|
|
|
|
|
|
(define (lux-standard-label l frame-time)
|
|
|
|
(~a l
|
|
|
|
": "
|
|
|
|
"Frame time: "
|
|
|
|
(~r frame-time
|
|
|
|
#:min-width 5
|
|
|
|
#:precision 1)
|
|
|
|
"ms; "
|
|
|
|
"FPS: "
|
|
|
|
(~r (fl/ 1000.0 frame-time)
|
|
|
|
#:min-width 10
|
|
|
|
#:precision 2)))
|
|
|
|
|
|
|
|
(define current-chaos (make-parameter #f))
|
|
|
|
|
|
|
|
(define (call-with-chaos c t)
|
2014-11-26 22:21:55 +00:00
|
|
|
(chaos-start! c)
|
2014-11-19 22:48:05 +00:00
|
|
|
(parameterize ([current-chaos c])
|
2014-11-26 22:21:55 +00:00
|
|
|
(begin0 (t)
|
|
|
|
(chaos-stop! c))))
|
2014-11-19 22:48:05 +00:00
|
|
|
|
|
|
|
(define (fiat-lux w)
|
|
|
|
(define c (current-chaos))
|
|
|
|
(unless c
|
|
|
|
(error 'fiat-lux "Not called within call-with-chaos"))
|
|
|
|
(factum-fiat-lux c w))
|
|
|
|
|
2015-07-30 19:56:57 +00:00
|
|
|
;; XXX In the process of adding (collect-garbage? #t) to this, I
|
|
|
|
;; noticed a problem with the way that things are timed. Right now, if
|
|
|
|
;; an input event occurs, then the alarm can be put off. Also, the
|
|
|
|
;; output occurs on input events even when there is an FPS. What
|
|
|
|
;; really should happen, however, is that output should only occur at
|
|
|
|
;; the FPS and the alarm deadlines should never reset. It needs to be
|
|
|
|
;; considerably changed to fix that. I did something in the last
|
|
|
|
;; reverted commit, but I don't likw it.
|
2014-11-19 22:48:05 +00:00
|
|
|
(define (factum-fiat-lux c w)
|
2015-07-30 19:54:41 +00:00
|
|
|
(define (update-word w f make-next-time)
|
2014-11-20 21:31:02 +00:00
|
|
|
(define start-time (current-inexact-milliseconds))
|
|
|
|
(define new-w (f w))
|
|
|
|
(match new-w
|
|
|
|
[#f
|
2015-09-03 19:12:53 +00:00
|
|
|
((LOG! word-return) w)]
|
2014-11-20 21:31:02 +00:00
|
|
|
[_
|
2015-09-03 19:12:53 +00:00
|
|
|
(chaos-output! c ((LOG! word-output) new-w))
|
2015-07-30 19:54:41 +00:00
|
|
|
(define end-time (current-inexact-milliseconds))
|
|
|
|
(define frame-time (fl- end-time start-time))
|
|
|
|
(define new-label
|
2015-09-03 19:12:53 +00:00
|
|
|
((LOG! word-label) new-w frame-time))
|
2015-07-30 19:54:41 +00:00
|
|
|
(chaos-label! c new-label)
|
|
|
|
(define next-time (make-next-time new-w start-time))
|
|
|
|
(body next-time new-w)]))
|
2014-11-21 20:46:17 +00:00
|
|
|
(define (compute-next-time start-time fps)
|
|
|
|
(define time-incr (fl* (fl/ 1.0 fps) 1000.0))
|
|
|
|
(define next-time (fl+ start-time time-incr))
|
|
|
|
next-time)
|
2015-07-30 19:54:41 +00:00
|
|
|
(define (body next-time w)
|
2015-04-14 17:49:09 +00:00
|
|
|
(define input-evt
|
2014-11-20 15:53:15 +00:00
|
|
|
(handle-evt
|
2015-09-03 19:12:53 +00:00
|
|
|
(choice-evt ((LOG! word-evt) w)
|
2015-04-14 18:00:59 +00:00
|
|
|
(chaos-event c))
|
2014-11-20 15:53:15 +00:00
|
|
|
(λ (e)
|
2014-11-20 21:31:02 +00:00
|
|
|
(update-word w
|
|
|
|
(λ (w)
|
2015-09-03 19:12:53 +00:00
|
|
|
((LOG! word-event) w e))
|
2014-11-21 20:46:17 +00:00
|
|
|
(λ (new-w start-time)
|
2015-09-03 19:12:53 +00:00
|
|
|
(define old-fps ((LOG! word-fps) w))
|
|
|
|
(define fps ((LOG! word-fps) new-w))
|
2014-11-21 20:46:17 +00:00
|
|
|
(if (= old-fps fps)
|
|
|
|
next-time
|
2015-04-14 17:49:09 +00:00
|
|
|
(compute-next-time start-time fps)))))))
|
|
|
|
(define refresh-evt
|
2014-11-20 15:53:15 +00:00
|
|
|
(handle-evt
|
2014-11-21 20:46:17 +00:00
|
|
|
(alarm-evt next-time)
|
2014-11-20 15:53:15 +00:00
|
|
|
(λ (_)
|
2014-11-20 21:31:02 +00:00
|
|
|
(update-word w
|
2015-09-03 19:12:53 +00:00
|
|
|
(LOG! word-tick)
|
2014-11-21 20:46:17 +00:00
|
|
|
(λ (new-w start-time)
|
2015-09-03 19:12:53 +00:00
|
|
|
(define fps ((LOG! word-fps) new-w))
|
2015-04-14 17:49:09 +00:00
|
|
|
(compute-next-time start-time fps))))))
|
|
|
|
(sync/timeout
|
|
|
|
(λ ()
|
|
|
|
(chaos-yield
|
|
|
|
c
|
|
|
|
(choice-evt input-evt refresh-evt)))
|
|
|
|
input-evt))
|
2015-07-30 19:54:41 +00:00
|
|
|
(chaos-swap! c (λ () (body 0 w))))
|
2014-11-19 22:48:05 +00:00
|
|
|
|
2015-09-03 19:12:53 +00:00
|
|
|
(define-syntax-rule (LOG! id)
|
|
|
|
(begin (LOG!* 'id) id))
|
|
|
|
(define (LOG!* i)
|
|
|
|
(writeln (cons (current-inexact-milliseconds) i))
|
|
|
|
(flush-output))
|
|
|
|
|
2014-11-19 22:48:05 +00:00
|
|
|
(provide
|
|
|
|
gen:word
|
|
|
|
(contract-out
|
2014-11-22 18:54:08 +00:00
|
|
|
[word?
|
|
|
|
(-> any/c word?)]
|
2014-11-19 22:48:05 +00:00
|
|
|
[lux-standard-label
|
|
|
|
(-> string? flonum?
|
|
|
|
string?)]
|
|
|
|
[call-with-chaos
|
|
|
|
(-> chaos? (-> any)
|
|
|
|
any)]
|
|
|
|
[fiat-lux
|
|
|
|
(-> word?
|
|
|
|
any)]))
|