由于本文章是对TinyRenderer的模仿,所以并不打算引入外部库。

那么我们第一步需要解决的就是图形输出的问题,毕竟,如果连渲染的结果都看不到,那还叫什么Renderer嘛。

由于不引入外部库,所以选择输出的图片格式应该越简单越好,各种位图就成为了我们的首选。

这里我们选择了生态较好的bmp位图。

技术上,由于只使用C++,所以各种文件流就成了我们构建图片的唯一工具。

本章目标

输出一张保存了我们渲染结果的bmp位图

需求:

  • 大小可以控制,也就是位图的尺寸可控
  • 控制某个像素点的颜色,精准更改set()
  • 对位图进行上下反转

实现

BMPImage.h

#ifndef BMP_IMAGE_H
#define BMP_IMAGE_H
#include <string>
#include <vector> #pragma pack(push, 1)
struct BMPFileHeader
{
uint16_t bfType; // BMP文件的类型,必须为"B"然后是"M"
uint32_t bfSize; // 文件大小
uint16_t bfReserved1; // 保留字,必须为0
uint16_t bfReserved2; // 从文件头到实际位图数据的偏移字节数
uint32_t bfOffBits; // 信息头的大小
}; struct BMPInfoHeader
{
uint32_t biSize; // info head size
int32_t biWidth; // 图像宽度
int32_t biHeight; // 图像高度
uint16_t biPlanes; // 图像的位面数
uint16_t biBitCount; // 每个像素的位数
uint32_t biCompression; // 压缩类型
uint32_t biSizeImage; // 图像的大小,以字节为单位
int32_t biXPelsPerMeter; // 水平分辨率
int32_t biYPelsPerMeter; // 垂直分辨率
uint32_t biClrUsed; // 位图实际使用的颜色表中的颜色数
uint32_t biClrImportant; // 位图显示过程中重要的颜色数
};
#pragma pack(pop) /**
* \brief custom the color format used
*/
enum ColorFormat
{
RGB,
CMYK
}; struct RGBPixel
{
uint8_t red;
uint8_t green;
uint8_t blue;
RGBPixel() : red(0), green(0), blue(0)
{
}
RGBPixel(uint8_t red, uint8_t green, uint8_t blue) : red(red), green(green), blue(blue)
{
}
}; class BMPImage
{
public:
BMPImage() = delete;
BMPImage(unsigned int width, unsigned int height, ColorFormat colorFormat = ColorFormat::RGB);
void loadData(std::vector<char>&& userData);
void generate(const std::string& fileName);
void loadDataAndGenerate(std::vector<char>&& userData, const std::string& fileName); void set(int x, int y, RGBPixel pixel);
void flipVertically(); private:
BMPFileHeader fileHeader;
BMPInfoHeader infoHeader; ColorFormat colorFormat;
std::vector<unsigned char> pixelData;
}; #endif

Important:

  • 在组织bmp文件头的部分,一定要使用预处理宏#pragma pack(push, 1)#pragma pack(pop),控制内存对齐方式为单字节,否则会由于编译器控制的内存对齐而导致文件格式错误,从而不能正确输出

BMPImage.cpp

#include "TinyRenderer/BMPImage.h"

