用 DrScheme 画简单的图形

发布于 2022-08-05 00:01:17 字数 2188 浏览 10 评论 9

SICP 的 2.2.4 节描述了一种图形语言,并假定用户的 scheme 环境提供了一个 draw-line 画线原语。scheme 环境很多,有一些提供了作图支持。虽然没有图形支持也可以完成此节的习题--只要写出基于 draw-line 的代码就可以了,但实践一下还是有好处的。

下面我将介绍一点  DrScheme 的画图功能,只涉及做 SICP 习题所需要的部分。

  • import 图形库


    1. (require scheme/class scheme/gui/base)
    复制代码
  • 建立一个窗体


    1. (define frame (new frame% (label "painter")
    2.                    (width 320)
    3.                    (height 240)))
    复制代码

    注意,上面的 frame 和书中的 frame 或框架不是同一个概念。DrScheme 的图形库基于 wxwindow,frame 的含义指一般的窗口。
  • 在窗体中加入一张画布


    1. (define canvas
    2.   (new canvas% (parent frame)
    3.        (paint-callback
    4.         (lambda (canvas dc) (draw-face dc)))))
    复制代码

    上面的代码指明了 canvas 的 parent 是 frame.
  • 显示窗口


    1. (send frame show #t)
    复制代码

    现在运行代码,可以得到一个空白窗口,之后就可以画图了。
  • 获取画布和生成画笔

    1. (define dc (send canvas get-dc))
    2. (define pen (make-object pen% "BLACK" 1 'solid))
    3. (define (draw-face dc)
    4.   (send dc set-pen pen)  )
    复制代码

    用 (send convas get-dc)  获取作图区域,然后可以用 send 向 dc 发送各种作图消息。
  • 画一条直线


    1. (send dc draw-line 0 0 320 240)
    复制代码

    try.drawline.png (4.7 KB, 下载次数: 23)

    下载附件

    2008-10-04 22:10 上传



    [ 本帖最后由 win_hate 于 2008-10-4 23:58 编辑 ]

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(9

怪我太投入 2022-08-17 21:17:42

原帖由 zft19 于 2008-10-18 13:16 发表
能画曲线吗?

DrScheme 提供的库能画曲线。

沫离伤花 2022-08-17 20:56:57

能画曲线吗?

情话墙 2022-08-17 20:07:42

把书上的其它几个图都画出来了。

  • wave4

    wave4.png (6.57 KB, 下载次数: 28)

    下载附件

    2008-10-07 18:48 上传


  • right split

    right.split.png (17.06 KB, 下载次数: 22)

    下载附件

    2008-10-07 18:48 上传


  • up split

    up.split.png (13.26 KB, 下载次数: 19)

    下载附件

    2008-10-07 18:48 上传


  • corner split

    corner.split.png (22.11 KB, 下载次数: 22)

    下载附件

    2008-10-07 18:48 上传


  • square limit

    square.limit.png (32.12 KB, 下载次数: 29)

    下载附件

    2008-10-07 18:48 上传


[ 本帖最后由 win_hate 于 2008-10-7 18:51 编辑 ]

许久 2022-08-17 19:20:20

原帖由 MMMIX 于 2008-10-6 22:38 发表

这个貌似拿透明的坐标纸来搞比较方便点

还没见过这么高级的装备呢,

[ 本帖最后由 win_hate 于 2008-10-7 18:49 编辑 ]

演多会厌 2022-08-17 19:02:13

原帖由 win_hate 于 2008-10-6 21:15 发表
我可是拿尺子按书上的图一个点一个点量的,苦。

这个貌似拿透明的坐标纸来搞比较方便点

打小就很酷 2022-08-17 00:04:48

注意那个 squash。以前看《西游记》的时候,那些妖精就是这样飞的,不知该剧的美工是不是看过 SICP?

污味仙女 2022-08-16 19:19:44

终于把那个邪恶的 wave 画出来了。我可是拿尺子按书上的图一个点一个点量的,苦。

  1. (define (list->segment xs)
  2.   (define (f r x xs)
  3.     (if (null? xs) r
  4.         (f (cons (make-segment x (car xs)) r) (car xs) (cdr xs))))
  5.   (f '() (car xs) (cdr xs)))

复制代码

  1. (define wave
  2.   (let ((v1 (list (make-vect 0 26)
  3.                   (make-vect 6 17)
  4.                   (make-vect 12 25)
  5.                   (make-vect 14 21)
  6.                   (make-vect 10 0)))
  7.         (v2 (list (make-vect 16 0)
  8.                   (make-vect 21 13)
  9.                   (make-vect 25 0)))
  10.         (v3 (list (make-vect 31 0)
  11.                   (make-vect 25 19)
  12.                   (make-vect 41 6)))
  13.         (v4 (list (make-vect 41 15)
  14.                   (make-vect 31 27)
  15.                   (make-vect 25 27)
  16.                   (make-vect 27 35)
  17.                   (make-vect 25 41)))
  18.         (v5 (list (make-vect 16 41)
  19.                   (make-vect 14 35)
  20.                   (make-vect 16 27)
  21.                   (make-vect 12 27)
  22.                   (make-vect 6 25)
  23.                   (make-vect 0 35))))                  
  24.     (let ((segments (map (lambda (z) (list->segment
  25.                                      (map (lambda (x) (map (lambda (y) (/ y 41.0)) x)) z)))
  26.                         (list v1 v2 v3 v4 v5))))
  27.       (segments->painter (foldr append '() segments)))))

复制代码

wave:

wave.png (5.32 KB, 下载次数: 21)

下载附件

2008-10-06 21:15 上传

flip, beside:

flip.beside.png (6.86 KB, 下载次数: 24)

下载附件

2008-10-06 21:15 上传

squash:

xie.png (6.26 KB, 下载次数: 17)

下载附件

2008-10-06 21:15 上传

[ 本帖最后由 win_hate 于 2008-10-6 21:19 编辑 ]

压抑⊿情绪 2022-08-15 01:35:50

要做 SICP 中的相关习题,上面的知识足够了。如果想继续深入,可以看 PLT Scheme 的帮助文档:

GUI: PLT Graphics Toolkit.

天邊彩虹 2022-08-06 00:59:13
  • SICP 上的 draw-line 接受两个点(向量),画出两点间的连线。我们可以定义一个同样接口的函数 Draw-line
    1. (define (Draw-line w v)
    2.   (send dc draw-line (xcor-vect w) (ycor-vect w)
    3.         (xcor-vect v) (ycor-vect v)))

    复制代码

    并把 segments->painter 中的 draw-line 替换为我们的 Draw-line. 注意,这里的 Draw-line 使用了“全局变量” dc.

  • 其中一个题目,画家 "X",为了看出效果,我在 "X" 上多加了一条线。
    1. (define X
    2.   (let* ((y (make-segment (make-vect 0 0) (make-vect 1 1)))
    3.          (z (list (make-segment (make-vect 0 1) (make-vect 1 0)) y))
    4.          (x (cons (make-segment (make-vect 0 0) (make-vect 1 0)) z)))
    5.     (segments->painter x)))

    复制代码

  • 标准框架
    1. (define normal-frame
    2.   (make-frame (make-vect 0 0) (make-vect 100 0) (make-vect 0 100)))

    复制代码

  • X 在标准框架内作画
    1. (X normal-frame)

    复制代码

    X.noraml.png (4.52 KB, 下载次数: 23)

    下载附件

    2008-10-04 22:23 上传


  • 反转的 "X",
    1. ((flip-vert X) normal-frame)

    复制代码

    flip.vert.png (4.52 KB, 下载次数: 23)

    下载附件

    2008-10-04 22:23 上传


  • 在另一个框架中的 "X"
    1. (let ((O (make-vect (- 160 (* (sqrt 2) 50)) 120))
    2.         (v1 (make-vect (* (sqrt 2) 50) (* (sqrt 2) -50)))
    3.         (v2 (make-vect (* (sqrt 2) 50) (* (sqrt 2) 50))))
    4.     (X (list O v1 v2)))

    复制代码

    X.transformed.png (4.14 KB, 下载次数: 20)

    下载附件

    2008-10-04 22:23 上传


[ 本帖最后由 win_hate 于 2008-10-4 22:28 编辑 ]

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文