aboutsummaryrefslogblamecommitdiffstats
path: root/Game/Code/Classes/UMain.pas
blob: bc3ac1b094e15651c78b25447c183ad24df37762 (plain) (tree)
1
2
3
4
5
6
7
8
9



                                                                                                  
                                                                                       



                          
                                           

















                                                                   

                                                         



























                                                                                        
                                                                            















                                                                                      

                         



























                                         



                                                                                                      
                                                                
                                                                   




                                               
                                                  







                                 
                        






                              

                  
 
                                                 
                       
                        






                                    
                      
       

           



                       



                                           
                        

                                                    
                                        
                
                                        

                                                                
                                       





                                                                
                                        
               



                  


                                            
             













                                                                                                     



                                                                  



























































                                                                                                                    

                                     


                                                                                        

           




                                                                                                                          

                                 


                                                

             










                                                                                          
 



                                                                            
                  

                                  


                  

                        















                                                                                                 

                                  

                                  



                                                                                         



                                                                                                                               

                                                                                       




                                                                                                    










                                     
                                         
                                                 




                                                                                                                                                  




                                                                  
                                                                                                                                            



                                         





                          
                                
                                         




                                                          



                                                                                    



                                                       

                                                
 


                                                                   
 


                                                                        
 


                                                                      
 

















                                                                                                                                        
 

                       
 
                                         
         
       

     
                                                         



                            


















                                      


                        




                                                  
         
                          
       

                          
                                               

     
                                                     



                       

                                                            
 

                                                              

                                                                              

                                           
                                            
         
       

     
                                                      








                     
                                                                                                                                

                     





                                                                               



                                  











                                                       
                                                      
      
                       






                     
                                                    


                          


                                     


                   
                   


                                                           



                                 





                   

             












                                                                                                                       
                                                        
            



                                        
              
                                                        
                









                                                                                                             
               
             
 
                   
 
                                                                   
              

                                                        
                


                                                                  
                  
                                                                                                 



                                                                                   
 
                                       
                                
                                           
 
                                           
                                                                                         
                                    
                    






















                                                                                                       
                     
 

                                                                                    
 


                                                                                                               
               
 

                                                    
                


























                                                                                                                     
                  

                                                                                                              
                 
 
 













                                                                                                         






                                                    
                                                                     
                



                                                                             








                                                                                                    













                                                                           












                                                                                                                       
                                                        
            



                                        
              
                                                        
                









                                                                                                             
               
             
 
                   
 
                                                                     
              

                                                        
                


                                                                  
                  
                                                     



                                                                                   
 
                                       
                                
                                           
               
                                           
                                                                                         
                                    
                    






















                                                                                                       
                     
 

                                                                                    
 


                                                                                                               

               

                                                            
                


























                                                                                                                     
                  

                                                                                                              
                 
 














                                                                                                         







                                                    
                                                                     
                



                                                                             








                                                                                                    












                                                                           

                      
 
                 
 









                                                                                                     


















                                                                             

                                      









                                                                                         
unit UMain;

interface
uses SDL, UGraphic, UMusic, URecord, UTime, SysUtils, UDisplay, UIni, ULog, ULyrics, UScreenSing,
  gl, zlportio {you can disable it and all PortWriteB calls}, UThemes{, UScreenPopup};

type
  TPlayer = record
    Name:         string;
    VoiceFile:    string; //Recorded Voice

    Score:        real;
    ScoreLine:    real;
    ScoreGolden:  real;

    ScoreI:       integer;
    ScoreLineI:   integer;
    ScoreGoldenI: integer;
    ScoreTotalI:  integer;



    //SingBar Mod
    ScoreLast:    Real;//Last Line Score
    ScorePercent:    integer;//Aktual Fillstate of the SingBar
    ScorePercentTarget:  integer;//Target Fillstate of the SingBar
    //end Singbar Mod

    ScoreMax:     integer; //max possible score (actual)

    //PhrasenBonus - Line Bonus Mod
    LineBonus_PosX:     Single;
    LineBonus_PosY:     Single;
    LineBonus_Alpha:    Single;
    LineBonus_Visible:  boolean;
    LineBonus_Text:     string;
    LineBonus_Color:    TRGB;
    LineBonus_Age:      Integer;

    //Variable vor Positioning -> Set on ScreenShow, different when Playercount Changes
    LineBonus_TargetX:  integer;
    LineBonus_TargetY:  integer;
    LineBonus_StartX:  integer;
    LineBonus_StartY:  integer;
    //PhrasenBonus - Line Bonus Mod End

    //PerfectLineTwinkle Mod (effect)
    LastSentencePerfect: Boolean;
    //PerfectLineTwinkle Mod end


//    Meter:        real;

    HighNut:  integer;
    IlNut:    integer;
    Nuta:     array of record
      Start:      integer;
      Dlugosc:    integer;
      //Detekt:     real;     // dokladne miejsce, w ktorym wykryto ta nute
      Ton:        real;
      Perfect:    boolean; // true if the note matches the original one, lit the star



      // Half size Notes Patch
      Hit:        boolean; // true if the note Hits the Line
      //end Half size Notes Patch



    end;
  end;

  TStats = record
    Player: array of TPlayer;
    SongArtist:   String;
    SongTitle:    String;
  end;

  TMedleyPlaylist = record
    Song:               array of integer;
    NumMedleySongs:     integer;
    CurrentMedleySong:  integer;
    ApplausePlayed:     boolean;
    Stats:              array of TStats;
    NumPlayer:          integer;
  end;


var
  OGL:      Boolean;
  Done:     Boolean;
  Event:    TSDL_event;
  FileName: string;
  Restart:  boolean;

  // gracz i jego nuty
  Player:         array of TPlayer;
  PlayersPlay:    integer;
  PlaylistMedley: TMedleyPlaylist;


procedure MainLoop;
procedure CheckEvents;
procedure Sing(Sender: TScreenSing);
procedure NewSentence(CP: integer; Sender: TScreenSing);
procedure NewBeat(CP: integer; Sender: TScreenSing); // executed when on then new beat
procedure NewBeatC(CP: integer; Sender: TScreenSing); // executed when on then new beat for click
procedure NewBeatD(CP: integer; Sender: TScreenSing); // executed when on then new beat for detection
//procedure NewHalf; // executed when in the half between beats
procedure NewNote(P: integer; Sender: TScreenSing); // detect note
function GetMidBeat(Time: real): real;
function GetTimeFromBeat(Beat: integer): real;
procedure ClearScores(PlayerNum: integer);

implementation
uses USongs, Math, UCommandLine, UVideo, UWebCam;

procedure MainLoop;
var
  Delay:    integer;
begin
  SDL_EnableKeyRepeat(125, 125);
  While not Done do
  Begin
    PerfLog.CycleStart;
    // keyboard events
    CheckEvents;

    // display
    done := not Display.Draw;
    SwapBuffers;

    // delay
    CountMidTime;

    Delay := Floor(1000 / 100 - 1000 * TimeMid);
    if Delay >= 1 then
      SDL_Delay(Delay);
    CountSkipTime;

    // reinitialization of graphics
    if Restart then begin
      Reinitialize3D;
      Restart := false;
    end;
    PerfLog.CycleEnd;
  End;
  wClose;
  acClose;
  FreeOpenGL;
End;