#include <fstream>
#include <iostream> BMPImage::BMPImage(unsigned width, unsigned height, ColorFormat colorFormat)
{
int rowSize = (width * 3 + 3) & (~3); // Ensure row size is a multiple of 4 bytes
int fileSize = rowSize * height + sizeof(BMPFileHeader) + sizeof(BMPInfoHeader); // Set BMP file header
fileHeader.bfType = 0x4D42; // 'BM'
fileHeader.bfSize = fileSize;
fileHeader.bfReserved1 = 0;
fileHeader.bfReserved2 = 0;
fileHeader.bfOffBits = sizeof(BMPFileHeader) + sizeof(BMPInfoHeader); // Set BMP info header
infoHeader.biSize = sizeof(BMPInfoHeader);
infoHeader.biWidth = width;
infoHeader.biHeight = height;
infoHeader.biPlanes = 1;
infoHeader.biBitCount = 24;
infoHeader.biCompression = 0;
infoHeader.biSizeImage = rowSize * height;
infoHeader.biXPelsPerMeter = 0;
infoHeader.biYPelsPerMeter = 0;
infoHeader.biClrUsed = 0;
infoHeader.biClrImportant = 0; // Initialize pixel data
pixelData.resize(rowSize * height, 0);
} // not important now
void BMPImage::loadData(std::vector<char>&& userData)
{
// TODO: load image
} void BMPImage::generate(const std::string& fileName)
{
std::ofstream file(fileName, std::ios::out | std::ios::binary);
if (!file)
{
std::cerr << "Error: Unable to open file for writing." << std::endl;
return;
} // Write headers
file.write(reinterpret_cast<const char*>(&fileHeader), sizeof(fileHeader));
file.write(reinterpret_cast<const char*>(&infoHeader), sizeof(infoHeader)); // Write pixel data
file.write(reinterpret_cast<const char*>(pixelData.data()), pixelData.size()); file.close();
} void BMPImage::loadDataAndGenerate(std::vector<char>&& userData, const std::string& fileName)
{
} void BMPImage::set(int x, int y, RGBPixel pixel)
{
if (x < 0 || y < 0 || x >= infoHeader.biWidth || y >= infoHeader.biHeight)
{
throw std::out_of_range("Pixel coordinates are out of bounds");
}
int rowSize = (infoHeader.biWidth * 3 + 3) & (~3);
int index = (infoHeader.biHeight - 1 - y) * rowSize + x * 3;
pixelData[index] = pixel.blue;
pixelData[index + 1] = pixel.green;
pixelData[index + 2] = pixel.red;
} void BMPImage::flipVertically()
{
int width = infoHeader.biWidth;
int height = infoHeader.biHeight;
int rowSize = (width * 3 + 3) & (~3); for (int y = 0; y < height / 2; ++y)
{
int topIndex = y * rowSize;
int bottomIndex = (height - 1 - y) * rowSize;
for (int x = 0; x < rowSize; ++x)
{
std::swap(pixelData[topIndex + x], pixelData[bottomIndex + x]);
}
}
}

测试

main.cpp

#include "TinyRenderer/TinyRenderer.h"
#include "TinyRenderer/BMPImage.h" int main()
{
BMPImage image(100, 100, ColorFormat::RGB);
RGBPixel white(255, 255, 255);
image.set(22, 77, white);
image.flipVertically();
image.generate("test.bmp");
std::cout << "Image Generated." << std::endl;
return 0;
}

请忽略TinyRenderer/TinyRenderer.h,里面仅是一些头文件。

输出结果

你能看到那个白点吗?那是我们的起点。

