免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 5385 | 回复: 9
打印 上一主题 下一主题

用 DrScheme 画简单的图形 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-10-04 22:10 |只看该作者 |倒序浏览
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)
    复制代码



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

论坛徽章:
0
2 [报告]
发表于 2008-10-04 22:23 |只看该作者
  • 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",


    1. ((flip-vert X) normal-frame)
    复制代码


  • 在另一个框架中的 "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)))
    复制代码




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

论坛徽章:
0
3 [报告]
发表于 2008-10-04 22:31 |只看该作者
要做 SICP 中的相关习题,上面的知识足够了。如果想继续深入,可以看 PLT Scheme 的帮助文档:

GUI: PLT Graphics Toolkit.

论坛徽章:
0
4 [报告]
发表于 2008-10-06 21:15 |只看该作者
终于把那个邪恶的 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:


flip, beside:


squash:


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

论坛徽章:
0
5 [报告]
发表于 2008-10-06 21:23 |只看该作者
注意那个 squash。以前看《西游记》的时候,那些妖精就是这样飞的,不知该剧的美工是不是看过 SICP?

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
6 [报告]
发表于 2008-10-06 22:38 |只看该作者
原帖由 win_hate 于 2008-10-6 21:15 发表
我可是拿尺子按书上的图一个点一个点量的,苦。


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

论坛徽章:
0
7 [报告]
发表于 2008-10-07 18:44 |只看该作者
原帖由 MMMIX 于 2008-10-6 22:38 发表

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


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

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

论坛徽章:
0
8 [报告]
发表于 2008-10-07 18:48 |只看该作者
把书上的其它几个图都画出来了。

  • wave4

  • right split

  • up split

  • corner split

  • square limit



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

论坛徽章:
0
9 [报告]
发表于 2008-10-18 13:16 |只看该作者
能画曲线吗?

论坛徽章:
0
10 [报告]
发表于 2008-10-18 14:24 |只看该作者
原帖由 zft19 于 2008-10-18 13:16 发表
能画曲线吗?


DrScheme 提供的库能画曲线。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP