Buchempfehlung
Windows System Programming
Windows System Programming
Das Kompendium liefert viele interessante Informationen zur Windows-Programmierung auf Englisch. [Mehr Infos...]
FreeBASIC-Chat
Es sind Benutzer im FreeBASIC-Chat online.
(Stand:  )
FreeBASIC bei Twitter
Twitter FreeBASIC-Nachrichten jetzt auch über Twitter erhalten. Follow us!

Code-Beispiel

Code-Beispiele » Grafik und Fonts

3D ohne Library (nur FB Befehle)

Lizenz:Erster Autor:Letzte Bearbeitung:
GPLMitgliedTJF 28.04.2011
Ausgabe
Vergrößern
Ausgabe

Dieses Beispiel zeigt einen Würfel als 3D-Darstellung, der im Raum gedreht wird. Die Darstellung erfolgt in drei Ansichten aus verschiedenen Blickrichtungen.

Die Berechnung der Raumpunkte und der perspektivischen Darstellung erfolgt unter Anwendung von Externer Link!homogenen Koordinaten (wie sie z. B. auch in OpenGl verwendet werden). Es wird hier jedoch auf die Verwendung von externen Bibliotheken verzichtet. Alle Berechnungen sind ausschließlich mit FreeBasic-Befehlen realisiert.

Der Code wurde in Kooperation mit owen im englischen Forum entwickelt, siehe Externer Link!FreeBasic.Forum.

#DEFINE MAT1 {{1.0,0.0,0.0,0.0},{0.0,1.0,0.0,0.0},{0.0,0.0,1.0,0.0},{0.0,0.0,0.0,1.0}}
#DEFINE E_Z TYPE<v3>(0.0, 0.0, 1.0)
#DEFINE ME_Z TYPE<v3>(0.0, 0.0, -1.0)
Const As Integer MAXPNTS = 50
Const As Single EPS = 1e-7
Const As Single PI = 4 * Atn(1)
Const As Single GRAD_RAD = PI / 180
Const As Single PIXEL_HOCH_ZU_BREIT = 1.0
Dim As Uinteger SCW = 800, SCH = 600
SCREENRES SCW, SCH, 8

SUB er(Byval T As String)
  If Len(T) Then ?T : Sleep
  End 0
End Sub

Sub NormM4(M() As Single)
  M(0, 0) = 1.0 : M(0, 1) = 0.0 : M(0, 2) = 0.0 : M(0, 3) = 0.0
  M(1, 0) = 0.0 : M(1, 1) = 1.0 : M(1, 2) = 0.0 : M(1, 3) = 0.0
  M(2, 0) = 0.0 : M(2, 1) = 0.0 : M(2, 2) = 1.0 : M(2, 3) = 0.0
  M(3, 0) = 0.0 : M(3, 1) = 0.0 : M(3, 2) = 0.0 : M(3, 3) = 1.0
End Sub

Sub MulMat(A() As Single, B() As Single, C() As Single)
  For i As Integer = 0 To 3
    C(0, i) = A(0,0)*B(0,i) + A(0,1)*B(1,i) + A(0,2)*B(2,i) + A(0,3)*B(3,i)
    C(1, i) = A(1,0)*B(0,i) + A(1,1)*B(1,i) + A(1,2)*B(2,i) + A(1,3)*B(3,i)
    C(2, i) = A(2,0)*B(0,i) + A(2,1)*B(1,i) + A(2,2)*B(2,i) + A(2,3)*B(3,i)
    C(3, i) = A(3,0)*B(0,i) + A(3,1)*B(1,i) + A(3,2)*B(2,i) + A(3,3)*B(3,i)
  Next
End Sub

