132 lines
4.0 KiB
Scheme
132 lines
4.0 KiB
Scheme
;;; -*-Scheme-*-
|
|
;;;
|
|
;;; VCR simulation
|
|
|
|
|
|
;; Initialization
|
|
|
|
(require 'motif)
|
|
(load-widgets shell bulletin-board row-column label push-button)
|
|
|
|
(define top (application-initialize 'vcr))
|
|
(define con (widget-context top))
|
|
|
|
|
|
;; The layout of the VCR's controls
|
|
|
|
(define vcr (create-managed-widget (find-class 'row-column) top))
|
|
|
|
(define panel (create-managed-widget (find-class 'bulletin-board) vcr))
|
|
|
|
(define tape (create-managed-widget (find-class 'push-button) panel))
|
|
(set-values! tape 'x 10 'y 10 'width 150 'border-width 2 'label-string 'empty
|
|
'recompute-size #f
|
|
'activate-callback (list (lambda _ (engine 'load))))
|
|
|
|
(define counter (create-managed-widget (find-class 'push-button) panel))
|
|
(set-values! counter 'x 170 'y 10 'width 50 'label-string "0"
|
|
'alignment "alignment_end" 'recompute-size #f)
|
|
|
|
(define function (create-managed-widget (find-class 'push-button) panel))
|
|
(set-values! function 'x 230 'y 10 'width 70 'label-string "stop"
|
|
'recompute-size #f)
|
|
|
|
(define buttons (create-managed-widget (find-class 'row-column) vcr))
|
|
(set-values! buttons 'orientation 'horizontal)
|
|
|
|
(define-macro (define-button label activate arm disarm)
|
|
`(let ((b (create-managed-widget (find-class 'push-button) buttons)))
|
|
(set-values! b 'label-string ,label)
|
|
(add-callback b 'activate-callback (lambda _ ,activate))
|
|
(add-callback b 'arm-callback (lambda _ ,arm))
|
|
(add-callback b 'disarm-callback (lambda _ ,disarm))))
|
|
|
|
(define-button 'eject (begin (engine 'stop) (engine 'empty)) #f #f)
|
|
(define-button 'play (engine 'play) #f #f)
|
|
(define-button 'stop (engine 'stop) #f #f)
|
|
(define-button 'forw (engine 'forw) (engine 'cue #t) (engine 'cue #f))
|
|
(define-button 'rew (engine 'rew) (engine 'review #t) (engine 'review #f))
|
|
(define-button 'pause (engine 'pause) #f #f)
|
|
|
|
|
|
;; The `logic' of the VCR
|
|
|
|
(define engine
|
|
(let ((timer #f) (interval) (loaded #f) (cnt 0) (state 'stop))
|
|
|
|
(define (advance x)
|
|
(set! cnt (modulo (+ cnt x) 10000000))
|
|
(set-values! counter 'label-string (format #f "~s" cnt)))
|
|
|
|
(define (timeout x)
|
|
(advance x)
|
|
(set! timer (context-add-timeout con interval (lambda _ (timeout x)))))
|
|
|
|
(define (set-timer when x)
|
|
(stop-timer)
|
|
(set! interval when)
|
|
(set! timer (context-add-timeout con when (lambda _ (timeout x)))))
|
|
|
|
(define (stop-timer)
|
|
(if timer (remove-timeout timer))
|
|
(set! timer #f))
|
|
|
|
(define (cue/review on? x)
|
|
(if on?
|
|
(if (not (eq? state 'play)) ; do nothing if not playing
|
|
state
|
|
(set-timer 100 x) ; else
|
|
'cue/review)
|
|
(if (not (eq? state 'cue/review)) ; do nothing if not in cue/review
|
|
state ; mode
|
|
(set-timer 1000 100) ; else switch back to play mode
|
|
'play)))
|
|
|
|
(lambda (op . args)
|
|
(call-with-current-continuation
|
|
(lambda (return)
|
|
(case op
|
|
(load
|
|
(set-values! tape 'label-string 'loaded)
|
|
(set! loaded #t))
|
|
(empty
|
|
(set-values! tape 'label-string 'empty)
|
|
(set! loaded #f))
|
|
(else
|
|
(if (not loaded)
|
|
(return #f))
|
|
(case op
|
|
(stop
|
|
(stop-timer))
|
|
(cue (set! op (cue/review (car args) 100)))
|
|
(review (set! op (cue/review (car args) -100)))
|
|
(pause
|
|
(cond ((eq? state 'pause)
|
|
(set-timer 1000 100)
|
|
(set! op 'play))
|
|
((eq? state 'play)
|
|
(stop-timer))
|
|
(else
|
|
(return #f))))
|
|
(forw
|
|
(cond ((eq? state 'pause)
|
|
(advance 4)
|
|
(set! op 'pause)) ; stay in pause mode
|
|
((not (eq? state 'cue/review))
|
|
(set-timer 1000 10000))
|
|
(else (set! op state)))) ; stay in the old mode
|
|
(rew
|
|
(cond ((eq? state 'pause)
|
|
(advance -4)
|
|
(set! op 'pause))
|
|
((not (eq? state 'cue/review))
|
|
(set-timer 1000 -10000))
|
|
(else (set! op state))))
|
|
(play
|
|
(set-timer 1000 100)))
|
|
(set! state op)
|
|
(set-values! function 'label-string op))))))))
|
|
|
|
(realize-widget top)
|
|
(context-main-loop con)
|