#pragma once #include "d3drmrenderer.h" #include "d3drmtexture_impl.h" #include "ddraw_impl.h" #include #include #include DEFINE_GUID(SOFTWARE_GUID, 0x682656F3, 0x0000, 0x0000, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02); struct TextureCache { Direct3DRMTextureImpl* texture; Uint8 version; SDL_Surface* cached; }; struct MeshCache { const MeshGroup* meshGroup; int version; bool flat; std::vector vertices; std::vector indices; }; class Direct3DRMSoftwareRenderer : public Direct3DRMRenderer { public: Direct3DRMSoftwareRenderer(DWORD width, DWORD height); ~Direct3DRMSoftwareRenderer() override; void PushLights(const SceneLight* vertices, size_t count) override; Uint32 GetTextureId(IDirect3DRMTexture* texture, bool isUi) override; Uint32 GetMeshId(IDirect3DRMMesh* mesh, const MeshGroup* meshGroup) override; void SetProjection(const D3DRMMATRIX4D& projection, D3DVALUE front, D3DVALUE back) override; void SetFrustumPlanes(const Plane* frustumPlanes) override; HRESULT BeginFrame() override; void EnableTransparency() override; void SubmitDraw( DWORD meshId, const D3DRMMATRIX4D& modelViewMatrix, const D3DRMMATRIX4D& worldMatrix, const D3DRMMATRIX4D& viewMatrix, const Matrix3x3& normalMatrix, const Appearance& appearance ) override; HRESULT FinalizeFrame() override; void Resize(int width, int height, const ViewportTransform& viewportTransform) override; void Clear(float r, float g, float b) override; void Flip() override; void Draw2DImage(Uint32 textureId, const SDL_Rect& srcRect, const SDL_Rect& dstRect, FColor color) override; void Download(SDL_Surface* target) override; void SetDither(bool dither) override; private: void ClearZBuffer(); void DrawTriangleProjected( const D3DRMVERTEX& v0, const D3DRMVERTEX& v1, const D3DRMVERTEX& v2, const Appearance& appearance ); void DrawTriangleClipped(const D3DRMVERTEX (&v)[3], const Appearance& appearance); void ProjectVertex(const D3DVECTOR& v, D3DRMVECTOR4D& p) const; Uint32 BlendPixel(Uint8* pixelAddr, Uint8 r, Uint8 g, Uint8 b, Uint8 a); SDL_Color ApplyLighting(const D3DVECTOR& position, const D3DVECTOR& normal, const Appearance& appearance); void AddTextureDestroyCallback(Uint32 id, IDirect3DRMTexture* texture); void AddMeshDestroyCallback(Uint32 id, IDirect3DRMMesh* mesh); SDL_Surface* m_renderedImage = nullptr; SDL_Palette* m_palette; SDL_Texture* m_uploadBuffer = nullptr; SDL_Renderer* m_renderer; const SDL_PixelFormatDetails* m_format; int m_bytesPerPixel; std::vector m_lights; std::vector m_textures; std::vector m_meshs; D3DVALUE m_front; D3DVALUE m_back; Matrix3x3 m_normalMatrix; D3DRMMATRIX4D m_projection; std::vector m_zBuffer; std::vector m_transformedVerts; Plane m_frustumPlanes[6]; }; inline static void Direct3DRMSoftware_EnumDevice(LPD3DENUMDEVICESCALLBACK cb, void* ctx) { D3DDEVICEDESC halDesc = {}; D3DDEVICEDESC helDesc = {}; helDesc.dcmColorModel = D3DCOLOR_RGB; helDesc.dwFlags = D3DDD_DEVICEZBUFFERBITDEPTH; helDesc.dwDeviceZBufferBitDepth = DDBD_32; helDesc.dwDeviceRenderBitDepth = DDBD_32; helDesc.dpcTriCaps.dwTextureCaps = D3DPTEXTURECAPS_PERSPECTIVE; helDesc.dpcTriCaps.dwShadeCaps = D3DPSHADECAPS_ALPHAFLATBLEND; helDesc.dpcTriCaps.dwTextureFilterCaps = D3DPTFILTERCAPS_LINEAR; EnumDevice(cb, ctx, "Miniwin Emulation", &halDesc, &helDesc, SOFTWARE_GUID); }