X-Git-Url: http://deadsoftware.ru/gitweb?a=blobdiff_plain;f=src%2Fgame%2Fg_player.pas;h=9aa72a832c3c6d2cf52adfdb1ac70adcee3565cf;hb=ee1df96f7123dcf3cdf0b0527cf7a790ec9d8d64;hp=c1454c81d7b5a3454b011f716fc2460e87c8d314;hpb=847038a14751d192fcc4af172d7a963e28144c72;p=d2df-sdl.git diff --git a/src/game/g_player.pas b/src/game/g_player.pas index c1454c8..9aa72a8 100644 --- a/src/game/g_player.pas +++ b/src/game/g_player.pas @@ -310,7 +310,8 @@ type procedure BFGHit(); function GetFlag(Flag: Byte): Boolean; procedure SetFlag(Flag: Byte); - function DropFlag(Silent: Boolean = True): Boolean; + function DropFlag(Silent: Boolean = True; DoThrow: Boolean = False): Boolean; + function TryDropFlag(): Boolean; procedure AllRulez(Health: Boolean); procedure RestoreHealthArmor(); procedure FragCombo(); @@ -854,148 +855,40 @@ begin end; function g_Player_CreateFromState (st: TStream): Word; -var - a, i: Integer; - ok, Bot: Boolean; - b: Byte; + var a: Integer; ok, Bot: Boolean; pos: Int64; begin - result := 0; - if (st = nil) then exit; //??? + assert(st <> nil); - // Ñèãíàòóðà èãðîêà + // check signature and entity type + pos := st.Position; if not utils.checkSign(st, 'PLYR') then raise XStreamError.Create('invalid player signature'); if (utils.readByte(st) <> PLR_SAVE_VERSION) then raise XStreamError.Create('invalid player version'); - - // Áîò èëè ÷åëîâåê: Bot := utils.readBool(st); + st.Position := pos; + // find free player slot ok := false; - a := 0; - - // Åñòü ëè ìåñòî â gPlayers: - for a := 0 to High(gPlayers) do if (gPlayers[a] = nil) then begin ok := true; break; end; + for a := 0 to High(gPlayers) do + if gPlayers[a] = nil then + begin + ok := true; + break; + end; - // Íåò ìåñòà - ðàñøèðÿåì gPlayers + // allocate player slot if not ok then begin SetLength(gPlayers, Length(gPlayers)+1); a := High(gPlayers); end; - // Ñîçäàåì îáúåêò èãðîêà + // create entity and load state if Bot then gPlayers[a] := TBot.Create() else gPlayers[a] := TPlayer.Create(); - gPlayers[a].FIamBot := Bot; - gPlayers[a].FPhysics := True; - - // UID èãðîêà - gPlayers[a].FUID := utils.readWord(st); - // Èìÿ èãðîêà - gPlayers[a].FName := utils.readStr(st); - // Êîìàíäà - gPlayers[a].FTeam := utils.readByte(st); - gPlayers[a].FPreferredTeam := gPlayers[a].FTeam; - // Æèâ ëè - gPlayers[a].FAlive := utils.readBool(st); - // Èçðàñõîäîâàë ëè âñå æèçíè - gPlayers[a].FNoRespawn := utils.readBool(st); - // Íàïðàâëåíèå - b := utils.readByte(st); - if b = 1 then gPlayers[a].FDirection := TDirection.D_LEFT else gPlayers[a].FDirection := TDirection.D_RIGHT; // b = 2 - // Çäîðîâüå - gPlayers[a].FHealth := utils.readLongInt(st); - // Ôîðà - gPlayers[a].FHandicap := utils.readLongInt(st); - // Æèçíè - gPlayers[a].FLives := utils.readByte(st); - // Áðîíÿ - gPlayers[a].FArmor := utils.readLongInt(st); - // Çàïàñ âîçäóõà - gPlayers[a].FAir := utils.readLongInt(st); - // Çàïàñ ãîðþ÷åãî - gPlayers[a].FJetFuel := utils.readLongInt(st); - // Áîëü - gPlayers[a].FPain := utils.readLongInt(st); - // Óáèë - gPlayers[a].FKills := utils.readLongInt(st); - // Óáèë ìîíñòðîâ - gPlayers[a].FMonsterKills := utils.readLongInt(st); - // Ôðàãîâ - gPlayers[a].FFrags := utils.readLongInt(st); - // Ôðàãîâ ïîäðÿä - gPlayers[a].FFragCombo := utils.readByte(st); - // Âðåìÿ ïîñëåäíåãî ôðàãà - gPlayers[a].FLastFrag := utils.readLongWord(st); - // Ñìåðòåé - gPlayers[a].FDeath := utils.readLongInt(st); - // Êàêîé ôëàã íåñåò - gPlayers[a].FFlag := utils.readByte(st); - // Íàøåë ñåêðåòîâ - gPlayers[a].FSecrets := utils.readLongInt(st); - // Òåêóùåå îðóæèå - gPlayers[a].FCurrWeap := utils.readByte(st); - // Ñëåäóþùåå æåëàåìîå îðóæèå - gPlayers[a].FNextWeap := utils.readWord(st); - // ...è ïàóçà - gPlayers[a].FNextWeapDelay := utils.readByte(st); - // Âðåìÿ çàðÿäêè BFG - gPlayers[a].FBFGFireCounter := utils.readSmallInt(st); - // Áóôåð óðîíà - gPlayers[a].FDamageBuffer := utils.readLongInt(st); - // Ïîñëåäíèé óäàðèâøèé - gPlayers[a].FLastSpawnerUID := utils.readWord(st); - // Òèï ïîñëåäíåãî ïîëó÷åííîãî óðîíà - gPlayers[a].FLastHit := utils.readByte(st); - // Îáúåêò èãðîêà: - Obj_LoadState(@gPlayers[a].FObj, st); - // Òåêóùåå êîëè÷åñòâî ïàòðîíîâ - for i := A_BULLETS to A_HIGH do gPlayers[a].FAmmo[i] := utils.readWord(st); - // Ìàêñèìàëüíîå êîëè÷åñòâî ïàòðîíîâ - for i := A_BULLETS to A_HIGH do gPlayers[a].FMaxAmmo[i] := utils.readWord(st); - // Íàëè÷èå îðóæèÿ - for i := WP_FIRST to WP_LAST do gPlayers[a].FWeapon[i] := utils.readBool(st); - // Âðåìÿ ïåðåçàðÿäêè îðóæèÿ - for i := WP_FIRST to WP_LAST do gPlayers[a].FReloading[i] := utils.readWord(st); - // Íàëè÷èå ðþêçàêà - if utils.readBool(st) then Include(gPlayers[a].FRulez, R_ITEM_BACKPACK); - // Íàëè÷èå êðàñíîãî êëþ÷à - if utils.readBool(st) then Include(gPlayers[a].FRulez, R_KEY_RED); - // Íàëè÷èå çåëåíîãî êëþ÷à - if utils.readBool(st) then Include(gPlayers[a].FRulez, R_KEY_GREEN); - // Íàëè÷èå ñèíåãî êëþ÷à - if utils.readBool(st) then Include(gPlayers[a].FRulez, R_KEY_BLUE); - // Íàëè÷èå áåðñåðêà - if utils.readBool(st) then Include(gPlayers[a].FRulez, R_BERSERK); - // Âðåìÿ äåéñòâèÿ ñïåöèàëüíûõ ïðåäìåòîâ - for i := MR_SUIT to MR_MAX do gPlayers[a].FMegaRulez[i] := utils.readLongWord(st); - // Âðåìÿ äî ïîâòîðíîãî ðåñïàóíà, ñìåíû îðóæèÿ, èñîëüçîâàíèÿ, çàõâàòà ôëàãà - for i := T_RESPAWN to T_FLAGCAP do gPlayers[a].FTime[i] := utils.readLongWord(st); - - // Íàçâàíèå ìîäåëè: - gPlayers[a].FActualModelName := utils.readStr(st); - // Öâåò ìîäåëè - gPlayers[a].FColor.R := utils.readByte(st); - gPlayers[a].FColor.G := utils.readByte(st); - gPlayers[a].FColor.B := utils.readByte(st); - // Îáíîâëÿåì ìîäåëü èãðîêà - gPlayers[a].SetModel(gPlayers[a].FActualModelName); - - // Íåò ìîäåëè - ñîçäàíèå íåâîçìîæíî - if (gPlayers[a].FModel = nil) then - begin - gPlayers[a].Free(); - gPlayers[a] := nil; - g_FatalError(Format(_lc[I_GAME_ERROR_MODEL], [gPlayers[a].FActualModelName])); - exit; - end; - - // Åñëè êîìàíäíàÿ èãðà - êðàñèì ìîäåëü â öâåò êîìàíäû - if gGameSettings.GameMode in [GM_TDM, GM_CTF] then - gPlayers[a].FModel.Color := TEAMCOLOR[gPlayers[a].FTeam] - else - gPlayers[a].FModel.Color := gPlayers[a].FColor; + gPlayers[a].FPhysics := True; // ??? + gPlayers[a].LoadState(st); result := gPlayers[a].FUID; end; @@ -3488,7 +3381,8 @@ begin if SpawnerUID = FUID then begin // Ñàìîóáèëñÿ - if Srv and (DoFrags or (gGameSettings.GameMode = GM_TDM)) then + if Srv and (gGameSettings.GameMode = GM_TDM) then + Dec(gTeamStat[FTeam].Goals); begin Dec(FFrags); FLastFrag := 0; @@ -3984,6 +3878,8 @@ function TPlayer.PickItem(ItemType: Byte; arespawn: Boolean; var remove: Boolean var a: Boolean; + switchWeapon: Byte; + hadWeapon: Boolean; begin Result := False; if g_Game_IsClient then Exit; @@ -3991,7 +3887,6 @@ begin // a = true - ìåñòî ñïàâíà ïðåäìåòà: a := LongBool(gGameSettings.Options and GAME_OPTION_WEAPONSTAY) and arespawn; remove := not a; - case ItemType of ITEM_MEDKIT_SMALL: if (FHealth < PLAYER_HP_SOFT) or (FFireTime > 0) then @@ -4057,6 +3952,8 @@ begin ITEM_WEAPON_SAW: if (not FWeapon[WEAPON_SAW]) or ((not arespawn) and (gGameSettings.GameMode in [GM_DM, GM_TDM, GM_CTF])) then begin + hadWeapon := FWeapon[WEAPON_SAW]; + switchWeapon := WEAPON_SAW; FWeapon[WEAPON_SAW] := True; Result := True; if gFlash = 2 then Inc(FPickup, 5); @@ -4068,7 +3965,8 @@ begin begin // Íóæíî, ÷òîáû íå âçÿòü âñå ïóëè ñðàçó: if a and FWeapon[WEAPON_SHOTGUN1] then Exit; - + hadWeapon := FWeapon[WEAPON_SHOTGUN1]; + switchWeapon := WEAPON_SHOTGUN1; IncMax(FAmmo[A_SHELLS], 4, FMaxAmmo[A_SHELLS]); FWeapon[WEAPON_SHOTGUN1] := True; Result := True; @@ -4080,7 +3978,8 @@ begin if (FAmmo[A_SHELLS] < FMaxAmmo[A_SHELLS]) or not FWeapon[WEAPON_SHOTGUN2] then begin if a and FWeapon[WEAPON_SHOTGUN2] then Exit; - + hadWeapon := FWeapon[WEAPON_SHOTGUN2]; + switchWeapon := WEAPON_SHOTGUN2; IncMax(FAmmo[A_SHELLS], 4, FMaxAmmo[A_SHELLS]); FWeapon[WEAPON_SHOTGUN2] := True; Result := True; @@ -4092,7 +3991,8 @@ begin if (FAmmo[A_BULLETS] < FMaxAmmo[A_BULLETS]) or not FWeapon[WEAPON_CHAINGUN] then begin if a and FWeapon[WEAPON_CHAINGUN] then Exit; - + hadWeapon := FWeapon[WEAPON_CHAINGUN]; + switchWeapon := WEAPON_CHAINGUN; IncMax(FAmmo[A_BULLETS], 50, FMaxAmmo[A_BULLETS]); FWeapon[WEAPON_CHAINGUN] := True; Result := True; @@ -4104,7 +4004,8 @@ begin if (FAmmo[A_ROCKETS] < FMaxAmmo[A_ROCKETS]) or not FWeapon[WEAPON_ROCKETLAUNCHER] then begin if a and FWeapon[WEAPON_ROCKETLAUNCHER] then Exit; - + switchWeapon := WEAPON_ROCKETLAUNCHER; + hadWeapon := FWeapon[WEAPON_ROCKETLAUNCHER]; IncMax(FAmmo[A_ROCKETS], 2, FMaxAmmo[A_ROCKETS]); FWeapon[WEAPON_ROCKETLAUNCHER] := True; Result := True; @@ -4116,7 +4017,8 @@ begin if (FAmmo[A_CELLS] < FMaxAmmo[A_CELLS]) or not FWeapon[WEAPON_PLASMA] then begin if a and FWeapon[WEAPON_PLASMA] then Exit; - + switchWeapon := WEAPON_PLASMA; + hadWeapon := FWeapon[WEAPON_PLASMA]; IncMax(FAmmo[A_CELLS], 40, FMaxAmmo[A_CELLS]); FWeapon[WEAPON_PLASMA] := True; Result := True; @@ -4128,7 +4030,8 @@ begin if (FAmmo[A_CELLS] < FMaxAmmo[A_CELLS]) or not FWeapon[WEAPON_BFG] then begin if a and FWeapon[WEAPON_BFG] then Exit; - + switchWeapon := WEAPON_BFG; + hadWeapon := FWeapon[WEAPON_BFG]; IncMax(FAmmo[A_CELLS], 40, FMaxAmmo[A_CELLS]); FWeapon[WEAPON_BFG] := True; Result := True; @@ -4140,7 +4043,8 @@ begin if (FAmmo[A_SHELLS] < FMaxAmmo[A_SHELLS]) or not FWeapon[WEAPON_SUPERPULEMET] then begin if a and FWeapon[WEAPON_SUPERPULEMET] then Exit; - + switchWeapon := WEAPON_SUPERPULEMET; + hadWeapon := FWeapon[WEAPON_SUPERPULEMET]; IncMax(FAmmo[A_SHELLS], 4, FMaxAmmo[A_SHELLS]); FWeapon[WEAPON_SUPERPULEMET] := True; Result := True; @@ -4152,7 +4056,8 @@ begin if (FAmmo[A_FUEL] < FMaxAmmo[A_FUEL]) or not FWeapon[WEAPON_FLAMETHROWER] then begin if a and FWeapon[WEAPON_FLAMETHROWER] then Exit; - + switchWeapon := WEAPON_FLAMETHROWER; + hadWeapon := FWeapon[WEAPON_FLAMETHROWER]; IncMax(FAmmo[A_FUEL], 100, FMaxAmmo[A_FUEL]); FWeapon[WEAPON_FLAMETHROWER] := True; Result := True; @@ -4326,11 +4231,33 @@ begin if not (R_BERSERK in FRulez) then begin Include(FRulez, R_BERSERK); + (* if allowBerserkSwitching then begin FCurrWeap := WEAPON_KASTET; resetWeaponQueue(); FModel.SetWeapon(WEAPON_KASTET); + end; *) + if ( (g_Game_IsNet = False) or (NetMode = NET_SERVER) ) and ( ( (Self = gPlayer1) and (gPlayer1Settings.WeaponSwitch <> 0) ) or ( (gPlayer2 <> nil) and (Self = gPlayer2) and (gPlayer2Settings.WeaponSwitch <> 0) )) then + begin + if (Self = gPlayer1) then + begin + if (gPlayer1Settings.WeaponSwitch = 1) or ( (gPlayer1Settings.WeaponSwitch = 2) and (gPlayer1Settings.WeaponPreferences[WP_LAST+1] > gPlayer1Settings.WeaponPreferences[FCurrWeap]) ) then + begin + FCurrWeap := WEAPON_KASTET; + resetWeaponQueue(); + FModel.SetWeapon(WEAPON_KASTET); + end; + end + else + begin + if (gPlayer2Settings.WeaponSwitch = 1) or ( (gPlayer2Settings.WeaponSwitch = 2) and (gPlayer2Settings.WeaponPreferences[WP_LAST+1] > gPlayer2Settings.WeaponPreferences[FCurrWeap]) ) then + begin + FCurrWeap := WEAPON_KASTET; + resetWeaponQueue(); + FModel.SetWeapon(WEAPON_KASTET); + end; + end; end; if gFlash <> 0 then begin @@ -4399,6 +4326,26 @@ begin if gFlash = 2 then Inc(FPickup, 5); end; end; + if ( (g_Game_IsNet = False) or (NetMode = NET_SERVER) ) and ( ( (Self = gPlayer1) and (gPlayer1Settings.WeaponSwitch <> 0) ) or ( (gPlayer2 <> nil) and (Self = gPlayer2) and (gPlayer2Settings.WeaponSwitch <> 0) )) then + begin + if (hadWeapon = False) then + begin + if (Self = gPlayer1) and ( (gPlayer1Settings.WeaponSwitch = 1) or ( (gPlayer1Settings.WeaponSwitch = 2) + and (gPlayer1Settings.WeaponPreferences[switchWeapon] > gPlayer1Settings.WeaponPreferences[FCurrWeap]) ) ) then + begin + FCurrWeap := switchWeapon; + resetWeaponQueue(); + FModel.SetWeapon(switchWeapon); + end + else if (Self = gPlayer2) and ( (gPlayer2Settings.WeaponSwitch = 1) or ( (gPlayer2Settings.WeaponSwitch = 2) + and (gPlayer2Settings.WeaponPreferences[switchWeapon] > gPlayer2Settings.WeaponPreferences[FCurrWeap]) ) ) then + begin + FCurrWeap := switchWeapon; + resetWeaponQueue(); + FModel.SetWeapon(switchWeapon); + end; + end; + end; end; procedure TPlayer.Touch(); @@ -5034,7 +4981,7 @@ begin DoLerp(4); if NetServer then - if FClientID >= 0 then + if (FClientID >= 0) and (NetClients[FClientID].Peer <> nil) then begin FPing := NetClients[FClientID].Peer^.lastRoundTripTime; if NetClients[FClientID].Peer^.packetsSent > 0 then @@ -5871,10 +5818,19 @@ begin FModel.SetFlag(FFlag); end; -function TPlayer.DropFlag(Silent: Boolean = True): Boolean; +function TPlayer.TryDropFlag(): Boolean; +begin + if LongBool(gGameSettings.Options and GAME_OPTION_ALLOWDROPFLAG) then + Result := DropFlag(False, LongBool(gGameSettings.Options and GAME_OPTION_THROWFLAG)) + else + Result := False; +end; + +function TPlayer.DropFlag(Silent: Boolean = True; DoThrow: Boolean = False): Boolean; var s: String; a: Byte; + xv, yv: Integer; begin Result := False; if (not g_Game_IsServer) or (FFlag = FLAG_NONE) then @@ -5887,8 +5843,18 @@ begin Direction := FDirection; State := FLAG_STATE_DROPPED; Count := FLAG_TIME; - g_Obj_Push(@Obj, (FObj.Vel.X div 2)-2+Random(5), - (FObj.Vel.Y div 2)-2+Random(5)); + if DoThrow then + begin + xv := FObj.Vel.X + IfThen(Direction = TDirection.D_RIGHT, 10, -10); + yv := FObj.Vel.Y - 2; + end + else + begin + xv := (FObj.Vel.X div 2); + yv := (FObj.Vel.Y div 2) - 2; + end; + g_Obj_Push(@Obj, xv, yv); + positionChanged(); // this updates spatial accelerators if FFlag = FLAG_RED then