Chapter1 p1 Output Image的更多相关文章

  1. sql分页操作

    看到了网上关于分页的讲解  对最快的分页语句做了测试 还别说速度真快 总共6w条数据 速度确实so 快 前提是id是主键 或者是索引 declare @page int;--页数 declare @P ...

  2. C#解析json文件的方法

    C# 解析 json JSON(全称为JavaScript Object Notation) 是一种轻量级的数据交换格式.它是基于JavaScript语法标准的一个子集. JSON采用完全独立于语言的 ...

  3. 设计一个用于人事管理的People(人员)类

    #include <iostream> #include <string> using namespace std; class Date //日期类 { private: i ...

  4. JSON学习

    1.JSON 语法是 JavaScript 对象表示语法的子集. l  数据在名称/值对中 l  数据由逗号分隔 l  花括号保存对象 l  方括号保存数组 JSON 值可以是: l  数字(整数或浮 ...

  5. js的this和面向对象编程

    很奇怪的是很多书或资料没有把这个事情讲清楚. 关键就是在于没有一个整体的思维技术模式,问题被隔离了所以反而不容易理解. 我们先看this,这是js的关键字,指示函数的上下文对象. 这里问题就来了,比如 ...

  6. oracle,mysql,SqlServer三种数据库的分页查询的实例。

    MySql: MySQL数据库实现分页比较简单,提供了 LIMIT函数.一般只需要直接写到sql语句后面就行了.LIMIT子 句可以用来限制由SELECT语句返回过来的数据数量,它有一个或两个参数,如 ...

  7. C# 解析JSON格式数据

    JSON简介 JSON(全称为JavaScript ObjectNotation) 是一种轻量级的数据交换格式.它是基于JavaScript语法标准的一个子集.JSON采用完全独立于语言的文本格式,可 ...

  8. C# 解析 Json数据

    JSON(全称为JavaScript Object Notation) 是一种轻量级的数据交换格式.它是基于JavaScript语法标准的一个子集. JSON采用完全独立于语言的文本格式,可以很容易在 ...

  9. C# 解析 Json

    C# 解析 json JSON(全称为JavaScript Object Notation) 是一种轻量级的数据交换格式.它是基于JavaScript语法标准的一个子集. JSON采用完全独立于语言的 ...

  10. sql server 2000数据库 最近经常出现某进程一直占用资源,阻塞?死锁?

    OA的数据库最近多次出现某进程一直占用资源,导致其他进程无法执行.使用sp_who2 和 sql server profiler跟踪查询,发现有以下几个语句常常占用资源: 1.declare @P1 ...

随机推荐

  1. 力扣745(java&python)-达到终点数字(中等)

    题目: 在一根无限长的数轴上,你站在0的位置.终点在target的位置. 你可以做一些数量的移动 numMoves : 每次你可以选择向左或向右移动.第 i 次移动(从  i == 1 开始,到 i ...

  2. Flagger on ASM·基于Mixerless Telemetry实现渐进式灰度发布系列 1 遥测数据

    简介: 服务网格ASM的Mixerless Telemetry技术,为业务容器提供了无侵入式的遥测数据.遥测数据一方面作为监控指标被ARMPS/prometheus采集,用于服务网格可观测性:另一方面 ...

  3. 谈谈C++新标准带来的属性(Attribute)

    简介: 从C++11开始,标准引入了一个新概念"属性(attribute)",本文将简单介绍一下目前在C++标准中已经添加的各个属性以及常用属性的具体应用. 作者 | 寒冬来源 | ...

  4. 2019-11-29-C#-如何写-DEBUG-输出

    title author date CreateTime categories C# 如何写 DEBUG 输出 lindexi 2019-11-29 08:28:35 +0800 2018-2-13 ...

  5. Part-DB 配置流程

    介绍 Part-DB是一个开源的器件管理工具,博主用于管理个人的电子器材,最近捣鼓了一下这个工具,由于手头还有一块闲置的赛昉·星光2的开发板,所以我打算一起拿来捣鼓一下,如果不成功,就用树莓派(生气) ...

  6. 举例useContext性能低下的样例,同时推荐用什么方法改进

    在React中,useContext 是一种非常方便的全局状态管理工具,它可以让我们在组件之间共享状态,而不需要通过层层传递 props.然而,当我们在一个大型的 React 应用中过度使用 useC ...

  7. 为什么需要学习ITSM/ITIL

    假如你需要管理一个超过20人的IT服务组织,一般会面临以下问题: 人多事杂活重,每个人都很累,工作却还是一团糟糕, 用户方怨声载道,领导也颇有微词,同事间也经常互相甩锅埋坑, 工作只是救火或者混日子, ...

  8. SpringBoot连接redis报错:exception is io.lettuce.core.RedisException: java.io.IOException: 远程主机强迫关闭了一个现有的连接

    一.解决思路 (1).检查redis的配置是否正确 spring redis: host: localhost port: 6379 password: 123456 database: 0 time ...

  9. 用 C 语言开发一门编程语言 — 条件分支

    目录 文章目录 目录 前文列表 条件分支 排序函数 等于函数 if 函数 递归函数 源代码 前文列表 <用 C 语言开发一门编程语言 - 交互式解析器> <用 C 语言开发一门编程语 ...

  10. PageOffice6最简集成代码(Asp.Net)

    本文描述了PageOffice产品在普通的Asp.Net项目中如何集成调用. 新建Asp.Net项目:PageOffice6-Net-Simple 在您的web项目的"依赖项-包-管理NuG ...