Sub InvMat(M() As Single, R() As Single)
  Dim As Single d
  Dim As Integer i, j, i1, i2, i3, j1, j2, j3, fl = 1
  For i = 0 To 3
    i1 = (i + 1) And 3 : i2 = (i + 2) And 3 : i3 = (i + 3) And 3
    For j = 0 To 3
      j1 = (j + 1) And 3 : j2 = (j + 2) And 3 : j3 = (j + 3) And 3
      R(j, i) = M(i1,j1) * (M(i2,j2)*M(i3,j3) - M(i2,j3)*M(i3,j2)) + _
                M(i1,j2) * (M(i2,j3)*M(i3,j1) - M(i2,j1)*M(i3,j3)) + _
                M(i1,j3) * (M(i2,j1)*M(i3,j2) - M(i2,j2)*M(i3,j1))
      R(j, i) *= fl
      fl = -fl
    Next
    fl = -fl
  Next
  For i = 0 To 3 : d += M(0, i) * R(i, 0) : Next
  If Abs(d) < EPS Then er("InvMat: Matrix nicht regulaer")
  For i = 0 To 3
    For j = 0 To 3
      R(i, j) /= d
    Next
  Next
End Sub



Type v3
  Declare Function Norm() As Integer
  As Single x, y, z
End Type

Function v3.Norm() As Integer
  Dim As Single n = Sqr(x*x + y*y + z*z)
  If Abs(n) < EPS Then Return -1
  x /= n : y /= n : z /= n : Return 0
End Function


Type v2
  As Single x, y
End Type

Type vert
  As v3 crd, nrm
  As INTEGER c
End Type

Type ansicht
PUBLIC:
  Declare Sub Clr()
  Declare Sub Scr(Byval As Single, Byval As Single, Byval As Single, Byval As Single)
  Declare Sub MCtoDC(Byref V As v3, Byref X As Single, Byref Y As Single)
  Declare Sub Vie(Byval As Single, Byval As Single, Byval As Single, _
                  Byval As Single, Byval As Single, Byval As Single, _
                  Byval As Single, Byval As Single, Byval As Single, _
                  Byval As Single, Byval As Single, Byval As Single, _
                  Byval As Single, Byval As Single, Byval As Single)
  Declare Sub LocalInit()
  Declare Sub updateMats()

  As Single MC_WC(3, 3), MC_DC(3, 3)

PRIVATE:
  Declare Sub Init()
  Declare Sub SetV()
  Declare Sub SetS(Byval Lu As v2, Byval Ro As v2)
  Declare Sub CalcVRC_NPC()
  Declare Sub CalcWC_VRC()

  As Uinteger maxint
  As Integer frame, MaChe

  As Single lux, luy, rox, roy
  As Single YsubO, YsubU, SpanMemSize
  As Single VPD, FPD, BPD
  As Single xMin, yMin, xMax, yMax, xSize, ySize
  As Single xwinsize, ywinsize, xwinoffs, ywinoffs

  As v3 VRP, NRP, VUP
  As v3 eyePoint

  As Single WC_NPC(3, 3), MC_NPC(3, 3)
  As Single WC_MC(3, 3), WC_DC(3, 3), NPC_MC(3, 3), NPC_DC(3, 3)
  As Single WC_VRC(3, 3), VRC_NPC(3, 3), NPC_WC(3, 3)
End Type

Sub ansicht.Clr()
  Line(rox, roy) - (lux, luy), 1, BF
End Sub

Sub ansicht.Scr(Byval Lx As Single, Byval Uy As Single, _
                Byval Rx As Single, Byval Oy As Single)
  lux = Lx
  luy = Uy
  rox = Rx
  roy = Oy
  YsubO = CINT(roy)
  YsubU = CINT(roy - 1)
  SpanMemSize = CINT((rox + 1) * 2)
  ySize = xSize * PIXEL_HOCH_ZU_BREIT * (luy - roy) / (rox - lux)
End Sub

