128
Tags:
delphi
Skrevet af
Bruger #1474
@ 24.11.2003
Denne funktion kan vi erstatte med vores tidligere glTexImage2D(). Yderligere skal vi ændre vores gl_Nearest parametere i vores to glTexParameteri() til MipMap parametere: gl_Linear_Mipmap_Nearest eller gl_Linear_Mipmap_Linear. I stedet for at slette vores gamle kode kan vi udvide vores LoadTexture funktion med en Boolean. Hvis Boolean er True, vil der blive generet MipMaps ellers vil der blive anvendt vores standard projektering.
function LoadTexture(FileName : string; MipMap : Boolean) : Integer;
var
Bitmap : TBitmap;
X, Y, I : Integer;
A : array of TRGB;
begin
Bitmap := TBitmap.Create;
Bitmap.LoadFromFile(FileName);
SetLength(A,0);
for Y := Bitmap.Height-1 downto 0 do
for X := 0 to Bitmap.Width-1 do
begin
SetLength(A,High(A)+2);
I := ColorToRGB(Bitmap.Canvas.Pixels[X,Y]);
A[High(A)].Red := GetRValue(I);
A[High(A)].Green := GetGValue(I);
A[High(A)].Blue := GetBValue(I);
end;
glGenTextures(1, I);
glBindTexture(gl_Texture_2D, I);
glTexEnvi(gl_Texture_Env, gl_Texture_Env_Mode, gl_Modulate);
glTexParameteri(gl_Texture_2D, GL_Texture_Wrap_S, gl_Repeat);
glTexParameteri(gl_Texture_2D, GL_Texture_Wrap_T, gl_Repeat);
if MipMap then
begin
//MipMapping
glTexParameteri(gl_Texture_2D, gl_Texture_Mag_Filter, gl_Linear_MipMap_Nearest);
glTexParameteri(gl_Texture_2D, gl_Texture_Min_Filter, gl_Linear_Mipmap_Linear);
gluBuild2DMipmaps(gl_Texture_2D, 3, Bitmap.Width, Bitmap.Height, gl_RGB, gl_Unsigned_Byte, A);
end
else
begin
//Standard
glTexParameteri(gl_Texture_2D, gl_Texture_Mag_Filter, gl_Nearest);
glTexParameteri(gl_Texture_2D, gl_Texture_Min_Filter, gl_Nearest);
glTexImage2D(gl_Texture_2D, 0, 3, Bitmap.Width, Bitmap.Height, 0, gl_RGB, gl_Unsigned_Byte, A);
end;
glEnable(gl_Texture_2D);
Bitmap.Free;
//Retunere ID
Result := I;
end;
I vores FormCreate procedure kan vi spørge efter hvilken form for mapping der skal anvendes ved at bruge MessageDlg().
procedure TForm1.FormCreate(Sender: TObject);
var
B : Boolean;
begin
Context := SetupContext(Form1);
if MessageDlg('Anvend MipMapping?',mtCustom,[mbYes,mbNo],0) = mrYes then
B := True
else
B := False;
Gulv_ID := LoadTexture('C:\\Gulv.Bmp',B);
Vaeg_ID := LoadTexture('C:\\Væg.Bmp',B);
Loft_ID := LoadTexture('C:\\Loft.Bmp',B);
//Skjuler cursoren
ShowCursor(False);
//ShowMessage(glGetString(gl_Vendor));
Application.OnIdle := Idle;
end;
Prøv at kompile og se selv forskellen.
Standard
MipMap
OpenGL er også i stand til at shade vores geometri, så det ser ud som om der findes lyskilder i vores 3D verden. OpenGL kan beregne op til 8 lyskilder ad gangen. Umiddelbart lyder det ikke af alverden. Men hvis du ser på nogle af de mange spil der findes på markedet, er det faktisk sjældent at du kan se mere end 8 lyskilder ad gangen på din skærm. Når vi snakker om lyskilder, i en kunstigt skabt 3D verden, er det meget naturligt at vi også snakker om Normals. De to er tæt knyttet til hinanden. Normals er en vektor der bliver defineret med OpenGL funktionen: glNormal3f(). Denne funktion forventer tre koordinater (X,Y,Z) ligesom vores glVertex3f(). OpenGL bruger vores normal til at beregne en korrekt shading af en overflade i forhold til lyskilden. Det er derfor vigtigt at fokusere på vinklen mellem vores normal og dens flade. Som standard tager OpenGL ikke højde for, hvor lange vores normals er. Vi kan derfor gøre vores normals så lange som vi har lyst til. Men vi kan gøre OpenGL hurtigere til at beregne shadingen, hvis vores normals er kun 1 unit lang. Alt vi skal gøre er at fortælle OpenGL at vi vores normals skal normaliseres med glEnable(gl_Normalze), og selvfølig gøre vore normals 1 unit lang. Lad os derfor implementere vores normals til vores geometri.
procedure TForm1.FormPaint(Sender: TObject);
begin
glClearColor(0,0,0,0);
glClear(gl_Color_Buffer_Bit or gl_Depth_Buffer_Bit);
glEnable(gl_Depth_Test);
//Normalisere vores normals
glEnable(gl_Normalize);
glPushMatrix;
with Rotation do
begin
glRotatef(X,1,0,0);
glRotatef(Y,0,1,0);
glRotatef(Z,0,0,1);
end;
with Translation do
glTranslatef(X,Y,Z);
//Aktivere mit gulv texture
glBindTexture(gl_Texture_2D, Gulv_ID);
//Tegner mit gulv
glBegin(gl_Quads);
//Definere en normal for gulvet, der peger lodret op ad!
glNormal3f(0, 1, 0);
glTexCoord2f(0,4);
glVertex3f(-50, -3, -50);
glTexCoord2f(0,0);
glVertex3f(-50, -3, +50);
glTexCoord2f(4,0);
glVertex3f(+50, -3, +50);
glTexCoord2f(4,4);
glVertex3f(+50 ,-3 ,-50);
glEnd;
//Aktivere mit væg texture
glBindTexture(gl_Texture_2D, Vaeg_ID);
//Tegner min væg
glBegin(gl_Quads);
//DFefinere en normal for væggen, der peger imod os (ud fra væggen)
glNormal3f(0, 0, 1);
glTexCoord2f(0,1);
glVertex3f(-50, -3, -50);
glTexCoord2f(0,0);
glVertex3f(-50, +22, -50);
glTexCoord2f(4,0);
glVertex3f(+50, +22, -50);
glTexCoord2f(4,1);
glVertex3f(+50, -3, -50);
glEnd;
//Aktivere mit loft texture
glBindTexture(gl_Texture_2D, Loft_ID);
//Tegner mit loft
glBegin(gl_Quads);
//Definere en normal for loftet, der peger ned mod gulvet
glNormal3f(0,-1, 0);
glTexCoord2f(0,4);
glVertex3f(-50, +22, -50);
glTexCoord2f(0,0);
glVertex3f(-50, +22, +50);
glTexCoord2f(4,0);
glVertex3f(+50, +22, +50);
glTexCoord2f(4,4);
glVertex3f(+50 ,+22 ,-50);
glEnd;
glPopMatrix;
SwapBuffers(Context.DeviceContext);
end;
Hvad synes du om denne artikel? Giv din mening til kende ved at stemme via pilene til venstre og/eller lægge en kommentar herunder.
Del også gerne artiklen med dine Facebook venner:
Kommentarer (33)
Denne artikel er (ligesom 1'eren) genialt skrevet! Let forståelig, men stadig med en masse information. Et fem tal fra mig.
Det er bare så dejligt at læse.... Igen en Top-artikel fra dig og den fortjener intet andet end 5...
Super flot arktikel
Jeg spekulerer på hvor lang tid han har gået og gemt disse genistreger oppe i ærmet. Det er jo guldkorn det her
Flot artikel. Der var nogle småfejl, så som en constant der var stavet forkert, og man skulle lige nærlæse artiklen for at få defineret KassePosition, men bortset fra det så virker det fint. Så vil vi gerne have et kursus i at tilføje bevægelige elementer, så som modstandere ...
I øvrigt et 5 tal fra mig.
kan ikke lave en exe(eller hvad det nu er) så man kan prøve det
Super artikel... Så skal der laves 3Dspil ;D
Kommer der måske en 3ér ??
Nice artikel (er)
Ikke lige i forløbet, jeg skal havde mine eksamner overstået først
!
Nice artitel men jeg kan stadig ikke bestemme mig for om jeg skal begynde at lave spil i c++ eller delphi hvad syntes du er bedst???
Hvis det bare er for at laere OpenGL og lave dine egne 3D OpenGL applikationer kan det gor fuldstaendig ligemeget om du vaelger Delphi eller C++. Men satser du paa engang at blive professionel Spil Programmoer paa et kommersielt udvikler Studie er det nok C++ du skal kigge paa. Jeg valgte Delphi fordi jeg er vild med dens syntaxer og den maade Delphi er konstrureret paa. Men selve OpenGL kaldende er de samme uanset hvilket sprog du benytter!
Hej jeg ved godt du programmere i delphi men du ved til feldigvis ikke hvor jeg kan finde en compiler som kan compile opengl i c++ jeg bruger dev-c++ men den kommer hele tiden med errors når jeg compillere.
øhhh jooo næsten alle C++ kompilere kan kompile OpenGL i et Windows miljø. Eksempelvis Borland's C++ Builder, Micrsoft's Visual C++ og Intels C++ kompiler... Nu ved jeg selvfølgelig ikke hvilket kode du anvender siden din kompiler viser fejlmeddelser!
5 Her fra!!!
Men dog, kommer der en pop-up før jeg starter programmet der siger NVIDIA Corparation???
he he... ja, det er vist rigtigt! Det er følgende kode:
ShowMessage(glGetString(gl_Vendor));
Der er grunden til det! Det kan du bare slette! Det fortæller bare hvilket firma der har produceret dit grafikkort! Der er også gl_Renderer, den fortæller, hvad dit grafikkort hedder!
Mange tak for din rating!
er der en der en der kan hjælpe mig jeg kan ik komme vider med artikel 1 der hvor man skal lave et nyt opject hvordan gør jeg det
Lars>> Send mig en udvikler mail, omkring dit problem saa vil jeg proeve at svare hurtigst muligt!
Fantastisk...
5 fra mig
Genial artikel...
Jeg gav ofcourse 5
fed artikel 5 her fra...
men min siger "stream read error" en der kan fortælle mig hvad jeg gør galt ??
Genial artikel!
Jeg har lige 2 spørgsmål:
1) Hvordan man kan gøre dele af bitmap'erne gennemsigtige? - er der en TransparentColor eller noget lignende?
2) Ville det trække FPS væsentligt ned, hvis man tilføjede 1000 grafikelementer som også skal tegnes i TForm1.FormPaint, men som alligevel ikke kan ses på skærmen, fordi man fx "står med ryggen til" eller "noget andet står foran"?
(finder OpenGL selv ud af hvad der skal tegnes?)
1) Du kan gøre en bitmap transparent på flere forskellige måder! Enten per vertex, eller per texture (AlphaMap) eller per pixel (Pixel Shading)! Til alle funktioner skal glEnable(gl_Blend) tilføjes i starten af dit kode! Derefter skal du yderligere definere hvilken slags blend du vil foretage dig med funktionen glBlendFunc()
Det kunne eksempelvis være dette:
glBlendFunc(gl_Src_Alpha,gl_One_Minus_Src_Alpha);
Kig i din openGL.Pas fil for andre blend specifikationer!
Per Vertex!
Ved hver coordinat af dit geometry (trekant, firkant, polygon osv.) kan du indsætte en glColor4f(). Den sidste parameter er en Alpha channel og når den er 0 er dit geometry helt gennemsigtigt omkring den vertex. (Lidt ligesom da vi farvede hver enkelt vertex i den første artikel).
Per Texture (Alpha Map)
Du kan også indlæse en texture (på normal vis) og benytte den som en alpha map. Bitmappen du benytter som alpha map skal helst bestå af grå-skalaer (sort/hvid). Det der er mørkest vil blive mest transparent! Vær venligst opmærksom på at bitmappen må IKKE bestå af IndexColors (Selv om det ville være fristende at bruge, det SKAL være RGB colors). Husk også at rette din glBlendFunc() funktion til, så det passer dig bedst!
Per Pixel (PixelShading)
Dette er for de meget avancerede OpenGL programmøre og tager RIGTIGT RIGTIGT langt tid at forklare! Men du kan selv læse om det på enten www.nvidia.com eller www.ati.com
2) Jo flere polygoner du vil have tegnet jo længere tid vil det tage for dit grafik-kort. Hverken OpenGL eller DirectX tjekker, hvilke polygoner der kan ses på skærmen. Der bliver bare tegnet! For at optimere din applikation kan du kigge lidt "Frustum Culling"! Den metode er lavet til lige præcist det du snakker om! Derfor vil jeg anbefale dig at søge efter artikler på nettet der omhandler "Frustum Culling". Der findes allerede et par klasser skrevet i Delphi rundt om på nettet - frit tilgængeligt for alle!
Held og Lykke
Søren Klit Lambæk
NB: Tak for rosen fra jer alle!
MANGE tak!!!
Hej alle sammen, jeg er lige blevet færdig med denne fantaktiske artikel, men jeg har lidt problemer da når jeg starter, så starter jeg "i" kassen og når jeg bevæger mig rundt så sidder kassen fast på mig, meget fint hvis jeg skulle have en kjole, men lidt noget skidt når jeg skal skubbe med den
Nogen der kan forklarer hvorfor den gør det....
God Jul (skulle lige sige det)
Du har sikkert overset et eller andet i koden!
Det lyder som om at det er nogle matrix definationer der gaar galt for dig, samtidig med at du muligvis ikke flytter din kasse overhovedet!
Send mig en udvikler post med dit kode og jeg vil se paa sagen!
Med venlig hilsen
Søren Klit Lambæk
(God jul til dig - og tak for rosen)!
Så er den sendt til mailen...
(for en uges tid siden)
Hmmm jeg har ikke modtaget nogen mails endnu! Prøv lige at send den igen!
Tak
Sendt til info@opengl-delphi.dk
Hej allesammen, jeg har lige fået lidt hjælp af den flinke Søren og ville lige sige at hvis I vil have det sådan at kassen er flyttet lidt fra centrum (så I ikke starter inde i kassen) skal I skrive følgende ved variablen KassePosition:
var
KassePosition : TVektor = (X: -10; Y: 0; Z: -10);
Endnu et guldkorn fra Søren....
Hvis I ønsker Quake effekten med at man maks kan se 90 grader op og 90 grader ned skal I indsætte følgende kode under glPushMatrix;
if Rotation.X > 90 then Rotation.X := 90;
if Rotation.X < -90 then Rotation.X := -90;
Hov glemte at sige at den skulle indsættes i FormPaint proceduren
Da jeg var færdig me artiklen følte jeg kunne lave Hitman 2 uden problemer!
MEGET flot skrevet. 5 herfra!!
Super lavet..
5 du ;P
men tænkte lige.. Hvad er "FormPaint proceduren " og hvor skal koden sættes ind henne ?
Hej Dennis,
FormPaint proceduren kan du finde ved hjælp af Delphi's "Object Inspector". Vælg flaget "Events" og dobbeltklik på "OnPaint". Det vil genere en FormPaint procedure for dig. Hvis du er meget ny i Delphi eller programmering generalt er denne artikel nok en lidt for stor en mundfuld for dig!
Held og Lykke
Søren Klit Lambæk
Du skal være
logget ind for at skrive en kommentar.