×

学习笔记,VB.NET实现DirectDraw9 (2) 动画

Kalet Kalet 发表于2009-03-20 12:00:13 浏览363 评论0

抢沙发发表评论

===========日记================
发现最近比较懒惰,代码稍微长那么一点,就不想看了.还是看书比较好.
考虑做游戏,但是目前所学的知识还不够.革命尚未成功,同志们仍需努力啊学习笔记,VB.NET实现DirectDraw9 (2) 动画学习笔记,VB.NET实现DirectDraw9 (2) 动画
===========End 日记=============

今天是完成全屏幕动画,1024x768,外加多层绘制(鼠标位置就不值得一提了)

先回想一下我们平时是怎样画图的.(应该是做图)
1 准备一张纸,这个要给别人看的 (PrimarySurface)        
2 一支笔   (Draw方法)
然后自由发挥就可以了 (找坐标,画..)

这样是可以的,你的观众(坐在显示器前面的人)不但能看见你的绘制的作品,而且连你如何绘制的都能看见(也许你正在用描点法画图...). 很明显,我们不希望观众看见我们绘图时的“优雅姿态“.我们要的是速度以及效果,这么慢就不叫DirectDraw了

方案2
1 准备一张纸,这个是给观众欣赏的(PrimarySurface)
2 再准备一个成型的作品,比如达芬奇画过的鸡蛋 (Surface)
3 画笔,尖刀,胶水
大概你知道我要做什么了吧,用粘贴的办法自然要快于描点.但是,观众们依然可以知道你是粘贴上去的!当演示一幅一幅的动画的时候,往往会粘贴的到处都是.....

方案3
1 找个工作台,请观众坐下慢慢看当前的图像(PrimarySurface)
2 自己再另外找一张或者更多的画板 (BackSurface)
3 各种工具,只要能方便使用就好 (DrawFast DrawRect DrawText DrawCircle .....)
4 请一位助手帮忙更换工作台的作品.(PrimarySurface.flip)
这样您可以解放出来,再也不用担心您的观众看到您绘制的过程了.虽然他们可以“猜测“您是如何绘制的.我们这样做,在backSurface上面用任意的快速方法做图,当然可以从其他Surface上面复制,然后粘贴到BackSurface,提交给PrimarySurface,让他Flip,用户看到的就是您绘制完之后的结果(PrimarySurface),当观众对您的作品赞不绝口的时候,您正在刚才flip之后的画板上修改,然后把修改之后的这个画板再flip...

这样,两张画板解决了问题.同时只有1个是观众正在看的,另外1个是您手头上正在画的,您的助手(PrimarySurface)很聪明,不会搞错,所以你只要安心画当前的画板(BackBuffer)就够了

强调一下,您绘制的总是Backbuffer,flip方法会把您的backbuffer呈现到PrimaryBuffer,然后绘制,呈现,绘制,呈现...
或者您理解为复制到PrimarySurface也可(实际上是内存的块移动)

大体步骤跟前文说得一致,仅仅是多生命了几个Surface用于复制,源代码不过才200行,主要代码也就不到100行,里面有比较详细的注释.所以再这里就没有必要解释了

==================================================

Imports Microsoft.DirectX.DirectDraw

 

Public Class Form1

    Inherits System.Windows.Forms.Form

    Private Structure PointAPI

        Public x As Integer

        Public y As Integer

    End Structure

    '''API用习惯了....也就继续用吧...

    Private Declare Function GetCursorPos Lib "user32" (ByRef lpPoint As PointAPI) As Integer

 

    Dim dev As New Device(CreateFlags.Default)

 

    Dim PS As Surface           'primarySurface

    Dim BS As Surface           'BackSurface

    Dim S1 As Surface           'Surface1 用于储存图像的,想象成一个BMP就行了

    Dim S2 As Surface           'Surface2 同上

    '''分别对应上面的四个Surface

    Dim desc1 As SurfaceDescription

    Dim desc2 As SurfaceDescription

    Dim desc3 As SurfaceDescription

    Dim desc4 As SurfaceDescription

    ''分别对应上面的Surface

    Dim RP As Rectangle

    Dim RB As Rectangle

    Dim R1 As Rectangle

    Dim R2 As Rectangle

 

    '''计时器相关

    Dim tLast As TimeSpan

    Dim fps As String

    Dim tfp As Integer = 0

    Dim mytime As Date = DateTime.Now

    Dim ts As New TimeSpan

    Dim qiqi As Double

    '''游戏控制

    Dim running As Boolean = False

    Dim TT As Threading.Thread

    '''鼠标位置

    Dim M As PointAPI

    '''需要读取的图像

    Const FN1 = "d:\nerv.bmp"

    Const FN2 = "d:\logo.bmp"

 

 