Sub ansicht.Vie(Byval Vrpx As Single, Byval Vrpy As Single, Byval Vrpz As Single, _
                Byval Nrpx As Single, Byval Nrpy As Single, Byval Nrpz As Single, _
                Byval Vupx As Single, Byval Vupy As Single, Byval Vupz As Single, _
                Byval Minx As Single, Byval Miny As Single, Byval SizeX As Single, _
                Byval Vpd_ As Single, Byval Fpd_ As Single, Byval Bpd_ As Single)
  VRP.x = Vrpx
  VRP.y = Vrpy
  VRP.z = Vrpz
  NRP.x = Nrpx
  NRP.y = Nrpy
  NRP.z = Nrpz
  VUP.x = Vupx
  VUP.y = Vupy
  VUP.z = Vupz
  xMin  = Minx
  yMin  = Miny
  xSize = SizeX
  VPD = Vpd_
  FPD = Fpd_
  BPD = Bpd_
End Sub

Sub ansicht.Init()
  YSubU = CINT(luy)
  xMax = xMin + xSize
  yMax = yMin + ySize * (luy - YsubO) / (luy - roy)
  SetV()
  SetS(TYPE<v2>(lux, YsubU), TYPE<v2>(rox, YsubO))
End Sub

Sub ansicht.CalcWC_VRC()
  Dim As v3 vpn, vup_
  Dim As Single scal
  WC_VRC(0, 0) = 1.0
  WC_VRC(0, 1) = VRP.x
  WC_VRC(0, 2) = VRP.y
  WC_VRC(0, 3) = VRP.z
  vpn.x = NRP.x - VRP.x
  vpn.y = NRP.y - VRP.y
  vpn.z = NRP.z - VRP.z
  If vpn.Norm() Then er("CalcWC_VRC: VRP und NRP sind identisch!")
  WC_VRC(3, 0) = 0.0
  WC_VRC(3, 1) = vpn.x
  WC_VRC(3, 2) = vpn.y
  WC_VRC(3, 3) = vpn.z
  scal = VUP.x * vpn.x + VUP.y * vpn.y + VUP.z * vpn.z
  vup_.x = VUP.x - scal * vpn.x
  vup_.y = VUP.y - scal * vpn.y
  vup_.z = VUP.z - scal * vpn.z
  If vup_.Norm() Then er("CalcWC_VRC: VUP liegt kollinear zu (VRP, NRP)!")
  WC_VRC(2, 0) = 0.0
  WC_VRC(2, 1) = vup_.x
  WC_VRC(2, 2) = vup_.y
  WC_VRC(2, 3) = vup_.z
  WC_VRC(1, 0) = 0.0
  WC_VRC(1, 1) = vup_.y * vpn.z - vup_.z * vpn.y
  WC_VRC(1, 2) = vup_.z * vpn.x - vup_.x * vpn.z
  WC_VRC(1, 3) = vup_.x * vpn.y - vup_.y * vpn.x
  eyePoint.x = vpn.x * VPD + VRP.x
  eyePoint.y = vpn.y * VPD + VRP.y
  eyePoint.z = vpn.z * VPD + VRP.z
End Sub

Sub ansicht.CalcVRC_NPC()
  Dim As Single k, l
  k = (VPD - BPD) / VPD
  l = (FPD - BPD) / (VPD - FPD)
  VRC_NPC(0, 0) = 0.0
  VRC_NPC(0, 1) = k * xMin
  VRC_NPC(0, 2) = k * yMin
  VRC_NPC(0, 3) = BPD
  VRC_NPC(1, 0) = 0.0
  VRC_NPC(1, 1) = k * (xMax - xMin)
  VRC_NPC(1, 2) = 0.0
  VRC_NPC(1, 3) = 0.0
  VRC_NPC(2, 0) = 0.0
  VRC_NPC(2, 1) = 0.0
  VRC_NPC(2, 2) = k * (yMax - yMin)
  VRC_NPC(2, 3) = 0.0
  VRC_NPC(3, 0) = 1.0
  VRC_NPC(3, 1) = 0.0
  VRC_NPC(3, 2) = 0.0
  VRC_NPC(3, 3) = l * VPD
End Sub

