【c++|SDL】二、读取图片、显示图片、动画制作

every blog every motto: You can do more than you think.
https://blog.csdn.net/weixin_39190382?type=blog

0. 前言

读取图片,显示图片,动画


SDL中有两种在屏幕上显示的方法

  • SDL_Surface: 使用软件渲染处理
  • SDL_Texture: 使用硬件加速渲染处理

1. 在界面上显示图片

修改Game.h

#ifndef __Game__
#define __Game__#include <SDL2/SDL.h>
class Game
{public:Game() {};~Game() {};// simply set the running variable to truebool init(const char* title, int xpos, int ypos,int width, int height, bool fullscreen);void render();void update();void handleEvents();void clean();// a function to access the private running variablebool running() { return m_bRunning; }
private:bool m_bRunning;// SDL_Window* m_pWindow;// SDL_Renderer* m_pRenderer;SDL_Window* m_pWindow;SDL_Renderer* m_pRenderer;SDL_Texture* m_pTexture; // the new SDL_Texture variableSDL_Rect m_sourceRectangle; // the first rectangleSDL_Rect m_destinationRectangle; // another rectangle
};#endif /* defined(__Game__) */

修改Game.cpp

修改其中的init和render函数

#include "Game.h"
#include <iostream>using namespace std;bool Game::init(const char* title, int xpos, int ypos, int width, int height, bool fullscreen)
{   int flags = 0;if (fullscreen){flags = SDL_WINDOW_FULLSCREEN;}// attempt to initialize SDLif(SDL_Init(SDL_INIT_EVERYTHING) == 0){std::cout << "SDL init success\n";// init the windowm_pWindow = SDL_CreateWindow(title, xpos, ypos, width, height, flags);if(m_pWindow != 0) // window init success{std::cout << "window creation success\n";m_pRenderer = SDL_CreateRenderer(m_pWindow, -1, 0);if(m_pRenderer != 0) // renderer init success{std::cout << "renderer creation success\n";SDL_SetRenderDrawColor(m_pRenderer,255,255,255,255);}else{std::cout << "renderer init fail\n";return false; // renderer init fail}}else{std::cout << "window init fail\n";return false; // window init fail}}else{std::cout << "SDL init fail\n";return false; // SDL init fail}std::cout << "init success\n";SDL_Surface* pTempSurface = SDL_LoadBMP("assets/rider.bmp");m_pTexture = SDL_CreateTextureFromSurface(m_pRenderer, pTempSurface);SDL_FreeSurface(pTempSurface);SDL_QueryTexture(m_pTexture, NULL, NULL,&m_sourceRectangle.w,&m_sourceRectangle.h); // 获取长宽m_destinationRectangle.x = m_sourceRectangle.x = 0;m_destinationRectangle.y = m_sourceRectangle.y = 0;m_destinationRectangle.w = m_sourceRectangle.w;m_destinationRectangle.h = m_sourceRectangle.h;m_bRunning = true; // everything inited successfully, start the main loopreturn true;
}void Game::render()
{SDL_RenderClear(m_pRenderer); // clear the renderer to the draw colorSDL_RenderCopy(m_pRenderer, m_pTexture,&m_sourceRectangle,&m_destinationRectangle);SDL_RenderPresent(m_pRenderer); // draw to the screen
}void Game::handleEvents()
{SDL_Event event;if(SDL_PollEvent(&event)){switch (event.type){case SDL_QUIT:m_bRunning = false;break;default:break;}}
}void Game::clean()
{std::cout << "cleaning game\n";SDL_DestroyWindow(m_pWindow);SDL_DestroyRenderer(m_pRenderer);SDL_Quit();
}void Game::update()
{}

在这里插入图片描述

2. 显示位置和范围

修改m_sourceRectangle和m_destinationRectangle
具体在Game.cpp SDL_QueryTexture代码下面

Game.cpp init函数代码如下,其他代码相同

