c - 从内存创建的 SDL 纹理仅以黑白呈现
问题描述
我正在尝试将游戏从 DDraw 移植到 SDL2。
原始程序加载图像并将它们 blits 到一个后缓冲区,然后将其翻转到一个主缓冲区。
我在想我可以在技术上缩短部分过程,只需获取内存中的后缓冲区,然后将其转换为纹理并将其传送到屏幕上。这种作品已经唯一的问题是屏幕是黑白的。
这是一些代码。持有后备缓冲区的变量是 destmemarea
if (SDL_Init(SDL_INIT_EVERYTHING) != 0) {
SDL_Log("Unable to initialize SDL: %s", SDL_GetError());
}
SDL_Window* window = NULL;
SDL_Texture *bitmapTex = NULL;
SDL_Surface *bitmapSurface = NULL;
SDL_Surface *MySurface = NULL;
SDL_DisplayMode DM;
SDL_GetCurrentDisplayMode(0, &DM);
auto Width = DM.w;
auto Height = DM.h;
window = SDL_CreateWindow("SDL Tutorial ", Width = DM.w - SCREEN_WIDTH, 32, SCREEN_WIDTH *4, SCREEN_HEIGHT, SDL_WINDOW_SHOWN);
if (window == NULL)
{
printf("Window could not be created! SDL_Error: %s\n", SDL_GetError());
}
SDL_Renderer * renderer = SDL_CreateRenderer(window, -1, 0);
int w, h;
SDL_GetRendererOutputSize(renderer, &w, &h);
SDL_Surface * image = SDL_CreateRGBSurfaceFrom( destmemarea, 640, 0, 32, 640, 0, 0, 0,0);
SDL_Texture * texture = SDL_CreateTextureFromSurface(renderer, image);
SDL_RenderCopy(renderer, texture, NULL, NULL);
SDL_RenderPresent(renderer);
SDL_Delay(10000);
SDL_DestroyTexture(texture);
SDL_DestroyRenderer(renderer);
SDL_DestroyWindow(window);
}
不确定这是否有帮助,但这就是他看起来的 DDRAW 堡垒所使用的东西......
dd.dwWidth = 768;
dd.lPitch = 768;
dd.dwSize = 108;
dd.dwFlags = DDSD_PIXELFORMAT|DDSD_PITCH|DDSD_WIDTH|DDSD_HEIGHT|DDSD_CAPS;
dd.ddsCaps.dwCaps = DDSCAPS_SYSTEMMEMORY|DDSCAPS_OFFSCREENPLAIN;
dd.dwHeight = 656;
dd.ddpfPixelFormat.dwSize = 32;
解决方案
所以,我不是 100% 确定我理解你想要做什么,但我有一些假设。
您说您正在从 DDraw 移植代码库,所以我假设您提到的后备缓冲区是您正在分配的内部后备缓冲区,并且在您的应用程序的其余部分中正在对它进行渲染。
如果我在这个假设中是正确的,那么您当前的方法就是您需要做的,但需要指定正确的参数来SDL_CreateRGBSurfaceFrom
width
并且height
是...宽度和高度(以像素为单位)depth
是单个像素中的位数。这取决于写入内存缓冲区的其余渲染代码。如果我们假设您正在执行标准 RGBA,其中每个通道是 8 位,那么它将是 32。pitch
是表面中单行的字节width * (depth / 8)
大小 - 应该等于.4 个遮罩、
Rmask
、Gmask
、Bmask
,并Amask
描述每个depth
大小的像素如何分布通道。同样,取决于您如何渲染到内存缓冲区,以及目标平台的字节序。从文档中,2 种可能的标准布局:#if SDL_BYTEORDER == SDL_BIG_ENDIAN
rmask = 0xff000000; gmask = 0x00ff0000; bmask = 0x0000ff00; amask = 0x000000ff;
#else
rmask = 0x000000ff; gmask = 0x0000ff00; bmask = 0x00ff0000; amask = 0xff000000;
#endif
一定不要忘记通过调用来释放你的表面SDL_FreeSurface()
说了这么多......我认为你从错误的角度来处理你的问题。正如我在评论中所说,SDL 为您处理双缓冲。而不是让自定义代码呈现到内存中的缓冲区,然后尝试从该内存创建一个表面并将其呈现到 SDL 后台缓冲区,然后调用 present... 您应该跳过中间人并直接绘制到 SDL 后台缓冲区。
这是通过各种 SDL 渲染函数完成的,其中 RenderCopy 是其中的一员。你的渲染循环基本上应该做三件事:
称呼
SDL_RenderClear()
循环遍历要呈现到屏幕的每个对象,并使用 SDL 渲染函数之一 - 在图像的最常见情况下,即
SDL_RenderCopy
. 这意味着,在您的整个代码库中,加载您的图像,为它们创建SDL_Surface
并SDL_Texture
保留它们,并在每个帧调用SDL_RenderCopy
或SDL_RenderCopyEx
最后,您每帧只调用一次 SDL_RenderPresent。这将交换缓冲区,并将您的图像呈现到屏幕上。
推荐阅读
- office365 - 拦截 Exchange Server 和 Office365 上的邮件流量
- google-apps-script - Google 表格中的静态时间戳
- python - 导出具有多个值的字典并在不使用 pandas 的情况下导入它们以供以后使用
- python - apk中的Kivy presplash过早结束,留下几秒钟的空白屏幕
- python - 三元树路径
- java - 获取两个 SIM 卡的信号强度,并在其中一个信号在 android 中较弱时切换数据
- c# - 无法按照教程进行操作:连接到数据存储 [11 of 18] | 初学者系列:使用 mongoDB 的 Web API
- excel - 如何从网站下载嵌入式 CSV 文件
- apache-nifi - 尼菲|| 我们可以在单个数据库会话中执行多个 sql 查询吗
- json - 如何联合不同层次的对象?