{* UltraStar Deluxe - Karaoke Game * * UltraStar Deluxe is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * * $URL$ * $Id$ *} unit UScreenCredits; interface {$IFDEF FPC} {$MODE Delphi} {$ENDIF} {$I switches.inc} uses SysUtils, SDL, SDL_Image, gl, UMenu, UDisplay, UTexture, UMusic, UFiles, UThemes, UPath, UGraphicClasses; { beat detection constants and types } const SubChannelCount = 32; HistoryLength = 44; SamplesPerChannel = (FFTSize div 2) div SubChannelCount; BeatEnergyModifier = 80; // modifies detected energy // higher values equal a more sensitive detection type TEnergyHistory = array [0..HistoryLength-1] of single; TSubchannelHistory = array [0..SubChannelCount-1] of TEnergyHistory; type TCreditsStages=(InitialDelay, Intro, MainPart, Outro); TScreenCredits = class(TMenu) private CreditsPath: IPath; Credits_X: real; Credits_Time: cardinal; CTime_hold: cardinal; credits_bg_tex: TTexture; credits_bg_ovl: TTexture; //credits_bg_logo: TTexture; credits_names: array of TTexture; intro_layer01: TTexture; intro_layer02: TTexture; intro_layer03: TTexture; intro_layer04: TTexture; intro_layer05: TTexture; intro_layer06: TTexture; intro_layer07: TTexture; intro_layer08: TTexture; intro_layer09: TTexture; outro_bg: TTexture; outro_esc: TTexture; outro_exd: TTexture; CurrentScrollStart, CurrentScrollEnd: integer; CRDTS_Stage: TCreditsStages; { beat detection } SubChannelHistory: TSubchannelHistory; { mouse movement easter eggs: } MouseMoved: boolean; MouseX, MouseY: double; { saves last x and y angle for easter egg } LogoAngleX, LogoAngleY: single; procedure LoadNameTextures; { draw different stages } procedure DrawInitialDelay; { Intro } procedure DrawIntro; procedure DrawLayeredLogo(Separation, Scale, AngleX, AngleY, AngleZ: single); { Main } procedure DrawMain; procedure DrawMainBG; procedure DrawFunkyText; procedure DrawMainFG; procedure DrawNames; procedure DoLogoBling; { Outro } procedure DrawOutro; { beat detection } procedure DetectBeat; protected { beat detection stuff protected cause we need this information for "on beat effect"} LastBeatTime: cardinal; BeatDetected: boolean; CTime: cardinal; public Fadeout: boolean; constructor Create; override; function ParseInput(PressedKey: cardinal; CharCode: UCS4Char; PressedDown: boolean): boolean; override; function ParseMouse(MouseButton: integer; BtnDown: boolean; X, Y: integer): boolean; override; procedure OnShow; override; procedure OnHide; override; function Draw: boolean; override; end; const Funky_Text: string = 'Grandma Deluxe has arrived! Thanks to Corvus5 for the massive work on UltraStar, Wome for the nice tune you are hearing, '+ 'all the people who put massive effort and work in new songs (do not forget UltraStar w/o songs would be nothing), ppl from '+ 'irc helping us - eBandit and Gabari, scene ppl who really helped instead of compiling and running away. Greetings to DennisTheMenace for betatesting, '+ 'Demoscene.tv, pouet.net, KakiArts, Sourceforge,..'; { texture names (loaded from gameshared/resources/credits} CRDTS_BG_FILE = 'credits_v5_bg.png'; CRDTS_OVL_FILE = 'credits_v5_overlay.png'; INTRO_L01_FILE = 'intro-l-01.png'; INTRO_L02_FILE = 'intro-l-02.png'; INTRO_L03_FILE = 'intro-l-03.png'; INTRO_L04_FILE = 'intro-l-04.png'; INTRO_L05_FILE = 'intro-l-05.png'; INTRO_L06_FILE = 'intro-l-06.png'; INTRO_L07_FILE = 'intro-l-07.png'; INTRO_L08_FILE = 'intro-l-08.png'; INTRO_L09_FILE = 'intro-l-09.png'; OUTRO_BG_FILE = 'outro-bg.png'; OUTRO_ESC_FILE = 'outro-esc.png'; OUTRO_EXD_FILE = 'outro-exit-dark.png'; { some timings } Delay_Before_Start = 20; Intro_Flare_Start = 60; Intro_Zoom_End = 149; Intro_Stand_End = 155; Intro_Separation_End = 170; Intro_FadeToWhite_Start = 261; Intro_Zoomout_Start = 271; Main_Start = 271; Main_OnBeatTwinkle_Start = 280; Main_Names_Start = 359; Main_Names_End = 2833; Main_FadeOut_Start = 3096; Tune_End = 3366; { cosntants for developer names } type TFadeEffect = procedure (const Tex: TTexture; Progress: double); TCRTZ_Developer = record Name: string; // developer name for texture loading (names_"devel".png) Twinkle: boolean; // should there be twinkles on show FadeIn: TFadeEffect; // fade in effect Draw: TFadeEffect; // effect during draw FadeOut: TFadeEffect; // fade out effect end; { effects are called with blending, texture and matrix prepared } procedure Effect_Draw (const Tex: TTexture; Progress: double); procedure Effect_OnBeatJitter (const Tex: TTexture; Progress: double); procedure Effect_Rotate_Left_Top (const Tex: TTexture; Progress: double); procedure Effect_Rotate_Right_Bot (const Tex: TTexture; Progress: double); procedure Effect_ZoomIn_Rotate (const Tex: TTexture; Progress: double); procedure Effect_ZoomOut_Shift (const Tex: TTexture; Progress: double); procedure Effect_Shift_Left (const Tex: TTexture; Progress: double); procedure Effect_Shift_Right_Top (const Tex: TTexture; Progress: double); procedure Effect_Flip_Bot (const Tex: TTexture; Progress: double); procedure Effect_Flip_Right_Top (const Tex: TTexture; Progress: double); procedure Effect_Flip_Right (const Tex: TTexture; Progress: double); procedure Effect_Flip_Right_Bot (const Tex: TTexture; Progress: double); procedure Effect_Rotate_Right_Top (const Tex: TTexture; Progress: double); procedure Effect_Shift_Weird (const Tex: TTexture; Progress: double); procedure Effect_Shift_Right_Bot (const Tex: TTexture; Progress: double); procedure Effect_Rotate_Right_Top2(const Tex: TTexture; Progress: double); procedure Effect_Flip_Left_Bot (const Tex: TTexture; Progress: double); procedure Effect_Flip_Right_Top2 (const Tex: TTexture; Progress: double); procedure Effect_Twinkle_Down (const Tex: TTexture; Progress: double); const Developers: array[0..10] of TCRTZ_Developer = ( (Name: 'alexanders'; Twinkle: true; FadeIn: Effect_Rotate_Left_Top; Draw: Effect_OnBeatJitter; FadeOut: Effect_Rotate_Right_Bot), (Name: 'blindy'; Twinkle: true; FadeIn: Effect_ZoomIn_Rotate; Draw: Effect_OnBeatJitter; FadeOut: Effect_ZoomOut_Shift), (Name: 'brunzel'; Twinkle: true; FadeIn: Effect_Shift_Left; Draw: Effect_Draw; FadeOut: Effect_Shift_Right_Top), (Name: 'canni'; Twinkle: true; FadeIn: Effect_Flip_Bot; Draw: Effect_Draw; FadeOut: Effect_Flip_Right_Top), (Name: 'hennymcc'; Twinkle: true; FadeIn: Effect_Flip_Right; Draw: Effect_OnBeatJitter; FadeOut: Effect_Flip_Right_Bot), (Name: 'jaybinks'; Twinkle: true; FadeIn: Effect_Rotate_Right_Top; Draw: Effect_OnBeatJitter; FadeOut: Effect_Shift_Weird), (Name: 'krueger'; Twinkle: true; FadeIn: Effect_Shift_Right_Bot; Draw: Effect_OnBeatJitter; FadeOut: Effect_Rotate_Right_Top2), (Name: 'mezzox'; Twinkle: true; FadeIn: Effect_Flip_Left_Bot; Draw: Effect_OnBeatJitter; FadeOut: Effect_Flip_Right_Top), (Name: 'mischi'; Twinkle: true; FadeIn: Effect_Shift_Weird; Draw: Effect_OnBeatJitter; FadeOut: Effect_Flip_Bot), (Name: 'mog'; Twinkle: false; FadeIn: Effect_Twinkle_Down; Draw: Effect_OnBeatJitter; FadeOut: Effect_ZoomIn_Rotate), (Name: 'whiteshark'; Twinkle: true; FadeIn: Effect_Rotate_Right_Top2; Draw: Effect_OnBeatJitter; FadeOut: Effect_Shift_Left) ); { name specific times } TimePerName = (Main_Names_End - Main_Names_Start) div Length(Developers); NameFadeTime = 12; // duration of fade in/out in 1/100 secs NameWaitTime = 5; // delay between fade out and fade in of the next devel in 1/100 secs NameTwinkleTime = 2; // duration of star effects in 1/100 secs BeatJitterTime = 3; // duration of on beat jitter effect { position at which the names show up note: due to use of translate this is the center of the names not the upper left corner as usual } NameX = 223; NameY = 329; NameW = 326; NameH = 258; implementation uses Math, ULog, UGraphic, UMain, UIni, USongs, Textgl, ULanguage, UCommon, UPathUtils; function TScreenCredits.ParseInput(PressedKey: cardinal; CharCode: UCS4Char; PressedDown: boolean): boolean; begin Result := true; if (PressedDown) then begin // Key Down case PressedKey of SDLK_ESCAPE, SDLK_BACKSPACE, SDLK_RETURN : begin FadeTo(@ScreenMain); AudioPlayback.PlaySound(SoundLib.Back); end; { SDLK_SPACE: begin setlength(CTime_hold,length(CTime_hold)+1); CTime_hold[high(CTime_hold)]:=CTime; end; } end; // esac end; // fi end; function TScreenCredits.ParseMouse(MouseButton: integer; BtnDown: boolean; X, Y: integer): boolean; begin Result := inherited ParseMouse(MouseButton, BtnDown, X, Y); { calculate mouse coordinates from -1 to 1 relative to screen center } MouseX := (X - (ScreenW / Screens) / 2) / ((ScreenW / Screens) / 2); MouseY := (Y - ScreenH / 2) / (ScreenH / 2); MouseMoved := true; end; procedure TScreenCredits.LoadNameTextures; var I: integer; begin SetLength(credits_names, Length(Developers)); for I := 0 to High(Developers) do begin credits_names[I] := Texture.LoadTexture(CreditsPath.Append('names_' + Developers[I].Name + '.png'), TEXTURE_TYPE_TRANSPARENT, 0); end; end; constructor TScreenCredits.Create; begin inherited Create; CreditsPath := ResourcesPath.Append('credits', pdAppend); credits_bg_tex := Texture.LoadTexture(CreditsPath.Append(CRDTS_BG_FILE), TEXTURE_TYPE_PLAIN, 0); credits_bg_ovl := Texture.LoadTexture(CreditsPath.Append(CRDTS_OVL_FILE), TEXTURE_TYPE_TRANSPARENT, 0); LoadNameTextures; intro_layer01 := Texture.LoadTexture(CreditsPath.Append(INTRO_L01_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer02 := Texture.LoadTexture(CreditsPath.Append(INTRO_L02_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer03 := Texture.LoadTexture(CreditsPath.Append(INTRO_L03_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer04 := Texture.LoadTexture(CreditsPath.Append(INTRO_L04_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer05 := Texture.LoadTexture(CreditsPath.Append(INTRO_L05_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer06 := Texture.LoadTexture(CreditsPath.Append(INTRO_L06_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer07 := Texture.LoadTexture(CreditsPath.Append(INTRO_L07_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer08 := Texture.LoadTexture(CreditsPath.Append(INTRO_L08_FILE), TEXTURE_TYPE_TRANSPARENT, 0); intro_layer09 := Texture.LoadTexture(CreditsPath.Append(INTRO_L09_FILE), TEXTURE_TYPE_TRANSPARENT, 0); outro_bg := Texture.LoadTexture(CreditsPath.Append(OUTRO_BG_FILE), TEXTURE_TYPE_PLAIN, 0); outro_esc := Texture.LoadTexture(CreditsPath.Append(OUTRO_ESC_FILE), TEXTURE_TYPE_TRANSPARENT, 0); outro_exd := Texture.LoadTexture(CreditsPath.Append(OUTRO_EXD_FILE), TEXTURE_TYPE_TRANSPARENT, 0); CRDTS_Stage:=InitialDelay; end; procedure TScreenCredits.OnShow; begin inherited; { pause background music } SoundLib.PauseBgMusic; CRDTS_Stage := InitialDelay; CTime := 0; Credits_X := 580; { open credits tune, we play it after initial delay } AudioPlayback.Open(soundpath.Append('wome-credits-tune.mp3')); // thank you wetue { reset twinkling stars } GoldenRec.KillAll; { reset mouse coords } MouseMoved := false; MouseX := 0; MouseY := 0; { hide cursor } Display.SetCursor; end; procedure TScreenCredits.OnHide; begin AudioPlayback.Stop; { show cursor } Display.SetCursor; SoundLib.StartBgMusic; end; function TScreenCredits.Draw: boolean; var T: cardinal; begin Result := true; // reset beat detection BeatDetected := false; T := SDL_GetTicks() div 33; if T <> Credits_Time then begin Credits_Time := T; inc(CTime); inc(CTime_hold); Credits_X := Credits_X-2; if (CRDTS_Stage = InitialDelay) and (CTime >= Delay_Before_Start) then begin CRDTS_Stage := Intro; CTime := 0; AudioPlayback.Play; end else if (CRDTS_Stage = Intro) and (CTime >= Main_Start) then begin CRDTS_Stage := MainPart; end else if (CRDTS_Stage = MainPart) and (CTime >= Tune_End) then begin CRDTS_Stage := Outro; end; // dis does teh muiwk y0r to be translated :-) DetectBeat; end; case CRDTS_Stage of InitialDelay: DrawInitialDelay; Intro: DrawIntro; MainPart: DrawMain; Outro: DrawOutro; end; // make the stars shine GoldenRec.Draw; end; procedure TScreenCredits.DrawInitialDelay; begin glClearColor(0, 0, 0, 0); glClear(GL_COLOR_BUFFER_BIT or GL_DEPTH_BUFFER_BIT); end; procedure TScreenCredits.DrawIntro; var Separation, Scale, AngleX, AngleY, AngleZ: single; FlareX, FlareY: single; I: integer; begin glClearColor(0, 0, 0, 0); glClear(GL_COLOR_BUFFER_BIT or GL_DEPTH_BUFFER_BIT); { rotate logo anti clockwise and make it grow } if (CTime >= Intro_Separation_End) then begin Separation := 1; Scale := 1 + sqr(CTime - Intro_Separation_End) / (32 * (Main_Start - Intro_Separation_End)); AngleX := 0; AngleY := 0; AngleZ := 20 * sqr(CTime - Intro_Separation_End) / sqr((Main_Start - Intro_Separation_End) / 2); end { separate layers } else if (CTime >= Intro_Stand_End) then begin Separation := 0.5 + 0.5 * (CTime - Intro_Stand_End) / (Intro_Separation_End - Intro_Stand_End); Scale := 1; AngleX := 0; AngleY := 0; AngleZ := 0; end { stand still } else if (CTime >= Intro_Zoom_End) then begin Separation := 0.5; Scale := 1; AngleX := 0; AngleY := 0; AngleZ := 0; end { rotate left } else begin Separation := 0.5 + 0.5 * (Intro_Zoom_End - CTime) / (Intro_Zoom_End); Scale := 1; AngleX := 10 * (Intro_Zoom_End - CTime) / (Intro_Zoom_End); AngleY := 20 * (Intro_Zoom_End - CTime) / (Intro_Zoom_End); AngleZ := 0; end; { the user moved the mouse, overwrite X and Y angle with according to mouse position } if (MouseMoved) then begin // calculate destination angle AngleX := 30 * MouseY; AngleY := 30 * MouseX; { move angle towards destination } if not SameValue(LogoAngleX, AngleX, 0.001) then AngleX := LogoAngleX + 0.05 * (AngleX - LogoAngleX); if not SameValue(LogoAngleY, AngleY, 0.001) then AngleY := LogoAngleY + 0.05 * (AngleY - LogoAngleY); end; // save last angle LogoAngleX := AngleX; LogoAngleY := AngleY; DrawLayeredLogo(Separation, Scale, AngleX, AngleY, AngleZ); { do some sparkling effects } if (CTime < Intro_Zoom_End) and (CTime > Intro_Flare_Start) then begin for I := 1 to 3 do begin FlareX := 410 + Floor((CTime - Intro_Flare_Start) / (Intro_Zoom_End - Intro_Flare_Start) * (536 - 410)) + RandomRange(-5, 5); FlareY := Floor((Intro_Zoom_End - CTime) / 22) + RandomRange(285, 301); GoldenRec.Spawn(FlareX, FlareY, 1, 16, 0, -1, Flare, 0); end; end; { fade to white at end } if Ctime > Intro_FadeToWhite_Start then begin glColor4f(1, 1, 1, sqr(CTime - Intro_FadeToWhite_Start) * (CTime - Intro_FadeToWhite_Start) / sqr(Main_Start - Intro_FadeToWhite_Start)); glEnable(GL_BLEND); glBegin(GL_QUADS); glVertex2f( 0, 0); glVertex2f( 0, 600); glVertex2f(800, 600); glVertex2f(800, 0); glEnd; glDisable(GL_BLEND); end; end; procedure Start3D; begin glMatrixMode(GL_PROJECTION); glPushMatrix; glLoadIdentity; glFrustum(-0.3 * 4 / 3, 0.3 * 4 / 3, -0.3, 0.3, 1, 1000); glMatrixMode(GL_MODELVIEW); glLoadIdentity; end; procedure End3D; begin glMatrixMode(GL_PROJECTION); glPopMatrix; glMatrixMode(GL_MODELVIEW); end; procedure TScreenCredits.DrawLayeredLogo(Separation, Scale, AngleX, AngleY, AngleZ: single); var TotalAngle: single; begin Start3D; glPushMatrix; glEnable(GL_TEXTURE_2D); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glEnable(GL_BLEND); glTranslatef(0, 0, -5 + 0.5 * Separation); TotalAngle := Abs(AngleX) + Abs(AngleY) + Abs(AngleZ); if not isZero(TotalAngle) then glRotatef(TotalAngle, AngleX / TotalAngle, AngleY / TotalAngle, AngleZ / TotalAngle); glScalef(Scale, Scale, 1); glScalef(4/3, -1, 1); glColor4f(1, 1, 1, 1); glBindTexture(GL_TEXTURE_2D, intro_layer01.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, -0.4 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, -0.4 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, -0.4 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, -0.4 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer02.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, -0.3 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, -0.3 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, -0.3 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, -0.3 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer03.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, -0.2 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, -0.2 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, -0.2 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, -0.2 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer04.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, -0.1 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, -0.1 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, -0.1 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, -0.1 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer05.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, 0 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, 0 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, 0 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, 0 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer06.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, 0.1 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, 0.1 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, 0.1 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, 0.1 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer07.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, 0.2 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, 0.2 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, 0.2 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, 0.2 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer08.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, 0.3 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, 0.3 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, 0.3 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, 0.3 * Separation); glEnd; glBindTexture(GL_TEXTURE_2D, intro_layer09.TexNum); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex3f(-1, -1, 0.22 * Separation); glTexCoord2f(0, 1); glVertex3f(-1, 1, 0.22 * Separation); glTexCoord2f(1, 1); glVertex3f( 1, 1, 0.22 * Separation); glTexCoord2f(1, 0); glVertex3f( 1, -1, 0.22 * Separation); glEnd; glDisable(Gl_Texture_2D); glDisable(GL_BLEND); glPopMatrix; End3D; end; procedure TScreenCredits.DrawMain; begin DrawMainBG; DrawFunkyText; DrawNames; DrawMainFG; DoLogoBling; // fade out at end of main part if (Ctime > Main_FadeOut_Start) then begin glColor4f(0, 0, 0, (CTime - Main_FadeOut_Start) / (Tune_End - Main_FadeOut_Start)); glEnable(GL_BLEND); glBegin(GL_QUADS); glVertex2f( 0, 0); glVertex2f( 0, 600); glVertex2f(800, 600); glVertex2f(800, 0); glEnd; glDisable(GL_BLEND); end; end; procedure TScreenCredits.DrawMainBG; begin glEnable(GL_TEXTURE_2D); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glEnable(GL_BLEND); glColor4f(1, 1, 1, 1); glBindTexture(GL_TEXTURE_2D, credits_bg_tex.TexNum); glBegin(Gl_Quads); glTexCoord2f( 0, 0); glVertex2f( 0, 0); glTexCoord2f( 0, 600/1024); glVertex2f( 0, RenderH); glTexCoord2f(800/1024, 600/1024); glVertex2f(RenderW, RenderH); glTexCoord2f(800/1024, 0); glVertex2f(RenderW, 0); glEnd; glDisable(GL_TEXTURE_2D); glDisable(GL_BLEND); end; procedure TScreenCredits.DrawFunkyText; var S: integer; X, Y, A: real; visibleText: string; begin SetFontSize(30); // init ScrollingText if (CTime = Main_Start) then begin // set position of text Credits_X := 600; CurrentScrollStart := 1; CurrentScrollEnd := 1; end; if (CTime > Main_Start) and (CurrentScrollStart < length(Funky_Text)) then begin X := 0; visibleText := Copy(Funky_Text, CurrentScrollStart, CurrentScrollEnd); for S := 1 to length(visibleText) do begin Y := abs(sin((Credits_X + X) * 0.93 { * (((Credits_X + X)) / 1200) } / 100 * pi)); SetFontPos(Credits_X + X, 538 - Y * (Credits_X + X) * (Credits_X + X) * (Credits_X + X) / 1000000); if (Credits_X + X > 32) then A := 17 else if (Credits_X + X >= 15) then A := Credits_X + X - 15 else A := 0; glColor4f(230 / 255 - 40 / 255 + Y * (Credits_X + X)/ 900, 200 / 255 - 30 / 255 + Y * (Credits_X + X)/ 1000, 155 / 255 - 20 / 255 + Y * (Credits_X + X)/ 1100, A / 17); glPrint(visibleText[S]); X := X + glTextWidth(visibleText[S]); end; if (Credits_X < 0) and (CurrentScrollStart < length(Funky_Text)) then begin Credits_X := Credits_X + glTextWidth(Funky_Text[CurrentScrollStart]); inc(CurrentScrollStart); end; visibleText := Copy(Funky_Text, CurrentScrollStart, CurrentScrollEnd); if (Credits_X + glTextWidth(visibleText) < 600) and (CurrentScrollEnd < length(Funky_Text)) then begin inc(CurrentScrollEnd); end; end; { // timing hack X:=5; SetFontStyle(2); SetFontItalic(false); SetFontSize(27); glColor4f(1, 1, 1, 1); for S := 0 to high(CTime_hold) do begin visibleText := inttostr(CTime_hold[S]); SetFontPos (500, X); glPrint(visibleText[0]); X := X + 20; end; } end; procedure TScreenCredits.DrawNames; var Dev: integer; Ticks: integer; DevTicks: integer; TwinkleW, TwinkleH: integer; begin Ticks := (CTime - Main_Names_Start); Dev := Ticks div TimePerName; DevTicks := Ticks mod TimePerName; {// debug stuff SetFontPos(20, 20); glPrint('Ticks: ' + IntToStr(Ticks)); SetFontPos(20, 45); glPrint('Dev: ' + IntToStr(Dev)); SetFontPos(20, 70); glPrint('DevTicks: ' + IntToStr(DevTicks)); //} if (Ticks >= 0) and (Dev <= High(Developers)) then begin { spawn twinkling stars } if (Developers[Dev].Twinkle) and (DevTicks >= NameFadeTime) and (DevTicks <= NameFadeTime + NameTwinkleTime) then begin TwinkleW := Round(NameW * 0.6); TwinkleH := Round(NameH * 0.6); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 0); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 1); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 5); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 0); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 1); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 5); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 0); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 1); GoldenRec.Spawn(NameX + RandomRange(-TwinkleW, TwinkleW), NameY + RandomRange(-TwinkleH, TwinkleH), 1, 16, 0, -1, PerfectLineTwinkle, 5); end; { prepare drawing } glPushMatrix; glTranslatef(NameX, NameY, 0); glBindTexture(GL_TEXTURE_2D, credits_names[Dev].TexNum); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glEnable(GL_BLEND); glEnable(GL_TEXTURE_2D); // calculate progress and call effect if (DevTicks <= NameFadeTime) then Developers[Dev].FadeIn(credits_names[Dev], DevTicks / NameFadeTime) else if (DevTicks >= TimePerName - NameFadeTime - NameWaitTime) then begin if (DevTicks < TimePerName - NameWaitTime) then Developers[Dev].FadeOut(credits_names[Dev], ((TimePerName - NameWaitTime) - DevTicks) / NameFadeTime); end else Developers[Dev].Draw(credits_names[Dev], (DevTicks - NameFadeTime) / (TimePerName - NameFadeTime * 2 - NameWaitTime)); glDisable(GL_TEXTURE_2D); glDisable(GL_BLEND); glPopMatrix; end; end; procedure TScreenCredits.DrawMainFG; begin glEnable(GL_TEXTURE_2D); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glEnable(GL_BLEND); glColor4f(1, 1, 1, 1); glBindTexture(GL_TEXTURE_2D, credits_bg_ovl.TexNum); glBegin(gl_Quads); glTexCoord2f( 0, 0); glVertex2f(800-393, 0); glTexCoord2f( 0, 600/1024); glVertex2f(800-393, 600); glTexCoord2f(393/512, 600/1024); glVertex2f(800, 600); glTexCoord2f(393/512, 0); glVertex2f(800, 0); glEnd; glDisable(GL_TEXTURE_2D); glDisable(GL_BLEND); end; procedure TScreenCredits.DoLogoBling; const myLogoCoords: array[0..27,0..1] of cardinal = ( ( 39,32),( 84,32),(100,16),(125,24), (154,31),(156,58),(168,32),(203,36), (258,34),(251,50),(274,93),(294,84), (232,54),(278,62),(319,34),(336,92), (347,23),(374,32),(377,58),(361,83), (385,91),(405,91),(429,35),(423,51), (450,32),(485,34),(444,91),(486,93) ); var Coords: integer; StartFrame: integer; begin if (CTime > Main_OnBeatTwinkle_Start ) and (CTime < Main_FadeOut_Start) then begin { spawn stars only in frames where a beat was detected } if BeatDetected then begin StartFrame := RandomRange(6, 16); Coords := RandomRange(0, 27); GoldenRec.Spawn(myLogoCoords[Coords,0], myLogoCoords[Coords,1], 16-StartFrame, StartFrame, 0, -1, PerfectNote, 0); end; end; end; procedure TScreenCredits.DrawOutro; begin if CTime = Tune_End then begin CTime_hold := 0; AudioPlayback.Stop; AudioPlayback.Open(SoundPath.Append('credits-outro-tune.mp3')); AudioPlayback.SetVolume(0.2); AudioPlayback.SetLoop(true); AudioPlayback.Play; end; if CTime_hold > 231 then begin AudioPlayback.Play; Ctime_hold := 0; end; glClearColor(0, 0, 0, 0); glClear(GL_COLOR_BUFFER_BIT or GL_DEPTH_BUFFER_BIT); // do something useful // outro background glEnable(GL_TEXTURE_2D); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glEnable(GL_BLEND); glColor4f(1, 1, 1, 1); glBindTexture(GL_TEXTURE_2D, outro_bg.TexNum); glBegin(gl_quads); glTexCoord2f( 0, 0); glVertex2f( 0, 0); glTexCoord2f( 0, 600/1024); glVertex2f( 0, 600); glTexCoord2f(800/1024, 600/1024); glVertex2f(800, 600); glTexCoord2f(800/1024, 0); glVertex2f(800, 0); glEnd; // outro overlays glColor4f(1, 1, 1, (2 + sin(CTime / 15)) / 3); glBindTexture(GL_TEXTURE_2D, outro_esc.TexNum); glBegin(Gl_Quads); glTexCoord2f( 0, 0); glVertex2f( 0, 0); glTexCoord2f( 0, 223/256); glVertex2f( 0, 223); glTexCoord2f(487/512, 223/256); glVertex2f(487, 223); glTexCoord2f(487/512, 0); glVertex2f(487, 0); glEnd; if (RandomRange(0,20) <= 18) then begin glColor4f(1, 1, 1, 1); glBindTexture(GL_TEXTURE_2D, outro_exd.TexNum); glBegin(Gl_Quads); glTexCoord2f( 0, 0); glVertex2f(800-310, 600-247); glTexCoord2f( 0, 247/256); glVertex2f(800-310, 600 ); glTexCoord2f(310/512, 247/256); glVertex2f(800, 600 ); glTexCoord2f(310/512, 0); glVertex2f(800, 600-247); glEnd; end; glDisable(GL_TEXTURE_2D); glDisable(GL_BLEND); // outro scrollers? // ... end; { name effects } { effects are called with blending texture and matrix prepared } procedure Effect_Draw (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, 1); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_OnBeatJitter (const Tex: TTexture; Progress: double); var Diff: cardinal; Alpha: double; begin Diff := ScreenCredits.CTime - ScreenCredits.LastBeatTime; if (Diff < BeatJitterTime) then Alpha := 0.5 + 0.5 * Diff / BeatJitterTime else Alpha := 1; glColor4f(1, 1, 1, Alpha); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Rotate_Left_Top (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); gltranslatef(-NameX, 0, 0); glrotatef(Progress * 90 + 270, 0, 0, 1); gltranslatef(NameX, 0, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Rotate_Right_Bot (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); gltranslatef(NameX, 0, 0); glrotatef((Progress - 1) * 90, 0, 0, 1); gltranslatef(-NameX, 0, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_ZoomIn_Rotate (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); glscalef(sqr(Progress), sqr(Progress), sqr(Progress)); glrotatef(Progress * 360, 0, 0, 1); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_ZoomOut_Shift (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := (1 - Progress); gltranslatef(X * 300, -X * 100, 0); glscalef(1 + X, 1 + X, 1 + X); glrotatef(X * 90, 0, 0, 1); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Shift_Left (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); glTranslatef((Progress - 1) * 210, 0, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Shift_Right_Top (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); glTranslatef((1 - Progress) * 210, (Progress - 1) * 105, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Flip_Bot (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := NameH * (1 - Progress); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2 - 1.5 * X, -NameH/2 + 1.5 * X); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2 + 1.5 * X, -NameH/2 + 1.5 * X); glEnd; end; procedure Effect_Flip_Right_Top (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := NameW * (1 - Progress); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2 + X, -NameH/2 - X/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2 + X, NameH/2 - (X * 1.5 * NameH / NameW)); glTexCoord2f(1, 1); glVertex2f( NameW/2 + X, NameH/2 + X / 4); glTexCoord2f(1, 0); glVertex2f( NameW/2 + X, -NameH/2 - X / 4); glEnd; end; procedure Effect_Flip_Right (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := NameW * (1 - Progress); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2 - X, NameH/2 + X * 1.5); glTexCoord2f(1, 0); glVertex2f( NameW/2 - X, -NameH/2 - X * 1.5); glEnd; end; procedure Effect_Flip_Right_Bot (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := NameW * (1 - Progress); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2 + X * 1.5, -NameH/2 + X * 1.5); glTexCoord2f(0, 1); glVertex2f(-NameW/2 + X * 1.2, NameH/2 + X); glTexCoord2f(1, 1); glVertex2f( NameW/2 + X / 2, NameH/2 + X / 4); glTexCoord2f(1, 0); glVertex2f( NameW/2 + X * 1.5, -NameH/2); glEnd; end; procedure Effect_Rotate_Right_Top (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); glTranslatef(NameX, 0, 0); glrotatef((1 - Progress) * 90, 0, 0, 1); glTranslatef(-NameX, 0, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Shift_Weird (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := (Progress - 1); glTranslatef(X * 200, X * 100, 0); glScalef(Progress, Progress, Progress); glRotatef(X * 90, 0, 0, 1); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Shift_Right_Bot (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); glTranslatef((1 - Progress) * 200, (1 - Progress) * 100, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Rotate_Right_Top2 (const Tex: TTexture; Progress: double); begin glColor4f(1, 1, 1, Progress); glTranslatef(0, -NameX, 0); glRotatef((Progress - 1) * 90, 0, 0, 1); glTranslatef(0, NameX, 0); glRotatef((1 - Progress) * 90, 0, 0, 1); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH/2); glTexCoord2f(0, 1); glVertex2f(-NameW/2, NameH/2); glTexCoord2f(1, 1); glVertex2f( NameW/2, NameH/2); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH/2); glEnd; end; procedure Effect_Flip_Left_Bot (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := (1 - Progress) * NameW; glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2 - X, -NameH/2 + X / 4); glTexCoord2f(0, 1); glVertex2f(-NameW/2 - X / 4, NameH/2 + X / 4); glTexCoord2f(1, 1); glVertex2f( NameW/2 - X * 1.2, NameH/2 + X / 2); glTexCoord2f(1, 0); glVertex2f( NameW/2 - X * 1.5, -NameH/2 + X * 1.5); glEnd; end; procedure Effect_Flip_Right_Top2 (const Tex: TTexture; Progress: double); var X: double; begin glColor4f(1, 1, 1, Progress); X := (1 - Progress) * NameW; glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2 + X, -NameH/2 - X / 2); glTexCoord2f(0, 1); glVertex2f(-NameW/2 + X, NameH/2 + X / 2); glTexCoord2f(1, 1); glVertex2f( NameW/2 + X / 4, NameH/2 - X / 4); glTexCoord2f(1, 0); glVertex2f( NameW/2 + X / 4, -NameH/2 + X / 4); glEnd; end; procedure Effect_Twinkle_Down (const Tex: TTexture; Progress: double); begin // draw name glColor4f(1, 1, 1, 1); glTranslatef(0, NameH/2, 0); glBegin(gl_Quads); glTexCoord2f(0, 0); glVertex2f(-NameW/2, -NameH * Progress); glTexCoord2f(0, Progress); glVertex2f(-NameW/2, 0); glTexCoord2f(1, Progress); glVertex2f( NameW/2, 0); glTexCoord2f(1, 0); glVertex2f( NameW/2, -NameH * Progress); glEnd; //spawn some stars on the edge GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 0); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 1); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 5); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 0); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 1); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 5); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 0); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 1); GoldenRec.Spawn(NameX + RandomRange(-NameW div 2, NameW div 2), NameY - NameH/2 + (1 - Progress) * NameH, 1, 16, 0, -1, PerfectLineTwinkle, 5); end; { beat detection algorithm based on a tutorial from Fr�d�ric Patin on gamedev.net http://www.gamedev.net/reference/programming/features/beatdetection/default.asp } { calculates average value of a history buffer } function Average(History: TEnergyHistory): single; var I: integer; begin Result := 0; for I := 0 to HistoryLength - 1 do Result := Result + History[I]; Result := Result / HistoryLength; end; { calculates variance value of a history buffer } function Variance(History: TEnergyHistory; Average: single): single; var I: integer; begin Result := 0; for I := 0 to HistoryLength - 1 do Result := Result + sqr(History[I] - Average); Result := Result / HistoryLength; end; { shifts all values of the history to the right and adds the new value at the front } procedure AddHistory(Value: single; var History: TEnergyHistory); var I: integer; begin for I := HistoryLength - 1 downto 1 do History[I] := History[I-1]; History[0] := Value; end; { calculates instant energy from FFT data for a specific subchannel (0..SubChannelCount - 1) } function CalculateInstantEnergy(SubChannel: integer; Data: TFFTData): single; var I: integer; begin Result := 0; for I := SubChannel * SamplesPerChannel to (SubChannel + 1) * SamplesPerChannel - 1 do Result := Result + Data[I] * BeatEnergyModifier; Result := Result / SamplesPerChannel; end; procedure TScreenCredits.DetectBeat; var Data: TFFTData; I: integer; Instant: single; C, E, V: single; begin AudioPlayback.GetFFTData(Data); // do beatdetection for every subchannel for I := 0 to SubChannelCount - 1 do begin Instant := CalculateInstantEnergy(I, Data); E := Average(SubchannelHistory[I]); V := Variance(SubchannelHistory[I], E); C := (-0.0025714 * V) + 1.5142857; AddHistory(Instant, SubChannelHistory[I]); if (Instant > 2) and (Instant > C * E) then begin // beat detected BeatDetected := true; LastBeatTime := CTime; end; end; end; end.