#Region " Windows 窗体设计器生成的代码 "

 

   Public Sub New()

        MyBase.New()

 

        '该调用是 Windows 窗体设计器所必需的。

        InitializeComponent()

 

        '在 InitializeComponent() 调用之后添加任何初始化

 

    End Sub

 

    '窗体重写 dispose 以清理组件列表。

    Protected Overloads Overrides Sub Dispose(ByVal disposing As Boolean)

        If disposing Then

            If Not (components Is Nothing) Then

                components.Dispose()

            End If

        End If

        MyBase.Dispose(disposing)

    End Sub

 

    'Windows 窗体设计器所必需的

    Private components As System.ComponentModel.IContainer

 

    '注意: 以下过程是 Windows 窗体设计器所必需的

    '可以使用 Windows 窗体设计器修改此过程。

    '不要使用代码编辑器修改它。

    Friend WithEvents Label1 As System.Windows.Forms.Label

    <System.Diagnostics.DebuggerStepThrough()> Private Sub InitializeComponent()

        Me.Label1 = New System.Windows.Forms.Label

        Me.SuspendLayout()

        '

        'Label1

        '

        Me.Label1.Location = New System.Drawing.Point(64, 64)

        Me.Label1.Name = "Label1"

        Me.Label1.Size = New System.Drawing.Size(80, 24)

        Me.Label1.TabIndex = 0

        Me.Label1.Text = "init&&play"

        '

        'Form1

        '

        Me.AutoScaleBaseSize = New System.Drawing.Size(6, 14)

        Me.ClientSize = New System.Drawing.Size(292, 273)

        Me.Controls.Add(Me.Label1)

        Me.Name = "Form1"

        Me.Text = "Form1"
学习笔记,VB.NET实现DirectDraw9 (2) 动画
        Me.ResumeLayout(False)

 

    End Sub

 

