Fixes to FrTime dynamic value snips (for textual content, at least).
* Expect DrRacket to make a copy of the copy of the snip we give it. * Add documentation explaining some of the trickiness involved in this code and how it might be possible to make it less brittle. * Add a note about editor-snip% not behaving as expected when non-textual elements are inserted and subsequently removed (or the underlying editor object is replaced).
This commit is contained in:
parent
f15ca607ed
commit
f418805c32
|
@ -35,33 +35,33 @@
|
|||
(send current draw dc x y left top right bottom dx dy draw-caret))
|
||||
(super-instantiate (" "))))
|
||||
|
||||
(define make-snip
|
||||
(case-lambda
|
||||
[(bhvr)
|
||||
(make-object string-snip%
|
||||
(let ([tmp (cond
|
||||
[(behavior? bhvr) (value-now bhvr)]
|
||||
[(event? bhvr) (signal-value bhvr)]
|
||||
[else bhvr])])
|
||||
(cond
|
||||
[(event-set? tmp) (format "#<event (last: ~a@~a)>"
|
||||
(event-set-events tmp) (event-set-time tmp))]
|
||||
[(undefined? tmp) "<undefined>"]
|
||||
[else (format "~a" tmp)])))]
|
||||
[(bhvr super-render-fun)
|
||||
(get-rendering (value-now bhvr) super-render-fun)]))
|
||||
(define (make-snip bhvr)
|
||||
(make-object string-snip%
|
||||
(let ([tmp (cond
|
||||
[(behavior? bhvr) (value-now bhvr)]
|
||||
[(event? bhvr) (signal-value bhvr)]
|
||||
[else bhvr])])
|
||||
(cond
|
||||
[(event-set? tmp) (format "#<event (last: ~a@~a)>"
|
||||
(event-set-events tmp) (event-set-time tmp))]
|
||||
[(undefined? tmp) "<undefined>"]
|
||||
[else (format "~a" tmp)]))))
|
||||
|
||||
(define value-snip%
|
||||
(class string-snip%
|
||||
(init-field bhvr)
|
||||
(init-field bhvr [ignore-copy-count 1])
|
||||
(field [copies empty]
|
||||
[loc-bhvr (proc->signal (lambda () (update)) bhvr)]
|
||||
[current (make-snip bhvr)])
|
||||
[current (make-snip bhvr)]
|
||||
[loc-bhvr (proc->signal (lambda () (update)) bhvr)])
|
||||
|
||||
(define/override (copy)
|
||||
(let ([ret (make-object value-snip-copy% current this)])
|
||||
(set! copies (cons ret copies))
|
||||
ret))
|
||||
(if (> ignore-copy-count 0)
|
||||
(begin
|
||||
(set! ignore-copy-count (sub1 ignore-copy-count))
|
||||
this)
|
||||
(let ([ret (make-object value-snip-copy% current this)])
|
||||
(set! copies (cons ret copies))
|
||||
ret)))
|
||||
|
||||
(define/public (update)
|
||||
(set! current (make-snip bhvr))
|
||||
|
@ -71,19 +71,27 @@
|
|||
|
||||
(define dynamic-snip-copy%
|
||||
(class editor-snip%
|
||||
(init-field current parent)
|
||||
(inherit get-editor)
|
||||
(define/public (set-current c)
|
||||
(init-field initial-content parent)
|
||||
(inherit get-editor set-editor)
|
||||
|
||||
(define/public (update content)
|
||||
(parameterize ([current-eventspace drs-eventspace])
|
||||
(queue-callback
|
||||
(lambda ()
|
||||
(send (get-editor) lock #f)
|
||||
(send (get-editor) delete 0 (send (get-editor) last-position))
|
||||
(for-each (lambda (thing)
|
||||
(send (get-editor) insert thing
|
||||
(send (get-editor) last-position) (send (get-editor) last-position)))
|
||||
c)
|
||||
(send (get-editor) lock #t)))))
|
||||
;; TODO(ghcooper): Figure out why this doesn't work properly for non-
|
||||
;; textual content. (Image snips don't seem to be deleted from the
|
||||
;; editor.) It doesn't even work if we create a completely new
|
||||
;; racket:text% each time, which suggests it's a bug in the snip
|
||||
;; rather than the editor itself.
|
||||
(let ([editor (get-editor)])
|
||||
(send editor lock #f)
|
||||
(send editor delete 0 (send editor last-position))
|
||||
(for-each (lambda (thing)
|
||||
(send editor insert thing
|
||||
(send editor last-position)
|
||||
(send editor last-position)))
|
||||
content)
|
||||
(send editor lock #t))))))
|
||||
|
||||
(super-new
|
||||
[editor (new racket:text%)]
|
||||
|
@ -92,24 +100,53 @@
|
|||
[right-margin 0]
|
||||
[top-margin 0]
|
||||
[bottom-margin 0])
|
||||
(set-current current)))
|
||||
(update initial-content)))
|
||||
|
||||
;; Class of objects to be given to DrRacket for rendering a signal in the
|
||||
;; interactions window. However, DrRacket won't actually embed this snip
|
||||
;; directly into the interactions window; instead it makes a copy, and then a
|
||||
;; copy of the copy, and the second copy is what's really rendered. This makes
|
||||
;; life challenging for us, because what we want (I believe) is ultimately an
|
||||
;; editor-snip% whose contents we can rewrite whenever the signal changes.
|
||||
;; We can't make this class inherit from editor-snip%, though, because we need
|
||||
;; custom copy behavior, and editor-snip%'s copy method is final. Instead, this
|
||||
;; class is designed NOT to be rendered, but just to be copied, to keep track of
|
||||
;; the copy that's actually displayed, and to make sure the copy gets updated
|
||||
;; when the signal changes. The displayed "copy" is not, in fact, a copy at all
|
||||
;; but an instance of the dynamic-snip-copy% class defined above.
|
||||
;;
|
||||
;; TODO(ghcooper): This code is very brittle; it breaks whenever DrRacket
|
||||
;; changes the length of the chain of copies it makes. A better approach might
|
||||
;; be to have a single class that HAS an editor-snip% (instead of inheriting
|
||||
;; from editor-snip%), delegates all relevant calls to the editor-snip%, and has
|
||||
;; a copy method that makes a proper copy of itself and (like this class) keeps
|
||||
;; track of copies so it can notify them when they need to be redrawn.
|
||||
(define dynamic-snip%
|
||||
(class snip%
|
||||
(init-field bhvr super-render-fun)
|
||||
(init-field
|
||||
;; The behavior we want to render dynamically.
|
||||
bhvr
|
||||
;; Procedure that generates a rendering of the current value of bhvr.
|
||||
super-render-fun
|
||||
;; Number of times the copy method will just return this object. Ick!
|
||||
[ignore-copy-count 1])
|
||||
|
||||
(field [copies empty]
|
||||
[loc-bhvr (proc->signal (lambda () (update)) bhvr)]
|
||||
[current (make-snip bhvr super-render-fun)])
|
||||
(field [copies empty] ; "Copies" of this snip that we need to update.
|
||||
[current (get-rendering (value-now bhvr) super-render-fun)]
|
||||
[loc-bhvr (proc->signal (lambda () (update)) bhvr)])
|
||||
|
||||
(define/override (copy)
|
||||
(let ([ret (make-object dynamic-snip-copy% current this)])
|
||||
(set! copies (cons ret copies))
|
||||
ret))
|
||||
(if (> ignore-copy-count 0)
|
||||
(begin
|
||||
(set! ignore-copy-count (sub1 ignore-copy-count))
|
||||
this)
|
||||
(let ([ret (make-object dynamic-snip-copy% current this)])
|
||||
(set! copies (cons ret copies))
|
||||
ret)))
|
||||
|
||||
(define/public (update)
|
||||
(set! current (make-snip bhvr super-render-fun))
|
||||
(for-each (lambda (copy) (send copy set-current current)) copies))
|
||||
(set! current (get-rendering (value-now bhvr) super-render-fun))
|
||||
(for-each (lambda (copy) (send copy update current)) copies))
|
||||
|
||||
(define/override (size-cache-invalid)
|
||||
(for-each
|
||||
|
@ -138,6 +175,9 @@
|
|||
; easy case
|
||||
(super-render-fun val)))
|
||||
|
||||
;; get-rendering : any (any port -> void) -> (listof (string U snip%))
|
||||
;; Applies super-render-fun to val and a port. Returns the sequence of values
|
||||
;; written to the port.
|
||||
(define (get-rendering val super-render-fun)
|
||||
(let-values ([(in out) (make-pipe-with-specials)])
|
||||
(thread (lambda () (super-render-fun val out) (close-output-port out)))
|
||||
|
|
Loading…
Reference in New Issue
Block a user