Sub ansicht.SetV()
  CalcWC_VRC()
  CalcVRC_NPC()
  MulMat(VRC_NPC(), WC_VRC(), NPC_WC())
  InvMat(NPC_WC(), WC_NPC())
  MaChe = 1
End Sub

Sub ansicht.SetS(Byval Lu As v2, Byval Ro As v2)
  Clr()
  xwinsize = Ro.x - Lu.x
  xwinoffs = CINT(Lu.x)
  ywinsize = Ro.y - Lu.y
  ywinoffs = CINT(Lu.y)
  NormM4(NPC_DC())
  NPC_DC(1, 1) = xwinsize
  NPC_DC(2, 2) = ywinsize
  NPC_DC(3, 3) = maxint
  NPC_DC(0, 1) = xwinoffs
  NPC_DC(0, 2) = ywinoffs
  MaChe = 1
End Sub

Sub ansicht.LocalInit()
  YSubU = CINT(luy)
  xMax = xMin + xSize
  yMax = yMin + ySize * (luy - YsubO) / (luy - roy)
  SetV()
  SetS(TYPE<v2>(lux, YsubU), TYPE<v2>(rox, YsubO))
End Sub

Sub ansicht.updateMats()
  If MaChe = 0 Then Exit Sub
  Dim As Single mc_npc(3, 3)
  MulMat(MC_WC(), WC_NPC(), mc_npc())
  MulMat(mc_npc(), NPC_DC(), MC_DC())
  InvMat(MC_WC(), WC_MC())
  InvMat(mc_npc(), NPC_MC())
  MaChe = 0
End Sub

Sub ansicht.MCtoDC(Byref V As v3, Byref X As Single, Byref Y As Single)
  Dim As Single w
  w = MC_DC(0, 0) + V.x * MC_DC(1, 0) + V.y * MC_DC(2, 0) + V.z * MC_DC(3, 0)
  If Abs(w) < EPS Then er("MC_DC: Punkt nicht definiert!")
  X = (MC_DC(0, 1) + V.x * MC_DC(1, 1) + V.y * MC_DC(2, 1) + V.z * MC_DC(3, 1)) / w
  Y = (MC_DC(0, 2) + V.x * MC_DC(1, 2) + V.y * MC_DC(2, 2) + V.z * MC_DC(3, 2)) / w
End Sub

SUB Rotate(Byval V As ansicht Ptr, BYVAL A AS INTEGER, BYVAL W AS SINGLE)
  DIM AS SINGLE s = SIN(W * GRAD_RAD), c = COS(W * GRAD_RAD)
  DIM AS Single mcdc(3, 3), M(3, 3) = MAT1
  WITH *V
    FOR z AS INTEGER = 0 TO 3
      FOR s AS INTEGER = 0 TO 3
        mcdc(z, s) = .MC_DC(z, s)
      NEXT
    NEXT
    SELECT CASE AS CONST A
    CASE 1 : M(2, 2) = c : M(2, 3) = s : M(3, 2) = -s : M(3, 3) = c
    CASE 2 : M(1, 1) = c : M(1, 3) = s : M(3, 1) = -s : M(3, 3) = c
    CASE 3 : M(1, 1) = c : M(1, 2) = s : M(2, 1) = -s : M(2, 2) = c
    END SELECT
    MulMat(M(), mcdc(), .MC_DC())
  END WITH
END SUB

Dim Shared mycube(8) As vert
Dim As Integer i,j
For i = 1 To 8
        With mycube(i).crd
                Read .x,.y,.z
        End With
Next

