// // Copyright (C) 1993-1996 by id Software, Inc. // // This source is available for distribution and/or modification // only under the terms of the DOOM Source Code License as // published by id Software. All rights reserved. // // The source is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // FITNESS FOR A PARTICULAR PURPOSE. See the DOOM Source Code License // for more details. // #include #include #include #include "p_lnspec.h" #include "templates.h" #include "doomdef.h" #include "m_swap.h" #include "i_system.h" #include "w_wad.h" #include "g_levellocals.h" #include "p_maputl.h" #include "swrenderer/things/r_visiblesprite.h" #include "swrenderer/things/r_voxel.h" #include "swrenderer/things/r_particle.h" #include "swrenderer/things/r_sprite.h" #include "swrenderer/things/r_wallsprite.h" #include "swrenderer/things/r_playersprite.h" #include "swrenderer/segments/r_drawsegment.h" #include "swrenderer/line/r_renderdrawsegment.h" #include "swrenderer/plane/r_visibleplane.h" #include "swrenderer/scene/r_portal.h" #include "swrenderer/scene/r_light.h" #include "swrenderer/viewport/r_viewport.h" #include "swrenderer/r_memory.h" #include "swrenderer/r_renderthread.h" EXTERN_CVAR(Bool, r_fullbrightignoresectorcolor); namespace swrenderer { void VisibleSprite::Render(RenderThread *thread) { short *clipbot = thread->clipbot; short *cliptop = thread->cliptop; VisibleSprite *spr = this; int i; int x1, x2; int r1, r2; short topclip, botclip; short *clip1, *clip2; FSWColormap *colormap = spr->Light.BaseColormap; int colormapnum = spr->Light.ColormapNum; F3DFloor *rover; Clip3DFloors *clip3d = thread->Clip3DFloors.get(); // [RH] Check for particles if (spr->IsParticle()) { // kg3D - reject invisible parts if ((clip3d->fake3D & FAKE3D_CLIPBOTTOM) && spr->gpos.Z <= clip3d->sclipBottom) return; if ((clip3d->fake3D & FAKE3D_CLIPTOP) && spr->gpos.Z >= clip3d->sclipTop) return; spr->Render(thread, nullptr, nullptr, 0, 0); return; } x1 = spr->x1; x2 = spr->x2; // [RH] Quickly reject sprites with bad x ranges. if (x1 >= x2) return; // Reject sprites outside the slice rendered by the thread if (x2 < thread->X1 || x1 > thread->X2) return; // [RH] Sprites split behind a one-sided line can also be discarded. if (spr->sector == nullptr) return; // kg3D - reject invisible parts if ((clip3d->fake3D & FAKE3D_CLIPBOTTOM) && spr->gzt <= clip3d->sclipBottom) return; if ((clip3d->fake3D & FAKE3D_CLIPTOP) && spr->gzb >= clip3d->sclipTop) return; // kg3D - correct colors now CameraLight *cameraLight = CameraLight::Instance(); if (!cameraLight->FixedColormap() && cameraLight->FixedLightLevel() < 0 && spr->sector->e && spr->sector->e->XFloor.lightlist.Size()) { if (!(clip3d->fake3D & FAKE3D_CLIPTOP)) { clip3d->sclipTop = spr->sector->ceilingplane.ZatPoint(ViewPos); } sector_t *sec = nullptr; FDynamicColormap *mybasecolormap = nullptr; for (i = spr->sector->e->XFloor.lightlist.Size() - 1; i >= 0; i--) { if (clip3d->sclipTop <= spr->sector->e->XFloor.lightlist[i].plane.Zat0()) { rover = spr->sector->e->XFloor.lightlist[i].caster; if (rover) { if (rover->flags & FF_DOUBLESHADOW && clip3d->sclipTop <= rover->bottom.plane->Zat0()) { break; } sec = rover->model; if (rover->flags & FF_FADEWALLS) { mybasecolormap = sec->ColorMap; } else { mybasecolormap = spr->sector->e->XFloor.lightlist[i].extra_colormap; } } break; } } // found new values, recalculate if (sec) { bool invertcolormap = (spr->RenderStyle.Flags & STYLEF_InvertOverlay) != 0; if (spr->RenderStyle.Flags & STYLEF_InvertSource) invertcolormap = !invertcolormap; if (RenderStyle == LegacyRenderStyles[STYLE_Add] && mybasecolormap->Fade != 0) { mybasecolormap = GetSpecialLights(mybasecolormap->Color, 0, mybasecolormap->Desaturate); } bool isFullBright = !foggy && (renderflags & RF_FULLBRIGHT); bool fadeToBlack = spr->RenderStyle == LegacyRenderStyles[STYLE_Add] && mybasecolormap->Fade != 0; int spriteshade = LIGHT2SHADE(sec->lightlevel + R_ActualExtraLight(spr->foggy)); Light.SetColormap(LightVisibility::Instance()->SpriteGlobVis() / MAX(MINZ, (double)spr->depth), spriteshade, mybasecolormap, isFullBright, invertcolormap, fadeToBlack); } } // [RH] Initialize the clipping arrays to their largest possible range // instead of using a special "not clipped" value. This eliminates // visual anomalies when looking down and should be faster, too. topclip = 0; botclip = viewheight; // killough 3/27/98: // Clip the sprite against deep water and/or fake ceilings. // [RH] rewrote this to be based on which part of the sector is really visible auto viewport = RenderViewport::Instance(); double scale = viewport->InvZtoScale * spr->idepth; double hzb = DBL_MIN, hzt = DBL_MAX; if (spr->IsVoxel() && spr->floorclip != 0) { hzb = spr->gzb; } if (spr->heightsec && !(spr->heightsec->MoreFlags & SECF_IGNOREHEIGHTSEC)) { // only things in specially marked sectors if (spr->FakeFlatStat != WaterFakeSide::AboveCeiling) { double hz = spr->heightsec->floorplane.ZatPoint(spr->gpos); int h = xs_RoundToInt(viewport->CenterY - (hz - ViewPos.Z) * scale); if (spr->FakeFlatStat == WaterFakeSide::BelowFloor) { // seen below floor: clip top if (!spr->IsVoxel() && h > topclip) { topclip = short(MIN(h, viewheight)); } hzt = MIN(hzt, hz); } else { // seen in the middle: clip bottom if (!spr->IsVoxel() && h < botclip) { botclip = MAX(0, h); } hzb = MAX(hzb, hz); } } if (spr->FakeFlatStat != WaterFakeSide::BelowFloor && !(spr->heightsec->MoreFlags & SECF_FAKEFLOORONLY)) { double hz = spr->heightsec->ceilingplane.ZatPoint(spr->gpos); int h = xs_RoundToInt(viewport->CenterY - (hz - ViewPos.Z) * scale); if (spr->FakeFlatStat == WaterFakeSide::AboveCeiling) { // seen above ceiling: clip bottom if (!spr->IsVoxel() && h < botclip) { botclip = MAX(0, h); } hzb = MAX(hzb, hz); } else { // seen in the middle: clip top if (!spr->IsVoxel() && h > topclip) { topclip = MIN(h, viewheight); } hzt = MIN(hzt, hz); } } } // killough 3/27/98: end special clipping for deep water / fake ceilings else if (!spr->IsVoxel() && spr->floorclip) { // [RH] Move floorclip stuff from R_DrawVisSprite to here //int clip = ((FLOAT2FIXED(CenterY) - FixedMul (spr->texturemid - (spr->pic->GetHeight() << FRACBITS) + spr->floorclip, spr->yscale)) >> FRACBITS); int clip = xs_RoundToInt(viewport->CenterY - (spr->texturemid - spr->pic->GetHeight() + spr->floorclip) * spr->yscale); if (clip < botclip) { botclip = MAX(0, clip); } } if (clip3d->fake3D & FAKE3D_CLIPBOTTOM) { if (!spr->IsVoxel()) { double hz = clip3d->sclipBottom; if (spr->fakefloor) { double floorz = spr->fakefloor->top.plane->Zat0(); if (ViewPos.Z > floorz && floorz == clip3d->sclipBottom) { hz = spr->fakefloor->bottom.plane->Zat0(); } } int h = xs_RoundToInt(viewport->CenterY - (hz - ViewPos.Z) * scale); if (h < botclip) { botclip = MAX(0, h); } } hzb = MAX(hzb, clip3d->sclipBottom); } if (clip3d->fake3D & FAKE3D_CLIPTOP) { if (!spr->IsVoxel()) { double hz = clip3d->sclipTop; if (spr->fakeceiling != nullptr) { double ceilingZ = spr->fakeceiling->bottom.plane->Zat0(); if (ViewPos.Z < ceilingZ && ceilingZ == clip3d->sclipTop) { hz = spr->fakeceiling->top.plane->Zat0(); } } int h = xs_RoundToInt(viewport->CenterY - (hz - ViewPos.Z) * scale); if (h > topclip) { topclip = short(MIN(h, viewheight)); } } hzt = MIN(hzt, clip3d->sclipTop); } if (topclip >= botclip) { spr->Light.BaseColormap = colormap; spr->Light.ColormapNum = colormapnum; return; } i = x2 - x1; clip1 = clipbot + x1; clip2 = cliptop + x1; do { *clip1++ = botclip; *clip2++ = topclip; } while (--i); // Scan drawsegs from end to start for obscuring segs. // The first drawseg that is closer than the sprite is the clip seg. DrawSegmentList *segmentlist = thread->DrawSegments.get(); for (unsigned int index = segmentlist->BeginIndex(); index != segmentlist->EndIndex(); index++) { DrawSegment *ds = segmentlist->Segment(index); // [ZZ] portal handling here //if (ds->CurrentPortalUniq != spr->CurrentPortalUniq) // continue; // [ZZ] WARNING: uncommenting the two above lines, totally breaks sprite clipping // kg3D - no clipping on fake segs if (ds->fake) continue; // determine if the drawseg obscures the sprite if (ds->x1 >= x2 || ds->x2 <= x1 || (!(ds->silhouette & SIL_BOTH) && ds->maskedtexturecol == nullptr && !ds->bFogBoundary)) { // does not cover sprite continue; } r1 = MAX(ds->x1, x1); r2 = MIN(ds->x2, x2); float neardepth, fardepth; if (!spr->IsWallSprite()) { if (ds->sz1 < ds->sz2) { neardepth = ds->sz1, fardepth = ds->sz2; } else { neardepth = ds->sz2, fardepth = ds->sz1; } } // Check if sprite is in front of draw seg: if ((!spr->IsWallSprite() && neardepth > spr->depth) || ((spr->IsWallSprite() || fardepth > spr->depth) && (spr->gpos.Y - ds->curline->v1->fY()) * (ds->curline->v2->fX() - ds->curline->v1->fX()) - (spr->gpos.X - ds->curline->v1->fX()) * (ds->curline->v2->fY() - ds->curline->v1->fY()) <= 0)) { RenderPortal *renderportal = thread->Portal.get(); // seg is behind sprite, so draw the mid texture if it has one if (ds->CurrentPortalUniq == renderportal->CurrentPortalUniq && // [ZZ] instead, portal uniq check is made here (ds->maskedtexturecol != nullptr || ds->bFogBoundary)) { RenderDrawSegment renderer(thread); renderer.Render(ds, r1, r2); } continue; } // clip this piece of the sprite // killough 3/27/98: optimized and made much shorter // [RH] Optimized further (at least for VC++; // other compilers should be at least as good as before) if (ds->silhouette & SIL_BOTTOM) //bottom sil { clip1 = clipbot + r1; clip2 = ds->sprbottomclip + r1 - ds->x1; i = r2 - r1; do { if (*clip1 > *clip2) *clip1 = *clip2; clip1++; clip2++; } while (--i); } if (ds->silhouette & SIL_TOP) // top sil { clip1 = cliptop + r1; clip2 = ds->sprtopclip + r1 - ds->x1; i = r2 - r1; do { if (*clip1 < *clip2) *clip1 = *clip2; clip1++; clip2++; } while (--i); } } // all clipping has been performed, so draw the sprite if (!spr->IsVoxel()) { spr->Render(thread, clipbot, cliptop, 0, 0); } else { // If it is completely clipped away, don't bother drawing it. if (cliptop[x2] >= clipbot[x2]) { for (i = x1; i < x2; ++i) { if (cliptop[i] < clipbot[i]) { break; } } if (i == x2) { spr->Light.BaseColormap = colormap; spr->Light.ColormapNum = colormapnum; return; } } // Add everything outside the left and right edges to the clipping array // for R_DrawVisVoxel(). if (x1 > 0) { fillshort(cliptop, x1, viewheight); } if (x2 < viewwidth - 1) { fillshort(cliptop + x2, viewwidth - x2, viewheight); } int minvoxely = spr->gzt <= hzt ? 0 : xs_RoundToInt((spr->gzt - hzt) / spr->yscale); int maxvoxely = spr->gzb > hzb ? INT_MAX : xs_RoundToInt((spr->gzt - hzb) / spr->yscale); spr->Render(thread, cliptop, clipbot, minvoxely, maxvoxely); } spr->Light.BaseColormap = colormap; spr->Light.ColormapNum = colormapnum; } }