Procedure CheckEvents;
Begin
  if not Assigned(Display.NextScreen) then
  While SDL_PollEvent( @event ) = 1 Do
  Begin
    Case Event.type_ Of
      SDL_ACTIVEEVENT: //workaround for alt-tab bug
        begin
          if (Event.active.gain=1) then
          begin
            SDL_SetModState(KMOD_NONE);
            if (Ini.FullScreen = 1) or (Params.FullScreen) then
              SDL_ShowCursor(0);
            //EnableVideoDraw := true;
          end;

          if (Event.active.gain=0) then
          begin
            if (Ini.FullScreen = 1) or (Params.FullScreen) then
              SDL_ShowCursor(1);
            //EnableVideoDraw := false;
          end;
        end;

      SDL_QUITEV:
        begin
        Display.Fade := 0;
        Display.NextScreenWithCheck := nil;
        Display.CheckOK := True;
        end;
{      SDL_MOUSEBUTTONDOWN:
        With Event.button Do
        Begin
          If State = SDL_BUTTON_LEFT Then
          Begin
            //
          End;
        End; // With}
      SDL_KEYDOWN:
        begin
          //ScreenShot hack. If Print is pressed-> Make screenshot and Save to Screenshots Path
          if (Event.key.keysym.sym = SDLK_SYSREQ) or (Event.key.keysym.sym = SDLK_PRINT) then
          begin
//            ScreenPopupError.ShowPopup('How dare you press the <Print> key'); //show error message
            if (SDL_GetModState and KMOD_LCTRL = KMOD_LCTRL) then
              Display.PrintScreen //jpeg
            else
              Display.ScreenShot; //bmp
          end
          // popup hack... if there is a visible popup then let it handle input instead of underlying screen
          // shoud be done in a way to be sure the topmost popup has preference (maybe error, then check)
          else if (ScreenPopupError <> NIL) and (ScreenPopupError.Visible) then
            done := not ScreenPopupError.ParseInput(Event.key.keysym.sym, Event.key.keysym.unicode, True)
          else if (ScreenPopupCheck <> NIL) AND (ScreenPopupCheck.Visible) then
            done := not ScreenPopupCheck.ParseInput(Event.key.keysym.sym, Event.key.keysym.unicode, True)
          else if (ScreenPopupHelp <> NIL) AND (ScreenPopupHelp.Visible) then
            done := not ScreenPopupHelp.ParseInput(Event.key.keysym.sym, Event.key.keysym.unicode, True)
          // end of popup hack

          else
          begin
            // check for Screen want to Exit
            done := Not Display.ActualScreen^.ParseInput(Event.key.keysym.sym, Event.key.keysym.unicode, True);

            //If Screen wants to Exit
            if done then
            begin
              //If Question Option is enabled then Show Exit Popup
              if (Ini.AskbeforeDel = 1) then
              begin
                Display.ActualScreen^.CheckFadeTo(NIL,'MSG_QUIT_USDX');
              end
              else //When asking for exit is disabled then simply exit
              begin
                Display.Fade := 0;
                Display.NextScreenWithCheck := nil;
                Display.CheckOK := True;
              end;
            end;

          end;            //        if (Not Display.ActualScreen^.ParseInput(Event.key.keysym.scancode, True)) then
        end;
//      SDL_JOYAXISMOTION:
//        begin
//          beep
//        end;
      SDL_JOYBUTTONDOWN:
        begin
          beep
        end;
    End; // Case Event.type_
  End; // While
End; // CheckEvents

function GetTimeForBeats(BPM, Beats: real): real;
begin
  Result := 60 / BPM * Beats;
end;

function GetBeats(BPM, msTime: real): real;
begin
  Result := BPM * msTime / 60;
end;

procedure GetMidBeatSub(BPMNum: integer; var Time: real; var CurBeat: real);
var
  NewTime:  real;
begin
  if High(AktSong.BPM) = BPMNum then
  begin
    // last BPM
    CurBeat := AktSong.BPM[BPMNum].StartBeat + GetBeats(AktSong.BPM[BPMNum].BPM, Time);
    Time := 0;
  end else
  begin
    // not last BPM
    // count how much time is it for start of the new BPM and store it in NewTime
    NewTime := GetTimeForBeats(AktSong.BPM[BPMNum].BPM, AktSong.BPM[BPMNum+1].StartBeat - AktSong.BPM[BPMNum].StartBeat);

    // compare it to remaining time
    if (Time - NewTime) > 0 then
    begin
      // there is still remaining time
      CurBeat := AktSong.BPM[BPMNum].StartBeat;
      Time := Time - NewTime;
    end else
    begin
      // there is no remaining time
      CurBeat := AktSong.BPM[BPMNum].StartBeat + GetBeats(AktSong.BPM[BPMNum].BPM, Time);
      Time := 0;
    end; // if
  end; // if
end;

function GetMidBeat(Time: real): real;
var
  CurBeat:  real;
  CurBPM:   integer;

begin
  Result := 0;
  if Length(AktSong.BPM) = 1 then Result := Time * AktSong.BPM[0].BPM / 60;

  (* more BPMs *)
  if Length(AktSong.BPM) > 1 then
  begin

    CurBeat := 0;
    CurBPM := 0;
    while (Time > 0) do
    begin
      GetMidBeatSub(CurBPM, Time, CurBeat);
      Inc(CurBPM);
    end;

    Result := CurBeat;
  end; // if
end;

function GetTimeFromBeat(Beat: integer): real;
var
  CurBPM:   integer;
begin
  Result := 0;
  if Length(AktSong.BPM) = 1 then Result := AktSong.GAP / 1000 + Beat * 60 / AktSong.BPM[0].BPM;

  (* more BPMs *)
  if Length(AktSong.BPM) > 1 then
  begin
    Result := AktSong.GAP / 1000;
    CurBPM := 0;
    while (CurBPM <= High(AktSong.BPM)) and (Beat > AktSong.BPM[CurBPM].StartBeat) do
    begin
      if (CurBPM < High(AktSong.BPM)) and (Beat >= AktSong.BPM[CurBPM+1].StartBeat) then
      begin
        // full range
        Result := Result + (60 / AktSong.BPM[CurBPM].BPM) * (AktSong.BPM[CurBPM+1].StartBeat - AktSong.BPM[CurBPM].StartBeat);
      end;

      if (CurBPM = High(AktSong.BPM)) or (Beat < AktSong.BPM[CurBPM+1].StartBeat) then
      begin
        // in the middle
        Result := Result + (60 / AktSong.BPM[CurBPM].BPM) * (Beat - AktSong.BPM[CurBPM].StartBeat);
      end;
      Inc(CurBPM);
    end;
  end; // if}
end;

procedure Sing(Sender: TScreenSing);
var
  Pet:    integer;
  PetGr:  integer;
  CP:     integer;
  Done:   real;
  N:      integer;
begin
  //Czas.Teraz := Czas.Teraz + TimeSkip;
  Czas.Teraz := Music.Position+Ini.LipSync*0.01;

  Czas.OldBeat := Czas.AktBeat;
  Czas.MidBeat := GetMidBeat(Czas.Teraz - (AktSong.Gap{ + 90 I've forgotten for what it is}) / 1000); // new system with variable BPM in function
  Czas.AktBeat := Floor(Czas.MidBeat);

  Czas.OldBeatC := Czas.AktBeatC;
  Czas.MidBeatC := GetMidBeat(Czas.Teraz - (AktSong.Gap) / 1000);
  Czas.AktBeatC := Floor(Czas.MidBeatC);

  Czas.OldBeatD := Czas.AktBeatD;
  Czas.MidBeatD := -0.5+GetMidBeat(Czas.Teraz - Ini.LipSync*0.01 - (AktSong.Gap + 120 + Ini.Delay*10) / 1000); // MidBeat with addition GAP
  Czas.AktBeatD := Floor(Czas.MidBeatD);
  Czas.FracBeatD := Frac(Czas.MidBeatD);

  // sentences routines
  PetGr := 0;
  if AktSong.isDuet then
    PetGr := 1;
    
  for CP := 0 to PetGr do
  begin
    // ustawianie starej czesci
    Czas.OldCzesc[CP] := Czesci[CP].Akt;

    // wybieranie aktualnej czesci
    for Pet := 0 to Czesci[CP].High do
    begin
      if Czas.AktBeat >= Czesci[CP].Czesc[Pet].Start then
      begin
        if (GetTimeFromBeat(Czesci[CP].Czesc[Pet].StartNote) <= Czas.Teraz+10) then
          Czesci[CP].Akt := Pet;
      end;
    end;

    // czysczenie nut gracza, gdy to jest nowa plansza
    // (optymizacja raz na halfbeat jest zla)
    if Czesci[CP].Akt <> Czas.OldCzesc[CP] then
      NewSentence(CP, Sender);

    // wykonuje operacje raz na beat
    if (Czas.AktBeat >= 0) and (Czas.OldBeat <> Czas.AktBeat) then
      NewBeat(CP, Sender);

    // make some operations on clicks
    if {(Czas.AktBeatC >= 0) and }(Czas.OldBeatC <> Czas.AktBeatC) then
      NewBeatC(CP, Sender);

    // make some operations when detecting new voice pitch
    if (Czas.AktBeatD >= 0) and (Czas.OldBeatD <> Czas.AktBeatD) then
      NewBeatD(CP, Sender);

    // plynnie przesuwa text
    Done := 1;
    if (Length(Czesci[CP].Czesc[Czesci[CP].Akt].Nuta)>0) then
    begin
      for N := 0 to Czesci[CP].Czesc[Czesci[CP].Akt].HighNut do
        if (Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Start <= Czas.MidBeat)
        and (Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Start + Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Dlugosc >= Czas.MidBeat) then
          Done := (Czas.MidBeat - Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Start) / (Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Dlugosc);

      N := Czesci[CP].Czesc[Czesci[CP].Akt].HighNut;

      // wylacza ostatnia nute po przejsciu
      if (Ini.LyricsEffect = 1) and (Done = 1) and
        (Czas.MidBeat > Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Start + Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[N].Dlugosc)
        then
      begin
        Sender.LyricMain[CP].Selected := -1
      end;

      if Done > 1 then
        Done := 1;

      Sender.LyricMain[CP].Done := Done;
    end;
  end;
end;

procedure NewSentence(CP: integer; Sender: TScreenSing);
var
G: Integer;
begin
  // czyszczenie nut graczy
  if AktSong.isDuet then
  begin
    for G := 0 to High(Player) do
    begin
      if (G mod 2 = CP) then
      begin
        Player[G].IlNut := 0;
        Player[G].HighNut := -1;
        SetLength(Player[G].Nuta, 0);
      end;
    end;
  end else
  begin
    for G := 0 to High(Player) do
    begin
      Player[G].IlNut := 0;
      Player[G].HighNut := -1;
      SetLength(Player[G].Nuta, 0);
    end;
  end;

  // wstawianie tekstow
  with Sender do
  begin
    LyricMain[CP].AddCzesc(CP, Czesci[CP].Akt);
    if Czesci[CP].Akt < Czesci[CP].High then
      LyricSub[CP].AddCzesc(CP, Czesci[CP].Akt+1)
    else
      LyricSub[CP].Clear;
  end;
  
  //On Sentence Change...
  Sender.onSentenceChange(CP, Czesci[CP].Akt);
end;

procedure NewBeat(CP: integer; Sender: TScreenSing);
var
  Pet:      integer;
//  TempBeat: integer;
begin
  if (Length(Czesci[CP].Czesc[Czesci[CP].Akt].Nuta)=0) then
    Exit;

  for Pet := 0 to Czesci[CP].Czesc[Czesci[CP].Akt].HighNut do
  begin
    if (Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[Pet].Start <= Czas.AktBeat) and
      (Sender.LyricMain[CP].Selected <> Pet) then
    begin
      // operates on currently beated note
      Sender.LyricMain[CP].Selected := Pet;
    end;
  end;
end;

procedure NewBeatC(CP: integer; Sender: TScreenSing);
var
  Pet:    integer;
//  LPT_1:  integer;
//  LPT_2:  integer;
begin
//  LPT_1 := 1;
//  LPT_2 := 1;

  // beat click
  if (Ini.BeatClick = 1) and ((Czas.AktBeatC + Czesci[CP].Resolution + Czesci[CP].NotesGAP) mod Czesci[CP].Resolution = 0) then
    Music.PlayClick;

  if (Length(Czesci[CP].Czesc[Czesci[CP].Akt].Nuta)=0) then
    Exit;
    
  for Pet := 0 to Czesci[CP].Czesc[Czesci[CP].Akt].HighNut do
    if (Czesci[CP].Czesc[Czesci[CP].Akt].Nuta[Pet].Start = Czas.AktBeatC) then
    begin
      // click assist
      if Ini.ClickAssist = 1 then
        Music.PlayClick;


      // drum machine
(*      TempBeat := Czas.AktBeat;// + 2;
      if (TempBeat mod 8 = 0) then Music.PlayDrum;
      if (TempBeat mod 8 = 4) then Music.PlayClap;
//      if (TempBeat mod 4 = 2) then Music.PlayHihat;
      if (TempBeat mod 4 <> 0) then Music.PlayHihat;*)
    end;

  //PortWriteB($378, LPT_1 + LPT_2 * 2); // 0 zapala
end;

procedure NewBeatD(CP: integer; Sender: TScreenSing);
begin
  NewNote(CP, Sender);
end;

//procedure NewHalf;
//begin
//  NewNote;
//end;

procedure NewNote(P: integer; Sender: TScreenSing);
const
  DEBUG_NOTE_HIT = false;

var
  CP:     integer; // current player
  S:      integer; // sentence
  N:      integer;
  SumN:   real;
  NumS:   integer;
  tap:    integer;
  SMin:   integer;
  SMax:   integer;
  SDet:   integer; // temporary: sentence of detected note
  BRange: integer; // beat range

  AktTon: integer;

  Pet:    integer;
  Mozna:  boolean;
  Nowa:   boolean;
  Range:  integer;
  NoteHit:boolean;
begin
  SDet := 0;

  for CP := 0 to PlayersPlay-1 do
  begin
    if (not AktSong.isDuet) then
    begin
      // analyze buffer
      Sound[CP].AnalizujBufor;

      // 0.5.0: count min and max sentence range for checking (detection is delayed to the notes we see on the screen)
      SMin := Czesci[P].Akt-1;
      if SMin < 0 then
        SMin := 0;
      SMax := Czesci[P].Akt;

      for BRange := Czas.OldBeatD+1 to Czas.AktBeatD do
      begin
        SDet := SMin;
        // check if we can add new note
        Mozna := false;
        for S := SMin to SMax do
        begin
          for Pet := 0 to Czesci[P].Czesc[S].HighNut do
          begin
            if ((Czesci[P].Czesc[S].Nuta[Pet].Start <= BRange)
              and (Czesci[P].Czesc[S].Nuta[Pet].Start + Czesci[P].Czesc[S].Nuta[Pet].Dlugosc - 1 >= BRange))
              and (not Czesci[P].Czesc[S].Nuta[Pet].FreeStyle) // but don't allow when it's FreeStyle note
              and (Czesci[P].Czesc[S].Nuta[Pet].Dlugosc > 0) // and make sure the note lenghts is at least 1
              then
            begin
              SDet := S;
              Mozna := true;
              Break;
            end;
          end;
        end;

        S := SDet;

        if (Sound[CP].SzczytJest or DEBUG_NOTE_HIT) and Mozna then
        begin
          // operate on the actual note
          for Pet := 0 to Czesci[P].Czesc[S].HighNut do
          begin
            if (Czesci[P].Czesc[S].Nuta[Pet].Start <= BRange)
              and (Czesci[P].Czesc[S].Nuta[Pet].Start +
              Czesci[P].Czesc[S].Nuta[Pet].Dlugosc > BRange) then
            begin
              // przesuwanie tonu w odpowiednia game => shifting tone in the corresponding game?
              while (Sound[CP].TonGamy - Czesci[P].Czesc[S].Nuta[Pet].Ton > 6) do
                Sound[CP].TonGamy := Sound[CP].TonGamy - 12;
              while (Sound[CP].TonGamy - Czesci[P].Czesc[S].Nuta[Pet].Ton < -6) do
                Sound[CP].TonGamy := Sound[CP].TonGamy + 12;

              // Half size Notes Patch
              NoteHit := false;
              AktTon := Sound[CP].TonGamy;

              Range := 2 - Ini.Difficulty;
              if (abs(Czesci[P].Czesc[S].Nuta[Pet].Ton - Sound[CP].TonGamy) <= Range) or
                DEBUG_NOTE_HIT then
              begin
                AktTon := Czesci[P].Czesc[S].Nuta[Pet].Ton;

                // Half size Notes Patch
                NoteHit := true;

                if (Ini.LineBonus = 0) then
                begin
                  // add points without LineBonus
                  case Czesci[P].Czesc[S].Nuta[Pet].Wartosc of
                    1:  Player[CP].Score := Player[CP].Score + 10000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                    2:  Player[CP].ScoreGolden := Player[CP].ScoreGolden + 10000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                  end;
                end else
                begin
                  // add points with Line Bonus
                  case Czesci[P].Czesc[S].Nuta[Pet].Wartosc of
                    1:  Player[CP].Score := Player[CP].Score + 9000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                    2:  Player[CP].ScoreGolden := Player[CP].ScoreGolden + 9000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                  end;
                end;

                Player[CP].ScoreI := Floor(Player[CP].Score / 10) * 10;
                Player[CP].ScoreGoldenI := Floor(Player[CP].ScoreGolden / 10) * 10;

                Player[CP].ScoreTotalI := Player[CP].ScoreI + Player[CP].ScoreGoldenI + Player[CP].ScoreLineI;
              end;
            end; // operowanie
          end;

          // checking whether a new note, or extend
          if S = SMax then
          begin
            Nowa := true;
            // if the latter has the same tone
            if (Player[CP].IlNut > 0 ) and (Player[CP].Nuta[Player[CP].HighNut].Ton = AktTon)
              and (Player[CP].Nuta[Player[CP].HighNut].Start + Player[CP].Nuta[Player[CP].HighNut].Dlugosc = BRange)
              then
              Nowa := false;

            // If there is a new note on the checked Beat
            for Pet := 0 to Czesci[P].Czesc[S].HighNut do
              if (Czesci[P].Czesc[S].Nuta[Pet].Start = BRange) then
                Nowa := true;

            // add a new note
            if Nowa then
            begin
              // nowa nuta
              Player[CP].IlNut := Player[CP].IlNut + 1;
              Player[CP].HighNut := Player[CP].HighNut + 1;
              SetLength(Player[CP].Nuta, Player[CP].IlNut);
              Player[CP].Nuta[Player[CP].HighNut].Start := BRange;
              Player[CP].Nuta[Player[CP].HighNut].Dlugosc := 1;
              Player[CP].Nuta[Player[CP].HighNut].Ton := AktTon; // Ton || TonDokl
              //Player[CP].Nuta[Player[CP].HighNut].Detekt := Czas.MidBeat;

              // Half Note Patch
              Player[CP].Nuta[Player[CP].HighNut].Hit := NoteHit;
            end else
            begin
              // extend notes
              Player[CP].Nuta[Player[CP].HighNut].Dlugosc := Player[CP].Nuta[Player[CP].HighNut].Dlugosc + 1;
            end;


            // check for perfect note and then lit the star (on Draw)
            for Pet := 0 to Czesci[P].Czesc[S].HighNut do
            begin
              if (Czesci[P].Czesc[S].Nuta[Pet].Start = Player[CP].Nuta[Player[CP].HighNut].Start)
                and (Czesci[P].Czesc[S].Nuta[Pet].Dlugosc = Player[CP].Nuta[Player[CP].HighNut].Dlugosc)
                and (Czesci[P].Czesc[S].Nuta[Pet].Ton = Player[CP].Nuta[Player[CP].HighNut].Ton) then
              begin
                Player[CP].Nuta[Player[CP].HighNut].Perfect := true;
              end;

            end;// else beep; // if S = SMax
          end; //for
        end; // if moze
      end; // for BRange
      //calc score last
      SumN := 0;
      NumS := 0;
      for S := Czesci[P].Akt to Czesci[P].High do
      begin
        for N := 0 to Czesci[P].Czesc[S].HighNut do
        begin
          if (Czesci[P].Czesc[S].Nuta[N].Start > Czas.AktBeatD) then
          begin
            tap := Czesci[P].Czesc[S].Nuta[N].Dlugosc;
            if (Czesci[P].Czesc[S].Nuta[N].Start + tap < Czas.AktBeatD) then
              tap := Czas.AktBeatD - Czesci[P].Czesc[S].Nuta[N].Start - tap;

            if (tap<>0) then
            begin
              if (Ini.LineBonus = 0) then
                // add points without LineBonus
                SumN := SumN + 10000 / Czesci[P].Wartosc * Czesci[P].Czesc[S].Nuta[N].Wartosc * tap
              else
                // add points with Line Bonus
                SumN := SumN + 9000 / Czesci[P].Wartosc * Czesci[P].Czesc[S].Nuta[N].Wartosc * tap;
            end;
          end;
        end;

        if (Czesci[P].Czesc[S].TotalNotes>0) and
          (Czesci[P].Czesc[S].Koniec > Czas.AktBeatD) then
          Inc(NumS);
      end;

      N := (Czesci[P].Ilosc - ScreenSing.NumEmptySentences[P]);
      if (N>0) and (Ini.LineBonus > 0) then
        Player[CP].ScoreMax := Floor((SumN + NumS*1000 / N)/10)*10;

      Player[CP].ScoreMax := Player[CP].ScoreTotalI + Player[CP].ScoreMax;

    end else
    begin     //############################ DUET #####################
      if (CP mod 2 = P) then
      begin
      // analyze buffer
      Sound[CP].AnalizujBufor;

      // 0.5.0: count min and max sentence range for checking (detection is delayed to the notes we see on the screen)
      SMin := Czesci[P].Akt-1;
      if SMin < 0 then
        SMin := 0;
      SMax := Czesci[P].Akt;

      for BRange := Czas.OldBeatD+1 to Czas.AktBeatD do
      begin
        SDet := SMin;
        // check if we can add new note
        Mozna := false;
        for S := SMin to SMax do
        begin
          for Pet := 0 to Czesci[P].Czesc[S].HighNut do
          begin
            if ((Czesci[P].Czesc[S].Nuta[Pet].Start <= BRange)
              and (Czesci[P].Czesc[S].Nuta[Pet].Start + Czesci[P].Czesc[S].Nuta[Pet].Dlugosc - 1 >= BRange))
              and (not Czesci[P].Czesc[S].Nuta[Pet].FreeStyle) // but don't allow when it's FreeStyle note
              and (Czesci[P].Czesc[S].Nuta[Pet].Dlugosc > 0) // and make sure the note lenghts is at least 1
              then
            begin
              SDet := S;
              Mozna := true;
              Break;
            end;
          end;
        end;

        S := SDet;

        if (Sound[CP].SzczytJest or DEBUG_NOTE_HIT) and (Mozna) then
        begin
          // operowanie na ostatniej nucie
          for Pet := 0 to Czesci[P].Czesc[S].HighNut do
          begin
            if (Czesci[P].Czesc[S].Nuta[Pet].Start <= BRange)
              and (Czesci[P].Czesc[S].Nuta[Pet].Start +
              Czesci[P].Czesc[S].Nuta[Pet].Dlugosc > BRange) then
            begin
              // przesuwanie tonu w odpowiednia game
              while (Sound[CP].TonGamy - Czesci[P].Czesc[S].Nuta[Pet].Ton > 6) do
                Sound[CP].TonGamy := Sound[CP].TonGamy - 12;
              while (Sound[CP].TonGamy - Czesci[P].Czesc[S].Nuta[Pet].Ton < -6) do
                Sound[CP].TonGamy := Sound[CP].TonGamy + 12;

              // Half size Notes Patch
              NoteHit := false;
              AktTon := Sound[CP].TonGamy;
              
              Range := 2 - Ini.Difficulty;
              if (abs(Czesci[P].Czesc[S].Nuta[Pet].Ton - Sound[CP].TonGamy) <= Range) or
                DEBUG_NOTE_HIT then
              begin
                AktTon := Czesci[P].Czesc[S].Nuta[Pet].Ton;

                // Half size Notes Patch
                NoteHit := true;

                if (Ini.LineBonus = 0) then
                begin
                  // add points without LineBonus
                  case Czesci[P].Czesc[S].Nuta[Pet].Wartosc of
                    1:  Player[CP].Score := Player[CP].Score + 10000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                    2:  Player[CP].ScoreGolden := Player[CP].ScoreGolden + 10000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                  end;
                end else
                begin
                  // add points with Line Bonus
                  case Czesci[P].Czesc[S].Nuta[Pet].Wartosc of
                    1:  Player[CP].Score := Player[CP].Score + 9000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                    2:  Player[CP].ScoreGolden := Player[CP].ScoreGolden + 9000 / Czesci[P].Wartosc *
                          Czesci[P].Czesc[S].Nuta[Pet].Wartosc;
                  end;
                end;

                Player[CP].ScoreI := Floor(Player[CP].Score / 10) * 10;
                Player[CP].ScoreGoldenI := Floor(Player[CP].ScoreGolden / 10) * 10;

                Player[CP].ScoreTotalI := Player[CP].ScoreI + Player[CP].ScoreGoldenI + Player[CP].ScoreLineI;
              end;
            end; // operowanie
          end;

          // sprawdzanie czy to nowa nuta, czy przedluzenie
          if S = SMax then
          begin
            Nowa := true;
            // jezeli ostatnia ma ten sam ton
            if (Player[CP].IlNut > 0 ) and (Player[CP].Nuta[Player[CP].HighNut].Ton = AktTon)
              and (Player[CP].Nuta[Player[CP].HighNut].Start + Player[CP].Nuta[Player[CP].HighNut].Dlugosc = BRange)
              then
              Nowa := false;

            // jezeli jest jakas nowa nuta na sprawdzanym beacie
            for Pet := 0 to Czesci[P].Czesc[S].HighNut do
              if (Czesci[P].Czesc[S].Nuta[Pet].Start = BRange) then
                Nowa := true;

            // dodawanie nowej nuty
            if Nowa then
            begin
              // nowa nuta
              Player[CP].IlNut := Player[CP].IlNut + 1;
              Player[CP].HighNut := Player[CP].HighNut + 1;
              SetLength(Player[CP].Nuta, Player[CP].IlNut);
              Player[CP].Nuta[Player[CP].HighNut].Start := BRange;
              Player[CP].Nuta[Player[CP].HighNut].Dlugosc := 1;
              Player[CP].Nuta[Player[CP].HighNut].Ton := AktTon; // Ton || TonDokl
              //Player[CP].Nuta[Player[CP].HighNut].Detekt := Czas.MidBeat;

              // Half Note Patch
              Player[CP].Nuta[Player[CP].HighNut].Hit := NoteHit;
            end else
            begin
              // przedluzenie nuty
              Player[CP].Nuta[Player[CP].HighNut].Dlugosc := Player[CP].Nuta[Player[CP].HighNut].Dlugosc + 1;
            end;


            // check for perfect note and then lit the star (on Draw)
            for Pet := 0 to Czesci[P].Czesc[S].HighNut do
            begin
              if (Czesci[P].Czesc[S].Nuta[Pet].Start = Player[CP].Nuta[Player[CP].HighNut].Start)
                and (Czesci[P].Czesc[S].Nuta[Pet].Dlugosc = Player[CP].Nuta[Player[CP].HighNut].Dlugosc)
                and (Czesci[P].Czesc[S].Nuta[Pet].Ton = Player[CP].Nuta[Player[CP].HighNut].Ton) then
              begin
                Player[CP].Nuta[Player[CP].HighNut].Perfect := true;
              end;

            end;// else beep; // if S = SMax
          end; //for
        end; // if moze
      end; // for BRange

      //calc score last
      SumN := 0;
      NumS := 0;
      for S := Czesci[P].Akt to Czesci[P].High do
      begin
        for N := 0 to Czesci[P].Czesc[S].HighNut do
        begin
          if (Czesci[P].Czesc[S].Nuta[N].Start > Czas.AktBeatD) then
          begin
            tap := Czesci[P].Czesc[S].Nuta[N].Dlugosc;
            if (Czesci[P].Czesc[S].Nuta[N].Start + tap < Czas.AktBeatD) then
              tap := Czas.AktBeatD - Czesci[P].Czesc[S].Nuta[N].Start - tap;

            if (tap<>0) then
            begin
              if (Ini.LineBonus = 0) then
                // add points without LineBonus
                SumN := SumN + 10000 / Czesci[P].Wartosc * Czesci[P].Czesc[S].Nuta[N].Wartosc * tap
              else
                // add points with Line Bonus
                SumN := SumN + 9000 / Czesci[P].Wartosc * Czesci[P].Czesc[S].Nuta[N].Wartosc * tap;
            end;
          end;
        end;

        if (Czesci[P].Czesc[S].TotalNotes>0) and
          (Czesci[P].Czesc[S].Koniec > Czas.AktBeatD) then
          Inc(NumS);
      end;

      N := (Czesci[P].Ilosc - ScreenSing.NumEmptySentences[P]);
      if (N>0) and (Ini.LineBonus > 0) then
        Player[CP].ScoreMax := Floor((SumN + NumS*1000 / N)/10)*10;

      Player[CP].ScoreMax := Player[CP].ScoreTotalI + Player[CP].ScoreMax;
      end; //if mod 2
    end;

  end; // for CP

  //On Sentence End -> For LineBonus + SingBar
  if (sDet >= low(Czesci[P].Czesc)) AND (sDet <= high(Czesci[P].Czesc)) then
  begin
    if (Length(Czesci[P].Czesc[SDet].Nuta)>0) then
    begin
      if ((Czesci[P].Czesc[SDet].Nuta[Czesci[P].Czesc[SDet].HighNut].Start +
        Czesci[P].Czesc[SDet].Nuta[Czesci[P].Czesc[SDet].HighNut].Dlugosc - 1) = Czas.AktBeatD) then
        Sender.onSentenceEnd(P, sDet);
    end;
  end;
end;

procedure ClearScores(PlayerNum: integer);
begin
  Player[PlayerNum].Score := 0;
  Player[PlayerNum].ScoreI := 0;
  Player[PlayerNum].ScoreLine := 0;
  Player[PlayerNum].ScoreLineI := 0;
  Player[PlayerNum].ScoreGolden := 0;
  Player[PlayerNum].ScoreGoldenI := 0;
  Player[PlayerNum].ScoreTotalI := 0;


  //SingBar Mod
  Player[PlayerNum].ScoreLast := 0;
  Player[PlayerNum].ScorePercent := 50;// Sets to 50% when song starts
  Player[PlayerNum].ScorePercentTarget := 50;// Sets to 50% when song starts
  //end SingBar Mod

  Player[PlayerNum].ScoreMax := 9990;

  //PhrasenBonus - Line Bonus Mod
  Player[PlayerNum].LineBonus_Visible := False; //Hide Line Bonus
  Player[PlayerNum].LineBonus_Alpha   := 0;
  //Player[PlayerNum].LineBonus_TargetX := 70 + PlayerNum*500;  //will be done by onShow
  //Player[PlayerNum].LineBonus_TargetY := 30;                  //will be done by onShow
  //PhrasenBonus - Line Bonus Mod End

end;

end.