bool Game::init(const char* title, int xpos, int ypos, int width, int height, bool fullscreen)
{   int flags = 0;if (fullscreen){flags = SDL_WINDOW_FULLSCREEN;}// attempt to initialize SDLif(SDL_Init(SDL_INIT_EVERYTHING) == 0){std::cout << "SDL init success\n";// init the windowm_pWindow = SDL_CreateWindow(title, xpos, ypos, width, height, flags);if(m_pWindow != 0) // window init success{std::cout << "window creation success\n";m_pRenderer = SDL_CreateRenderer(m_pWindow, -1, 0);if(m_pRenderer != 0) // renderer init success{std::cout << "renderer creation success\n";SDL_SetRenderDrawColor(m_pRenderer,255,255,255,255);}else{std::cout << "renderer init fail\n";return false; // renderer init fail}}else{std::cout << "window init fail\n";return false; // window init fail}}else{std::cout << "SDL init fail\n";return false; // SDL init fail}std::cout << "init success\n";SDL_Surface* pTempSurface = SDL_LoadBMP("assets/rider.bmp");m_pTexture = SDL_CreateTextureFromSurface(m_pRenderer, pTempSurface);SDL_FreeSurface(pTempSurface);SDL_QueryTexture(m_pTexture, NULL, NULL,&m_sourceRectangle.w,&m_sourceRectangle.h); // 获取长宽m_sourceRectangle.w = 50;m_sourceRectangle.h = 50;m_destinationRectangle.x = 100;m_destinationRectangle.y = 100;m_sourceRectangle.x = 50;m_sourceRectangle.y = 50;// m_destinationRectangle.x = m_sourceRectangle.x = 0;// m_destinationRectangle.y = m_sourceRectangle.y = 0;m_destinationRectangle.w = m_sourceRectangle.w;m_destinationRectangle.h = m_sourceRectangle.h;m_bRunning = true; // everything inited successfully, start the main loopreturn true;
}

在这里插入图片描述

3. 动图

有多帧图片,每100ms显示其中一张,可以让图片动起来
在这里插入图片描述

#include "Game.h"
#include <iostream>using namespace std;bool Game::init(const char* title, int xpos, int ypos, int width, int height, bool fullscreen)
{   int flags = 0;if (fullscreen){flags = SDL_WINDOW_FULLSCREEN;}// attempt to initialize SDLif(SDL_Init(SDL_INIT_EVERYTHING) == 0){std::cout << "SDL init success\n";// init the windowm_pWindow = SDL_CreateWindow(title, xpos, ypos, width, height, flags);if(m_pWindow != 0) // window init success{std::cout << "window creation success\n";m_pRenderer = SDL_CreateRenderer(m_pWindow, -1, 0);if(m_pRenderer != 0) // renderer init success{std::cout << "renderer creation success\n";SDL_SetRenderDrawColor(m_pRenderer,255,255,255,255);}else{std::cout << "renderer init fail\n";return false; // renderer init fail}}else{std::cout << "window init fail\n";return false; // window init fail}}else{std::cout << "SDL init fail\n";return false; // SDL init fail}std::cout << "init success\n";// SDL_Surface* pTempSurface = SDL_LoadBMP("assets/rider.bmp");SDL_Surface* pTempSurface = SDL_LoadBMP("assets/animate.bmp");m_pTexture = SDL_CreateTextureFromSurface(m_pRenderer, pTempSurface);SDL_FreeSurface(pTempSurface);SDL_QueryTexture(m_pTexture, NULL, NULL,&m_sourceRectangle.w,&m_sourceRectangle.h); // 获取长宽m_sourceRectangle.x = 0;m_sourceRectangle.y = 0;m_sourceRectangle.w = 128;m_sourceRectangle.h = 82;m_destinationRectangle.x = 0;m_destinationRectangle.y = 0;// m_destinationRectangle.x = m_sourceRectangle.x = 0;// m_destinationRectangle.y = m_sourceRectangle.y = 0;m_destinationRectangle.w = m_sourceRectangle.w;m_destinationRectangle.h = m_sourceRectangle.h;m_bRunning = true; // everything inited successfully, start the main loopreturn true;
}void Game::render()
{SDL_RenderClear(m_pRenderer); // clear the renderer to the draw color// SDL_RenderCopy(m_pRenderer, m_pTexture,0,0);SDL_RenderCopy(m_pRenderer, m_pTexture,&m_sourceRectangle,&m_destinationRectangle);SDL_RenderPresent(m_pRenderer); // draw to the screen
}void Game::handleEvents()
{SDL_Event event;if(SDL_PollEvent(&event)){switch (event.type){case SDL_QUIT:m_bRunning = false;break;default:break;}}
}void Game::clean()
{std::cout << "cleaning game\n";SDL_DestroyWindow(m_pWindow);SDL_DestroyRenderer(m_pRenderer);SDL_Quit();
}void Game::update()
{m_sourceRectangle.x = 128 * int(((SDL_GetTicks() / 100) % 6));}

在这里插入图片描述

可以翻转图片

void Game::render()
{SDL_RenderClear(m_pRenderer); // clear the renderer to the draw color// SDL_RenderCopy(m_pRenderer, m_pTexture,0,0);// SDL_RenderCopy(m_pRenderer, m_pTexture,&m_sourceRectangle,&m_destinationRectangle);SDL_RenderCopyEx(m_pRenderer, m_pTexture,&m_sourceRectangle, &m_destinationRectangle,0, 0, SDL_FLIP_HORIZONTAL); // pass in the horizontal flipSDL_RenderPresent(m_pRenderer); // draw to the screen
}

3. SDL_image

前面用的是SDL_LoadBMP只能加载bmp图片,SDL_image可以加载很多格式的图片,包括png,jpg,bmp,tiff,pcx,tif,lbm,xpm,gif,tk和xv等。

Game.h中引入SDL_image

#include <SDL2/SDL_image.h>

修改加载图片代码Game.cpp

SDL_Surface* pTempSurface = IMG_Load("assets/animate.png");

编译修改为:

g++ m14main.cpp Game.cpp  -o m14 -lSDL2 -lSDL2_image

Game.cpp代码如下:

#include "Game.h"
#include <iostream>using namespace std;bool Game::init(const char* title, int xpos, int ypos, int width, int height, bool fullscreen)
{   int flags = 0;if (fullscreen){flags = SDL_WINDOW_FULLSCREEN;}// attempt to initialize SDLif(SDL_Init(SDL_INIT_EVERYTHING) == 0){std::cout << "SDL init success\n";// init the windowm_pWindow = SDL_CreateWindow(title, xpos, ypos, width, height, flags);if(m_pWindow != 0) // window init success{std::cout << "window creation success\n";m_pRenderer = SDL_CreateRenderer(m_pWindow, -1, 0);if(m_pRenderer != 0) // renderer init success{std::cout << "renderer creation success\n";SDL_SetRenderDrawColor(m_pRenderer,255,0,0,255);}else{std::cout << "renderer init fail\n";return false; // renderer init fail}}else{std::cout << "window init fail\n";return false; // window init fail}}else{std::cout << "SDL init fail\n";return false; // SDL init fail}std::cout << "init success\n";// SDL_Surface* pTempSurface = SDL_LoadBMP("assets/rider.bmp");// SDL_Surface* pTempSurface = SDL_LoadBMP("assets/animate.bmp");// SDL_Surface* pTempSurface = IMG_Load("assets/animate.png");SDL_Surface* pTempSurface = IMG_Load("assets/animate-alpha.png");m_pTexture = SDL_CreateTextureFromSurface(m_pRenderer, pTempSurface);SDL_FreeSurface(pTempSurface);SDL_QueryTexture(m_pTexture, NULL, NULL,&m_sourceRectangle.w,&m_sourceRectangle.h); // 获取长宽m_sourceRectangle.x = 0;m_sourceRectangle.y = 0;m_sourceRectangle.w = 128;m_sourceRectangle.h = 82;m_destinationRectangle.x = 0;m_destinationRectangle.y = 0;// m_destinationRectangle.x = m_sourceRectangle.x = 0;// m_destinationRectangle.y = m_sourceRectangle.y = 0;m_destinationRectangle.w = m_sourceRectangle.w;m_destinationRectangle.h = m_sourceRectangle.h;m_bRunning = true; // everything inited successfully, start the main loopreturn true;
}void Game::render()
{SDL_RenderClear(m_pRenderer); // clear the renderer to the draw color// SDL_RenderCopy(m_pRenderer, m_pTexture,0,0);// SDL_RenderCopy(m_pRenderer, m_pTexture,&m_sourceRectangle,&m_destinationRectangle);SDL_RenderCopyEx(m_pRenderer, m_pTexture,&m_sourceRectangle, &m_destinationRectangle,0, 0, SDL_FLIP_HORIZONTAL); // pass in the horizontal flipSDL_RenderPresent(m_pRenderer); // draw to the screen
}void Game::handleEvents()
{SDL_Event event;if(SDL_PollEvent(&event)){switch (event.type){case SDL_QUIT:m_bRunning = false;break;default:break;}}
}void Game::clean()
{std::cout << "cleaning game\n";SDL_DestroyWindow(m_pWindow);SDL_DestroyRenderer(m_pRenderer);SDL_Quit();
}void Game::update()
{m_sourceRectangle.x = 128 * int(((SDL_GetTicks() / 100) % 6));}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/196598.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

