//------------------------------------------------------------------------- /* Copyright (C) 2020 Christoph Oelckers This 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; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ //------------------------------------------------------------------------- #include "build.h" #include "interpolate.h" #include "xs_Float.h" #include "serializer.h" #include "gamecvars.h" struct Interpolation { double old, bak; int index; int type; }; static TArray interpolations; double Get(int index, int type) { switch(type) { case Interp_Sect_Floorz: return sector[index].floorz; case Interp_Sect_Ceilingz: return sector[index].ceilingz; case Interp_Sect_Floorheinum: return sector[index].floorheinum; case Interp_Sect_Ceilingheinum: return sector[index].ceilingheinum; case Interp_Sect_FloorPanX: return sector[index].floorxpan_; case Interp_Sect_FloorPanY: return sector[index].floorypan_; case Interp_Sect_CeilingPanX: return sector[index].ceilingxpan_; case Interp_Sect_CeilingPanY: return sector[index].ceilingypan_; case Interp_Wall_X: return wall[index].x; case Interp_Wall_Y: return wall[index].y; case Interp_Wall_PanX: return wall[index].xpan_; case Interp_Wall_PanY: return wall[index].ypan_; case Interp_Sprite_Z: return sprite[index].z; default: return 0; } } void Set(int index, int type, double val) { int old; switch(type) { case Interp_Sect_Floorz: sector[index].floorz = xs_CRoundToInt(val); break; case Interp_Sect_Ceilingz: sector[index].ceilingz = xs_CRoundToInt(val); break; case Interp_Sect_Floorheinum: sector[index].floorheinum = (short)xs_CRoundToInt(val); break; case Interp_Sect_Ceilingheinum: sector[index].ceilingheinum = (short)xs_CRoundToInt(val); break; case Interp_Sect_FloorPanX: sector[index].floorxpan_ = float(val); break; case Interp_Sect_FloorPanY: sector[index].floorypan_ = float(val); break; case Interp_Sect_CeilingPanX: sector[index].ceilingxpan_ = float(val); break; case Interp_Sect_CeilingPanY: sector[index].ceilingypan_ = float(val); break; case Interp_Wall_X: old = wall[index].x; wall[index].x = xs_CRoundToInt(val); if (wall[index].x != old) sector[wall[index].sector].dirty = 255; break; case Interp_Wall_Y: old = wall[index].y; wall[index].y = xs_CRoundToInt(val); if (wall[index].y != old) sector[wall[index].sector].dirty = 255; break; case Interp_Wall_PanX: wall[index].xpan_ = float(val); break; case Interp_Wall_PanY: wall[index].ypan_ = float(val); break; case Interp_Sprite_Z: sprite[index].z = xs_CRoundToInt(val); break; } } void StartInterpolation(int index, int type) { for (unsigned i = 0; i < interpolations.Size(); i++) { if (interpolations[i].index == index && interpolations[i].type == type) return; } int n = interpolations.Reserve(1); interpolations[n].index = index; interpolations[n].type = type; interpolations[n].old = Get(index, type); } void StopInterpolation(int index, int type) { for (unsigned i = 0; i < interpolations.Size(); i++) { if (interpolations[i].index == index && interpolations[i].type == type) { interpolations[i] = interpolations.Last(); interpolations.Pop(); return; } } } void UpdateInterpolations() { for (unsigned i = 0; i < interpolations.Size(); i++) { interpolations[i].old = Get(interpolations[i].index, interpolations[i].type); } } void DoInterpolations(double smoothratio) { if (!cl_interpolate) return; for (unsigned i = 0; i < interpolations.Size(); i++) { double bak; interpolations[i].bak = bak = Get(interpolations[i].index, interpolations[i].type); double old = interpolations[i].old; if (interpolations[i].type < Interp_Pan_First || fabs(bak-old) < 128.) { Set(interpolations[i].index, interpolations[i].type, old + (bak - old) * smoothratio); } else { // with the panning types we need to check for potential wraparound. if (bak < old) bak += 256.; else old += 256; double cur = old + (bak - old) * smoothratio; if (cur >= 256.) cur -= 256.; Set(interpolations[i].index, interpolations[i].type, cur); } } } void RestoreInterpolations() { if (!cl_interpolate) return; for (unsigned i = 0; i < interpolations.Size(); i++) { Set(interpolations[i].index, interpolations[i].type, interpolations[i].bak); } } void ClearInterpolations() { interpolations.Clear(); } void ClearMovementInterpolations() { // This clears all movement interpolations. Needed for Blood which destroys its interpolations each frame. for (unsigned i = 0; i < interpolations.Size();) { switch (interpolations[i].type) { case Interp_Sect_Floorz: case Interp_Sect_Ceilingz: case Interp_Sect_Floorheinum: case Interp_Sect_Ceilingheinum: case Interp_Wall_X: case Interp_Wall_Y: interpolations[i] = interpolations.Last(); interpolations.Pop(); break; default: i++; break; } } } void setsectinterpolate(int sectnum) { int j, k, startwall, endwall; auto sect = §or[sectnum]; startwall = sect->wallptr; endwall = startwall + sect->wallnum; for (j = startwall; j < endwall; j++) { StartInterpolation(j, Interp_Wall_X); StartInterpolation(j, Interp_Wall_Y); k = wall[j].nextwall; if (k >= 0) { StartInterpolation(k, Interp_Wall_X); StartInterpolation(k, Interp_Wall_Y); k = wall[k].point2; StartInterpolation(k, Interp_Wall_X); StartInterpolation(k, Interp_Wall_Y); } } } void clearsectinterpolate(int sectnum) { short j, startwall, endwall; auto sect = §or[sectnum]; startwall = sect->wallptr; endwall = startwall + sect->wallnum; for (j = startwall; j < endwall; j++) { StopInterpolation(j, Interp_Wall_X); StopInterpolation(j, Interp_Wall_Y); if (wall[j].nextwall >= 0) { StopInterpolation(wall[j].nextwall, Interp_Wall_X); StopInterpolation(wall[j].nextwall, Interp_Wall_Y); } } } FSerializer& Serialize(FSerializer& arc, const char* keyname, Interpolation& w, Interpolation* def) { if (arc.BeginObject(keyname)) { arc ("index", w.index) ("type", w.type) .EndObject(); } if (arc.isReading()) { w.old = Get(w.index, w.type); } return arc; } void SerializeInterpolations(FSerializer& arc) { arc("interpolations", interpolations); }