175 lines
5.1 KiB
Racket
175 lines
5.1 KiB
Racket
#lang racket/base
|
|
(require ffi/unsafe
|
|
racket/class
|
|
"utils.rkt"
|
|
"types.rkt"
|
|
"window.rkt"
|
|
"x11.rkt"
|
|
"win32.rkt"
|
|
"gl-context.rkt"
|
|
"../../lock.rkt"
|
|
"../common/backing-dc.rkt"
|
|
racket/draw/unsafe/cairo
|
|
racket/draw/private/dc
|
|
racket/draw/private/bitmap
|
|
racket/draw/private/local
|
|
ffi/unsafe/alloc)
|
|
|
|
(provide
|
|
(protect-out dc%
|
|
do-backing-flush
|
|
x11-bitmap%
|
|
|
|
gdk_gc_new
|
|
gdk_gc_unref
|
|
gdk_gc_set_rgb_fg_color
|
|
gdk_draw_rectangle))
|
|
|
|
(define-gdk gdk_cairo_create (_fun _pointer -> _cairo_t)
|
|
#:wrap (allocator cairo_destroy))
|
|
|
|
(define-gdk gdk_gc_unref (_fun _pointer -> _void)
|
|
#:wrap (deallocator))
|
|
(define-gdk gdk_gc_new (_fun _GdkWindow -> _pointer)
|
|
#:wrap (allocator gdk_gc_unref))
|
|
(define-gdk gdk_gc_set_rgb_fg_color (_fun _pointer _GdkColor-pointer -> _void))
|
|
(define-gdk gdk_draw_rectangle (_fun _GdkWindow _pointer _gboolean _int _int _int _int -> _void))
|
|
|
|
(define-cstruct _GdkVisual-rec ([type-instance _pointer]
|
|
[ref_count _uint]
|
|
[qdata _pointer]
|
|
[type _int]
|
|
[depth _int]))
|
|
(define-gdk gdk_visual_get_system (_fun -> _GdkVisual-rec-pointer))
|
|
|
|
(define x11-bitmap%
|
|
(class bitmap%
|
|
(init w h gdk-win)
|
|
(super-make-object (make-alternate-bitmap-kind w h))
|
|
|
|
(define pixmap (gdk_pixmap_new gdk-win (max 1 w) (max 1 h)
|
|
(if gdk-win
|
|
-1
|
|
(GdkVisual-rec-depth
|
|
(gdk_visual_get_system)))))
|
|
(define s
|
|
(cairo_xlib_surface_create (gdk_x11_display_get_xdisplay
|
|
(gdk_drawable_get_display pixmap))
|
|
(gdk_x11_drawable_get_xid pixmap)
|
|
(gdk_x11_visual_get_xvisual
|
|
(gdk_drawable_get_visual pixmap))
|
|
w
|
|
h))
|
|
|
|
;; initialize bitmap to white:
|
|
(let ([cr (cairo_create s)])
|
|
(cairo_set_source_rgba cr 1.0 1.0 1.0 1.0)
|
|
(cairo_paint cr)
|
|
(cairo_destroy cr))
|
|
|
|
;; `get-gdk-pixmap' and `install-gl-context' are
|
|
;; localized in "gl-context.rkt"
|
|
(define/public (get-gdk-pixmap) pixmap)
|
|
(define/public (install-gl-context new-gl) (set! gl new-gl))
|
|
|
|
(define gl #f)
|
|
(define/override (get-bitmap-gl-context) gl)
|
|
|
|
(define/override (ok?) #t)
|
|
(define/override (is-color?) #t)
|
|
(define/override (has-alpha-channel?) #f)
|
|
|
|
(define/override (get-cairo-surface) s)
|
|
|
|
(define/override (release-bitmap-storage)
|
|
(atomically
|
|
(cairo_surface_destroy s)
|
|
(gobject-unref pixmap)
|
|
(set! s #f)))))
|
|
|
|
(define win32-bitmap%
|
|
(class bitmap%
|
|
(init w h gdk-win)
|
|
(super-make-object (make-alternate-bitmap-kind w h))
|
|
|
|
(define s
|
|
(if (not gdk-win)
|
|
(cairo_win32_surface_create_with_dib CAIRO_FORMAT_RGB24 w h)
|
|
(atomically
|
|
(let ([hdc (GetDC (gdk_win32_drawable_get_handle gdk-win))])
|
|
(begin0
|
|
(cairo_win32_surface_create_with_ddb hdc
|
|
CAIRO_FORMAT_RGB24 w h)
|
|
(ReleaseDC hdc))))))
|
|
|
|
(define/override (ok?) #t)
|
|
(define/override (is-color?) #t)
|
|
(define/override (has-alpha-channel?) #f)
|
|
|
|
(define/override (get-cairo-surface) s)
|
|
|
|
(define/override (release-bitmap-storage)
|
|
(atomically
|
|
(cairo_surface_destroy s)
|
|
(set! s #f)))))
|
|
|
|
(define dc%
|
|
(class backing-dc%
|
|
(init [(cnvs canvas)]
|
|
transparent?)
|
|
(inherit end-delay)
|
|
(define canvas cnvs)
|
|
|
|
(super-new [transparent? transparent?])
|
|
|
|
(define gl #f)
|
|
(define/override (get-gl-context)
|
|
(or gl
|
|
(let ([v (create-widget-gl-context (send canvas get-client-gtk))])
|
|
(when v (set! gl v))
|
|
v)))
|
|
|
|
(define/override (make-backing-bitmap w h)
|
|
(cond
|
|
[(and (eq? 'unix (system-type))
|
|
(send canvas get-canvas-background))
|
|
(make-object x11-bitmap% w h (widget-window (send canvas get-client-gtk)))]
|
|
[(and (eq? 'windows (system-type))
|
|
(send canvas get-canvas-background))
|
|
(make-object win32-bitmap% w h (widget-window (send canvas get-client-gtk)))]
|
|
[else
|
|
(super make-backing-bitmap (max 1 w) (max 1 h))]))
|
|
|
|
(define/override (get-backing-size xb yb)
|
|
(send canvas get-client-size xb yb))
|
|
|
|
(define/override (get-size)
|
|
(let ([xb (box 0)]
|
|
[yb (box 0)])
|
|
(send canvas get-virtual-size xb yb)
|
|
(values (unbox xb) (unbox yb))))
|
|
|
|
(define/override (queue-backing-flush)
|
|
;; Re-enable expose events so that the queued
|
|
;; backing flush will be handled:
|
|
(end-delay)
|
|
(send canvas queue-backing-flush))
|
|
|
|
(define/override (flush)
|
|
(send canvas flush))
|
|
|
|
(define/override (request-delay)
|
|
(request-flush-delay (send canvas get-flush-window)))
|
|
(define/override (cancel-delay req)
|
|
(cancel-flush-delay req))))
|
|
|
|
(define (do-backing-flush canvas dc win)
|
|
(send dc on-backing-flush
|
|
(lambda (bm)
|
|
(let ([w (box 0)]
|
|
[h (box 0)])
|
|
(send canvas get-client-size w h)
|
|
(let ([cr (gdk_cairo_create win)])
|
|
(backing-draw-bm bm cr (unbox w) (unbox h))
|
|
(cairo_destroy cr))))))
|