iOS代码安全加固利器:深入探讨字符串和代码混淆器的作用

​ 在网上搜“代码混淆”关键词&#xff0c;可以看到n多教程。包括本篇博客&#xff0c;大部分重要内容也是从网上各位大神的博客里面看到然后摘取和总结出来的。虽然网上都有&#xff0c;但是对于我个人来说&#xff0c;很难找到一篇博客概括完全的&#xff0c;所以还是总结一…

现在的00后,实在是太卷了......

现在的小年轻真的卷得过分了。前段时间我们公司来了个00年的&#xff0c;工作没两年&#xff0c;跳槽到我们公司起薪18K&#xff0c;都快接近我了。后来才知道人家是个卷王&#xff0c;从早干到晚就差搬张床到工位睡觉了。 最近和他聊了一次天&#xff0c;原来这位小老弟家里条…

Java中各种数据类型之间的转换

低类型向高类型自动进行转换&#xff0c;高类型向低类型的准换会丢失数据&#xff0c;整数到字符类型的转换将获取对应编码的字符。 进行高精度向低精度的强制类型准换时&#xff0c;需要将想要转换成的数据类型加一个括号()。 如何完成自动转换呢&#xff1f; 转换前的数据类…

KNN算法实战-健康医疗

健康医疗 算法建模 knn 算法建模构建微观数据和疾病之间的关系knn 调整超参数&#xff0c;准确率提升数据归一化、标准化&#xff0c;提升更加明显 算法实战 导入包 import numpy as np import pandas as pd from sklearn.neighbors import KNeighborsClassifier from sklea…

LeetCode 每日一题 Day 3||深度优先搜索(DFS)

1038. 从二叉搜索树到更大和树 给定一个二叉搜索树 root (BST)&#xff0c;请将它的每个节点的值替换成树中大于或者等于该节点值的所有节点值之和。 提醒一下&#xff0c; 二叉搜索树 满足下列约束条件&#xff1a; 节点的左子树仅包含键 小于 节点键的节点。节点的右子树仅…

【23-24 秋学期】NNDL 作业12 优化算法2D可视化

简要介绍图中的优化算法&#xff0c;编程实现并2D可视化 1. 被优化函数 2. 被优化函数 3. 解释不同轨迹的形成原因 分析各个算法的优缺点 REF&#xff1a;图灵社区-图书 (ituring.com.cn) 深度学习入门&#xff1a;基于Python的理论与实现 NNDL 作业11&#xff1a;优化算…

Redis系列之incr和decr命令是线程安全的?

Redis是一个单线程的服务&#xff0c;所以正常来说redis的命令是会排队执行的。incr/decr命令是redis提供的可以实现递增递减的命令&#xff0c;所以这两个命令也是具有原子性的&#xff1f;是线程安全的&#xff1f;这个也是互联网公司面试的常见题&#xff0c;话不多说&#…

Leetcode 108 将有序数组转换为二叉搜索树

题意理解&#xff1a; 我们需要根据一个数组来构建一个二叉搜索树&#xff0c;且该二叉搜索树也是高度平衡二叉树。 什么是高度平衡二叉树呢? 即对于每个节点来说&#xff0c;左右子树高度差不超过1 思路&#xff1a;我们总是从数组的中间位置作为根节点构建该树&#xff0c;这…

AcW730.机器人跳跃问题(二分法)-Java版

import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader;//由题目可知,无论能量大与小,都满足 e 2 * e - h[i]; //初始能量越大,最终的结果越大,要找到一个满足条件的最小值 //可以根据二分的向左找模板: /*if(check(mid)) r mid;els…

茄子科技张韶全:跨多云大数据平台DataCake在OceanBase的实践