#End Region

 

    Private Sub Label1_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Label1.Click

        initDDraw()                 '初始化

        LoadSurfaces()              '读取图像

        If TT Is Nothing Then       '用于绘制的线程

            TT = New Threading.Thread(AddressOf mainloop)

            running = True

            TT.Start()

        End If

    End Sub

 

    Sub initDDraw()

        dev.SetCooperativeLevel(Me, CooperativeLevelFlags.FullscreenExclusiveAllowModex)

        dev.SetDisplayMode(1024, 768, 16, 0, False)

 

        'Primarybuffer的设置

        desc1 = New SurfaceDescription

        desc1.SurfaceCaps.VideoMemory = True

        desc1.SurfaceCaps.PrimarySurface = True

        desc1.SurfaceCaps.Flip = True

        desc1.SurfaceCaps.Complex = True

        desc1.BackBufferCount = 1

 

        PS = New Surface(desc1, dev)

 

        desc2 = New SurfaceDescription

        desc2.SurfaceCaps.BackBuffer = True

        BS = PS.GetAttachedSurface(desc2.SurfaceCaps)

 

        BS.ForeColor = System.Drawing.Color.Blue

        BS.FontTransparency = True

        ''OK只要把PrimaryBuffer跟BackBuffer设置好就算初始化完成,其他的图像都往上贴

 

 

    End Sub

 

    Sub LoadSurfaces()

        '''读取其他图层

        ''' 不要把Surface想得那么神秘,就是一个BMP附加上了更多的属性而已,这样理解简单很多

        desc3 = New SurfaceDescription

        desc3.SurfaceCaps.OffScreenPlain = True             '幕后的

        desc3.Height = BS.SurfaceDescription.Height         '大小

        desc3.Width = BS.SurfaceDescription.Width

        S1 = New Surface(FN1, desc3, dev)                   '读取

 

        desc4 = New SurfaceDescription

        desc4.SurfaceCaps.OffScreenPlain = True             '直接读

        S2 = New Surface(FN2, desc4, dev)

 

        Dim key As ColorKey                                 '用来设置透明的

        key.ColorSpaceHighValue = 0

        key.ColorSpaceLowValue = 0

        S2.SetColorKey(ColorKeyFlags.SourceDraw, key)       '设置透明色

        '''''设置矩形位置信息

        '''

        RB.Width = BS.SurfaceDescription.Width

        RB.Height = BS.SurfaceDescription.Height

 

        R1.Width = S1.SurfaceDescription.Width

        R1.Height = S1.SurfaceDescription.Height

 

        R2.Width = S2.SurfaceDescription.Width

        R2.Height = S2.SurfaceDescription.Height

        R2.X = 100

        R2.Y = 100

 

    End Sub

 

    Sub mainloop()

        While (running = True)                                      '''如果游戏没有结束

            blt()                                                   '''主要绘制过程

            tfp += 1                                                '''fps++

            If tfp = 200 Then                                       '''200次的时候计算时间

                tfp = 0

                ts = (DateTime.Now.Subtract(mytime))

                mytime = DateTime.Now

                If ts.TotalSeconds <> 0 Then

                    qiqi = 200 / (ts.TotalSeconds)

                    fps = qiqi.ToString("##.##") + "F c"

                End If

 

            End If

 

            TT.Sleep(10)                                            '''硬性规定休息一下,当然可以去掉发挥MAX速度

        End While

    End Sub

 

    Sub blt()

        If BS Is Nothing Then Exit Sub

        GetMousePos()       '得到鼠标位置

        '''下面的就是利用BackSurface的方法来绘制了,随便画,呵呵

        BS.DrawFast(0, 0, S1, R1, DrawFastFlags.Wait)

        BS.DrawText(10, 10, "1024x768 Frames per Second " + fps, False)

        BS.DrawText(10, 30, "当前位置:X=" + M.x.ToString + ",Y=" + M.y.ToString, False)

        BS.DrawText(10, 50, "ESC 退出", False)

        '''画出贴图

        BS.DrawFast(M.x, M.y, S2, DrawFastFlags.SourceColorKey Or DrawFastFlags.Wait)

        '''顺便在鼠标的位置打出它的坐标

        BS.DrawText(M.x + 50, M.y, "(" + M.x.ToString + "," + M.y.ToString + ")", False)

        '''关键一步,翻转,而且不要使用wait,这样可能会丢帧,但是不影响速度

        PS.Flip(Nothing, FlipFlags.NoVSync)

 

    End Sub

 

    Private Sub Form1_KeyUp(ByVal sender As Object, ByVal e As System.Windows.Forms.KeyEventArgs) Handles MyBase.KeyUp

        If e.KeyCode = Keys.Escape Then End 'ESC退出

    End Sub

    Sub GetMousePos()

        GetCursorPos(M)             '调用API,得到鼠标位置

    End Sub

End Class

 
=====================================
很多代码跟前面说得一样,大体思路也是跟我举的例子(方案3)一致

通过代码.可以了解具体步骤

DirectDraw再往下就是图像的透明度计算(这个有公式的),大小变换,旋转.具体请参考现成的代码以及公式,我计算机图形学没有学过,估计是很困难.而且DDraw还有很多内置的方法可以提供简单的转换方式,这个估计需要查阅英文原版的DirectDraw7才能找到相应的方法了...

OK DirectDraw到此结束,如果可能,还会写一个应用DirectDraw的Demo,但是估计就不会这样具体说明了,呵呵.
学习笔记,VB.NET实现DirectDraw9 (2) 动画
终于可以安心研究那个混音了

群贤毕至

访客