Sub DrawCube(Byval A As ansicht Ptr)
  Dim As Single x(8), y(8)
  For i As Integer = 1 To 8
    (*A).MCtoDC(mycube(i).crd, x(i), y(i))
  Next

  (*A).Clr()
  Line (x(1), y(1)) - (x(2), y(2)),14
  Line - (x(3), y(3)),14
  Line - (x(4), y(4)),14
  Line - (x(1), y(1)),14
  Line - (x(5), y(5)),12
  Line - (x(6), y(6)),13
  Line - (x(7), y(7)),13
  Line - (x(8), y(8)),13
  Line - (x(5), y(5)),13
  Line (x(2), y(2)) - (x(6), y(6)),12
  Line (x(3), y(3)) - (x(7), y(7)),12
  Line (x(4), y(4)) - (x(8), y(8)),12
  Circle (x(1), y(1)),10,15
End Sub

DIM AS INTEGER x1 = 10, x2 = SCW\2-5, x3 = x2+10, x4 = SCW-10
DIM AS INTEGER y1 = 10, y2 = SCH\2-5, y3 = y2+10, y4 = SCH-10
Dim As Single Mnx = -1.5, Mny = -1.0, Six = 3.0
Dim As Single Vpd = 6, Fpd = 1.5, Bpd = -1.5

Dim As ansicht PTR Ans = New ansicht[3]

Ans[0].Vie(0.0, 0.0, 0.0, _
           0.0, -1.0, 0.0, _
           0.0, 0.0, 1.0, _
          Mnx, Mny, Six, Vpd, Fpd, Bpd)
Ans[0].Scr(x1, y2, x2, y1)

Ans[1].Vie(0.0, 0.0, 0.0, _
           -1.0, 0.0, 0.0, _
           0.0, 0.0, 1.0, _
          Mnx, Mny, Six, Vpd, Fpd, Bpd)
Ans[1].Scr(x3, y2, x4, y1)

Ans[2].Vie(0.0, 0.0, 0.0, _
           0.0, 0.0, 1.0, _
           0.0, 1.0, 0.0, _
          Mnx, Mny, Six, Vpd, Fpd, Bpd)
Ans[2].Scr(x1, y4, x2, y3)

For i = 0 TO 2
  NormM4(Ans[i].MC_WC())
  Ans[i].LocalInit()
  Ans[i].updateMats()
Next

Dim h As String
Dim As Integer axis_x,axis_y,axis_z,ax
Dim As Single ro,st=1.0
Print "rotate axes using keys (y,h / u,j / i,k)"
Do
  Locate 50,60
  Print "axis x=";axis_x;"  "
  Locate 51,60
  Print "axis y=";axis_y;"  "
  Locate 52,60
  Print "axis z=";axis_z;"  "
  For i = 0 To 2
    DrawCube(@Ans[i])
  Next
  Sleep : Do : h=InKey : Loop Until Len(h)
  Select Case h
    Case Chr(27), Chr(255) + "k"
      Exit Do
    Case "y"
      axis_z=(axis_z+st) Mod 360 : ax=3 : ro=st
    Case "h"
      axis_z=(axis_z-st) Mod 360 : ax=3 : ro=-st
    Case "u"
      axis_y=(axis_y+st) Mod 360 : ax=2 : ro=st
    Case "j"
      axis_y=(axis_y-st) Mod 360 : ax=2 : ro=-st
    Case "i"
      axis_x=(axis_x+st) Mod 360 : ax=1 : ro=st
    Case "k"
      axis_x=(axis_x-st) Mod 360 : ax=1 : ro=-st
    Case Else
      ax=0 : ro=0.0 : Continue Do
  End Select
  For i = 0 To 2
    Rotate(@Ans[i],ax,ro)
  Next : While len(inkey) : Wend
Loop
Delete[] Ans
End

'cube
Data -1, 1, 1
Data -1,-1, 1
Data  1,-1, 1
Data  1, 1, 1

Data -1, 1,-1
Data -1,-1,-1
Data  1,-1,-1
Data  1, 1,-1

Zusätzliche Informationen und Funktionen
  • Das Code-Beispiel wurde am 25.09.2010 von MitgliedTJF angelegt.
  • Die aktuellste Version wurde am 28.04.2011 von MitgliedTJF gespeichert.
  Bearbeiten Bearbeiten  

  Versionen Versionen