11 月 16 日&#xff0c;OceanBase 在北京顺利举办 2023 年度发布会&#xff0c;正式宣布&#xff1a;将持续践行“一体化”产品战略&#xff0c;为关键业务负载打造一体化数据库。其中&#xff0c;在“数字化转型升级实践专场”&#xff0c;我们有幸邀请到了茄子科技大数据技术…

一个Blazor+WinForm+MAUI+PDA实现的条码比对系统

条码比对系统是由单机版桌面软件和Android版的PDA扫码软件组成&#xff0c;桌面软件采用Blazor与WinForm进行混合开发&#xff0c;PDA扫码软件采用MAUI进行开发&#xff0c;这个项目都是基于.NET技术进行构建&#xff0c;这也是将近期学习Blazor和MAUI这两门技术应用到实践当中…

刷题系列——排序算法

参考&#xff1a;README - 十大经典排序算法 1&#xff09;排序算法分为内部外部排序两种&#xff0c;这个之前并不了解&#xff0c;外部排序需要访问外存的这个就是指需要额外内存比如另一个list或者dict存储中间结果。 2&#xff09;稳定性&#xff1a;排序后 2 个相等键值…

openGauss训练营培训课程第1课时

课时1:openGauss全景介绍 1、介绍 openGauss 全景 1.1.openGauss总体架构介绍 本章节主要介绍了openGauss发展的历史&#xff0c;现状以及未来。对当前的DataPod和DataKit 2种openGauss当前主推的场景化产品进行了介绍。同时对openGauss的整个逻辑模块的视图进行了讲解。 …

算法通关村第十三关-黄金挑战数论问题

计数质数 描述 : 给定整数 n &#xff0c;返回 所有小于非负整数 n 的质数的数量 。 题目 : LeetCode 204.计数质数 : 204. 计数质数 分析 : 解决这个题有一个有效的方法&#xff0c;叫埃氏筛 , 后来又产生了线性筛&#xff0c;奇数筛等改进的方法。 基本思想是如果 x是…

12.04 二叉树中等题

513. 找树左下角的值 给定一个二叉树的 根节点 root&#xff0c;请找出该二叉树的 最底层 最左边 节点的值。 假设二叉树中至少有一个节点。 示例 1: 输入: root [2,1,3] 输出: 1 思路&#xff1a;找到最低层中最左侧的节点值&#xff0c;比较适合层序遍历&#xff0c;返回最…

【动态规划】LeetCode-198/LCR089.打家劫舍

&#x1f388;算法那些事专栏说明&#xff1a;这是一个记录刷题日常的专栏&#xff0c;每个文章标题前都会写明这道题使用的算法。专栏每日计划至少更新1道题目&#xff0c;在这立下Flag&#x1f6a9; &#x1f3e0;个人主页&#xff1a;Jammingpro &#x1f4d5;专栏链接&…

MS85163实时时钟/日历可Pin to Pin兼容PCF8563

MS85163/MS85163M是一款CMOS实时时钟(RTC) 和日历电路&#xff0c;针对低功耗进行了优化&#xff0c;内置了可编程的时钟输出、中断输出和低电压检测器。可Pin to Pin兼容PCF8563。所有寄存器地址和数据都通过两线双向I 2C总线进行串行传输&#xff0c;最大总线传输速度为 400k…

2023年【上海市安全员C3证】新版试题及上海市安全员C3证试题及解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 上海市安全员C3证新版试题是安全生产模拟考试一点通总题库中生成的一套上海市安全员C3证试题及解析&#xff0c;安全生产模拟考试一点通上上海市安全员C3证作业手机同步练习。2023年【上海市安全员C3证】新版试题及上…

干货分享:盘点8款优秀的自动化测试工具

如今&#xff0c;作为一名软件测试工程师&#xff0c;几乎所有人都需要具备自动化测试相关的知识&#xff0c;并且懂得如何去利用工具&#xff0c;来为企业减少时间成本和错误成本。这是为什么呢&#xff1f; 在以前&#xff0c;测试人员一般都只需要扮演终端用户&#xff0c;…

用Python创建日历详细指南与实用示例

更多Python学习内容&#xff1a;ipengtao.com 大家好&#xff0c;我是彭涛&#xff0c;今天为大家分享 用Python创建日历详细指南与实用示例&#xff0c;全文4800字&#xff0c;阅读大约15分钟。 在日常生活和工作中&#xff0c;创建和管理日历是一项关键任务。Python提供了丰富…