IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> 5.游戏引擎 纹理管理器实现 纹理数据绑定OpenGL 滤波方式选择线性滤波 -> 正文阅读

[游戏开发]5.游戏引擎 纹理管理器实现 纹理数据绑定OpenGL 滤波方式选择线性滤波

?????????这一章主要是讲如何把一张图片加载进内存中转换BGR->RGB格式,然后和OpenGL做绑定渲染出来。

? ? ? ? 这一章讲解的顺序完全反过来了。基础好的读者也可以反过来看,先看重点,然后再回过头来看源码。没接触过的读者可以先简单的读一遍源码,然后再看重点分析,这样看重点分析的话会更有感觉。源码的话文章写的差不多了,我再整理一下会放到群里。

先贴出完整代码

?TextureManager.h

#pragma once
#include <list>

struct  textureHeight
{
public:
	GLfloat m_iLowHeight=0;//最低高度
	GLfloat	m_iHighHeight=0;//最高高度
	GLfloat m_iOptimalHeight=0;//最佳高度
};

class CCTexture
{
	friend class CTextureManager;
private:
	char name[255]; //纹理路径
	GLuint type;	//类型,保存了一个16进制数
	GLuint texID;	//纹理ID
	UINT  width;     //图片的宽
	UINT height;	//图片的高
	UINT uReference; //引用计数
	int bytesPerPixel; //每个像素的位(Bit)
	GLubyte *imageData;//纹理数据(unsigned char)
	Vector3D centerPos ={0,0,1};

public:
	textureHeight *Tex=new textureHeight;
	CCTexture();
	CCTexture(const char *fileName,float width=1,float height=1);
	CCTexture(const char * name, int wid, int hei, int BitCount);
	int getbyte() { return bytesPerPixel; }
	GLuint getType() { return type; }
	char *getName() { return name; }
	GLuint getTextureID() { return texID; }
	UINT getWidth() { return width; }
	UINT getHeight() { return height; }
	BOOL saveBmp(const char * fileName);
	GLubyte *getImageData() { return imageData; }
	void setImageData(GLubyte *data) { this->imageData = data; }
	void setCenterPos(Vector3D centerPoint);
	void draw();

	~CCTexture() ;

public:
	
	
};

class CTextureManager
{
private:
	std::list<CCTexture*> textureList;
	static CTextureManager*instance;

public:
	static CTextureManager*getInstance();
	CCTexture* addTexture(char *fileName,float width=1,float height=1);
	CCTexture *addTexture(const char*fileName);
	CCTexture * addTexture(char *name, int width, int height, int pixelFormat);
	void resetRef();//把引用计数全都归零
	void releaseRef();//把引用计数全都归零
	void CreateGLTextures(int wid, int hei, int biBitCount, OUT GLuint &texture, OUT GLubyte*&textureData);

	CTextureManager();
	~CTextureManager();
};

TextureManager.cpp

#include "Engine.h"

CTextureManager*CTextureManager::instance = NULL;


CCTexture::CCTexture()
{
	texID = 0;
	uReference = 1;
	imageData = NULL;
}


CCTexture::CCTexture(const char* fileName,float wid,float hei)
{
	texID = 0;
	CImage img;
	HRESULT hr = img.Load(CUser::getInstance()->CharToWchar(fileName));
	if (FAILED(hr))
	{
		char temp[256];
		sprintf_s(temp, 256, "%s打开失败", fileName);
		MessageBox(NULL, CUser::getInstance()->CharToWchar(temp), L"错误", 0);
		return;
	}

	strcpy(name, fileName);
	HBITMAP hbmp = img;
	BITMAP bm;
	GetObject(hbmp, sizeof(bm), &bm);

	if (wid > 1 && hei > 1)
	{
		width = wid;	height = hei;
	}
	else 
	{
		width = bm.bmWidth;	height = bm.bmHeight;
	}
	
	bytesPerPixel = bm.bmBitsPixel / 8;

	if (bytesPerPixel == 3)
	{
		type = GL_RGB;
	}
	else if (bytesPerPixel == 4)
	{
		type = GL_RGBA;
	}
	else if (bytesPerPixel == 1)
	{
		type = GL_LUMINANCE;
	}
		
	imageData = new GLubyte[width*height*bytesPerPixel];
	memcpy(imageData, bm.bmBits, width*height*bytesPerPixel);
	// Convert From BGR To RGB Format And Add An Alpha Value Of 255
	if (bytesPerPixel != 1)
	{
		for (long i = 0; i < width * height; i++)
		{
			GLubyte temp = imageData[0 + i*bytesPerPixel];
			imageData[0 + i*bytesPerPixel] = imageData[2 + i*bytesPerPixel];
			imageData[2 + i*bytesPerPixel] = temp;
		}
	}
	else //等于8位时处理的
	{
		for (int i = 0; i < height; i++)
		{
			memcpy(imageData + i*width, img.GetPixelAddress(0, i), width);
		}
	}
	glGenTextures(1, (GLuint*)&texID);
	glBindTexture(GL_TEXTURE_2D, texID);
	glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
	glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR);//纹理过滤模式
	gluBuild2DMipmaps(GL_TEXTURE_2D, type, width, height, type, GL_UNSIGNED_BYTE, imageData);
	
	
	/*glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
		glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_S,GL_REPEAT);
		glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_T, GL_REPEAT);
		glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
		glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_NEAREST);
		glTexImage2D(GL_TEXTURE_2D, 0, type, width, height, 0, type, GL_UNSIGNED_BYTE, imageData);*/
}
	

CCTexture::CCTexture(const char *name, int wid, int hei, int BitCount)
{
	uReference =1;
	strcpy(this->name,name);
	bytesPerPixel = BitCount / 8;
	this->width = wid;
	this->height = hei;
	glGenTextures(1, &texID);
	glBindTexture(GL_TEXTURE_2D, texID);
	glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
	glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
	imageData = new GLubyte[width*height*bytesPerPixel];
	//memset(imageData, 0, width*height*bytesPerPixel);


	if (bytesPerPixel == 1)
	{
		for (int i = 0; i < width*height; ++i)
		{
			imageData[i] = 0;
		}
		type = GL_LUMINANCE;
	}
	else if (bytesPerPixel == 3)
	{
		for (int i = 0; i < width*height; ++i)
		{
			imageData[i * 3 + 0] = 0;
			imageData[i * 3 + 1] = 0;
			imageData[i * 3 + 2] = 0;
		}
		type = GL_RGB;
	}
	else if (bytesPerPixel == 4)
	{
		for (int i = 0; i < width*height; ++i)
		{
			imageData[i * 4 + 0] = 0;
			imageData[i * 4 + 1] = 0;
			imageData[i * 4 + 2] = 0;
			imageData[i * 4 + 3] = 0;
		}
		type = GL_RGBA;
	}

	glTexImage2D(GL_TEXTURE_2D, 0, type, wid, hei, 0, type, GL_UNSIGNED_BYTE, imageData);

}

BOOL CCTexture::saveBmp(const char *fileName)
{
	//如果位图数据指针为0,则没有数据传入,函数返回
	if (!imageData)
		return 0;

	//灰度图
	static RGBQUAD *pColorTable = NULL;
	if (pColorTable == NULL)
	{
		pColorTable = new RGBQUAD[256];
		for (int i = 0; i < 256; i++)
		{
			pColorTable[i].rgbBlue = i;
			pColorTable[i].rgbGreen = i;
			pColorTable[i].rgbRed = i;
			pColorTable[i].rgbReserved = 0;
		}
	}

	//颜色表大小,以字节为单位,灰度图像颜色表为1024字节,彩色图像颜色表大小为0
	int colorTablesize = 0;
	if (bytesPerPixel == 1)
		colorTablesize = 1024;
	
	//待存储图像数据每行字节数为4的倍数
	int lineByte = (width * bytesPerPixel + 3) / 4 * 4;

	//以二进制写的方式打开文件
	FILE *fp = fopen(fileName, "wb");
	if (fp == 0) return 0;

	//申请位图文件头结构变量,填写文件头信息
	BITMAPFILEHEADER fileHead;
	fileHead.bfType = 0x4D42;//bmp类型

							 //bfSize是图像文件4个组成部分之和
	fileHead.bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)
		+ colorTablesize + lineByte*height;
	fileHead.bfReserved1 = 0;
	fileHead.bfReserved2 = 0;

	//bfOffBits是图像文件前三个部分所需空间之和
	fileHead.bfOffBits = 54 + colorTablesize;

	//写文件头进文件
	fwrite(&fileHead, sizeof(BITMAPFILEHEADER), 1, fp);

	//申请位图信息头结构变量,填写信息头信息
	BITMAPINFOHEADER head;
	head.biBitCount = bytesPerPixel * 8;
	head.biClrImportant = 0;
	head.biClrUsed = 0;
	head.biCompression = 0;
	head.biHeight = height;
	head.biPlanes = 1;
	head.biSize = 40;
	head.biSizeImage = lineByte*height;
	head.biWidth = width;
	head.biXPelsPerMeter = 0;
	head.biYPelsPerMeter = 0;
	//写位图信息头进内存
	fwrite(&head, sizeof(BITMAPINFOHEADER), 1, fp);

	//如果灰度图像,有颜色表,写入文件
	if (bytesPerPixel == 1)
		fwrite(pColorTable, sizeof(RGBQUAD), 256, fp);

	//写位图数据进文件
	if (bytesPerPixel == 1)
	{
		fwrite(imageData, height*lineByte, 1, fp);
	}
	else
	{
		GLubyte *newData = new GLubyte[width * height*bytesPerPixel];
		memcpy(newData, imageData, width * height*bytesPerPixel);

		for (long i = 0; i < width * height; i++)
		{
			GLubyte temp = newData[0 + i*bytesPerPixel];
			newData[0 + i*bytesPerPixel] = newData[2 + i*bytesPerPixel];
			newData[2 + i*bytesPerPixel] = temp;
		}
		fwrite(newData, height*lineByte, 1, fp);
		delete[]newData;
	}

	//关闭文件
	fclose(fp);
	return 1;
}

void CCTexture::setCenterPos(Vector3D center)
{
	this->centerPos = center;
}


void CCTexture::draw()
{
	float halfWid = width / 2;
	float halfHei = height / 2;
	Vector3D temp = centerPos;
	glEnable(GL_TEXTURE_2D);
	glEnable(GL_BLEND);
	glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
	glBindTexture(GL_TEXTURE_2D, texID);
	glBegin(GL_QUADS);
	glTexCoord2f(0, 0);	glVertex3f(centerPos.x- halfWid, centerPos.y+halfHei, 1);
	glTexCoord2f(1, 0);	glVertex3f(centerPos.x+ halfWid, centerPos.y+halfHei, 1);
	glTexCoord2f(1, 1); glVertex3f(centerPos.x + halfWid, centerPos.y-halfHei, 1);
	glTexCoord2f(0, 1); glVertex3f(centerPos.x -halfWid, centerPos.y-halfHei, 1);
	glEnd();
	glDisable(GL_TEXTURE_2D);
	glDisable(GL_BLEND);
}


CCTexture::~CCTexture()
{
	glDeleteTextures(1,&texID);
	delete imageData;
}


CTextureManager::CTextureManager()
{
}


CTextureManager*CTextureManager::getInstance()
{
	if (instance == NULL)
	{
		instance = new CTextureManager;
	}
	return instance;
}


CCTexture* CTextureManager::addTexture(char *filename,float width, float height)
{
	list<CCTexture*>::iterator iter;
	for (iter = textureList.begin(); iter != textureList.end(); iter++)
	{
		if (strcmp((*iter)->getName(), filename) == 0)
		{
			(*iter)->uReference++;
			return *iter;
		}
	}
	textureList.push_back(new CCTexture(filename));
	return textureList.back();
}


CCTexture* CTextureManager::addTexture(const char *filename)
{
	const char*lp = strrchr(filename, '.');
	char temp[255];
	if(strcmp(lp,".bmp")==0||strcmp(lp,".jpg")==0)
	{
		strcpy(temp, filename);
	}
	else 
	{
		memset(temp, 0, 255);
		strncpy(temp,filename,lp-filename);
		strcat(temp, ".png");
	}



	list<CCTexture*>::iterator iter;
	for (iter = textureList.begin(); iter != textureList.end(); iter++)
	{
		if (strcmp((*iter)->getName(), filename) == 0)
		{
			(*iter)->uReference++;
			return *iter;
		}
	}
	textureList.push_back(new CCTexture(temp));
	return textureList.back();

}


CCTexture* CTextureManager::addTexture(char *name, int width, int height, int BitCount)
{
	list<CCTexture*>::iterator iter;
	for (iter = textureList.begin(); iter != textureList.end(); iter++)
	{
		if (strcmp((*iter)->getName(), name) == 0)
		{
			(*iter)->uReference++;
			return *iter;
		}
	}
	textureList.push_back(new CCTexture(name, width, height, BitCount));
	return textureList.back();
}


void CTextureManager::releaseRef()
{
	for (auto iter = textureList.begin(); iter != textureList.end();)
	{
		if ((*iter)->uReference == 0)
		{
			delete(*iter);
			textureList.erase(iter++);
		}
		else
			iter++;
	}

}



void CTextureManager::resetRef()
{
	for (auto iter = textureList.begin(); iter != textureList.end(); iter++)
	{
		(*iter)->uReference = 0;
	}
}




void CTextureManager::CreateGLTextures(int wid, int hei, int biBitCount, OUT GLuint &texture, OUT GLubyte*&textureData)
{
	glGenTextures(1, &texture);
	glBindTexture(GL_TEXTURE_2D, texture);
	if (biBitCount == 8)
	{
		textureData = new GLubyte[wid*hei];
		memset(textureData, 0, wid*hei);
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);		// 设置过滤器为线性过滤
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
		glTexImage2D(GL_TEXTURE_2D, 0, 1, wid, hei, 0, GL_LUMINANCE, GL_UNSIGNED_BYTE, textureData);
	}
	else if (biBitCount == 24)
	{
		textureData = new GLubyte[wid*hei * 3];
		memset(textureData, 0, wid*hei * 3);

		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);		// 设置过滤器为线性过滤
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
		glTexImage2D(GL_TEXTURE_2D, 0, 3, wid, hei, 0, GL_RGB, GL_UNSIGNED_BYTE, textureData);
	}

}

CTextureManager::~CTextureManager()
{
	for (auto iter = textureList.begin();iter != textureList.end();)
	{
		delete (*iter);
		textureList.erase(iter++);
	}
	textureList.clear();
}

保存BMP格式图片(只能保存BMP)

这个函数当时主要是能把编辑器里刷的地形保存出去使用的。

BOOL CCTexture::saveBmp(const char *fileName)
{
	//如果位图数据指针为0,则没有数据传入,函数返回
	if (!imageData)
		return 0;

	//灰度图
	static RGBQUAD *pColorTable = NULL;
	if (pColorTable == NULL)
	{
		pColorTable = new RGBQUAD[256];
		for (int i = 0; i < 256; i++)
		{
			pColorTable[i].rgbBlue = i;
			pColorTable[i].rgbGreen = i;
			pColorTable[i].rgbRed = i;
			pColorTable[i].rgbReserved = 0;
		}
	}

	//颜色表大小,以字节为单位,灰度图像颜色表为1024字节,彩色图像颜色表大小为0
	int colorTablesize = 0;
	if (bytesPerPixel == 1)
		colorTablesize = 1024;
	
	//待存储图像数据每行字节数为4的倍数
	int lineByte = (width * bytesPerPixel + 3) / 4 * 4;

	//以二进制写的方式打开文件
	FILE *fp = fopen(fileName, "wb");
	if (fp == 0) return 0;

	//申请位图文件头结构变量,填写文件头信息
	BITMAPFILEHEADER fileHead;
	fileHead.bfType = 0x4D42;//bmp类型

							 //bfSize是图像文件4个组成部分之和
	fileHead.bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)
		+ colorTablesize + lineByte*height;
	fileHead.bfReserved1 = 0;
	fileHead.bfReserved2 = 0;

	//bfOffBits是图像文件前三个部分所需空间之和
	fileHead.bfOffBits = 54 + colorTablesize;

	//写文件头进文件
	fwrite(&fileHead, sizeof(BITMAPFILEHEADER), 1, fp);

	//申请位图信息头结构变量,填写信息头信息
	BITMAPINFOHEADER head;
	head.biBitCount = bytesPerPixel * 8;
	head.biClrImportant = 0;
	head.biClrUsed = 0;
	head.biCompression = 0;
	head.biHeight = height;
	head.biPlanes = 1;
	head.biSize = 40;
	head.biSizeImage = lineByte*height;
	head.biWidth = width;
	head.biXPelsPerMeter = 0;
	head.biYPelsPerMeter = 0;
	//写位图信息头进内存
	fwrite(&head, sizeof(BITMAPINFOHEADER), 1, fp);

	//如果灰度图像,有颜色表,写入文件
	if (bytesPerPixel == 1)
		fwrite(pColorTable, sizeof(RGBQUAD), 256, fp);

	//写位图数据进文件
	if (bytesPerPixel == 1)
	{
		fwrite(imageData, height*lineByte, 1, fp);
	}
	else
	{
		GLubyte *newData = new GLubyte[width * height*bytesPerPixel];
		memcpy(newData, imageData, width * height*bytesPerPixel);

		for (long i = 0; i < width * height; i++)
		{
			GLubyte temp = newData[0 + i*bytesPerPixel];
			newData[0 + i*bytesPerPixel] = newData[2 + i*bytesPerPixel];
			newData[2 + i*bytesPerPixel] = temp;
		}
		fwrite(newData, height*lineByte, 1, fp);
		delete[]newData;
	}

	//关闭文件
	fclose(fp);
	return 1;
}

加载文件&绑定纹理选择滤波类型

因为使用的是Windows的图像API加载的,加载进来的图像格式是BGR格式的,要从BGR转换成OpenGL需要的RGB格式。图片过滤方式选择?GL_LINEAR(也叫线性过滤,(Bi)linear Filtering)它会基于纹理坐标附近的纹理像素,计算出一个插值,近似出这些纹理像素之间的颜色。

使用线性过滤的优点与缺点
? 优点:
? ? ? ? 1.质量高:避免了在远距离情况下的采样频率低和数据频率高造成的失真和摩尔纹,效果比无Mipmap好得多。

? ? ? ? 2.性能好:避免了不使用Mipmap下距离远时采样频率低和数据频率高而照成texture cache命中率不高(相邻Pixel采样Texel时uv相差比较大)使性能下降。

? 缺点:
? ? ? ? ?1.占用显存,可使用ue的纹理流缓存优化(IO换显存)。

CCTexture::CCTexture(const char* fileName,float wid,float hei)
{
	texID = 0;
	CImage img;
	HRESULT hr = img.Load(CUser::getInstance()->CharToWchar(fileName));
	if (FAILED(hr))
	{
		char temp[256];
		sprintf_s(temp, 256, "%s打开失败", fileName);
		MessageBox(NULL, CUser::getInstance()->CharToWchar(temp), L"错误", 0);
		return;
	}

	strcpy(name, fileName);
	HBITMAP hbmp = img;
	BITMAP bm;
	GetObject(hbmp, sizeof(bm), &bm);

	if (wid > 1 && hei > 1)
	{
		width = wid;	height = hei;
	}
	else 
	{
		width = bm.bmWidth;	height = bm.bmHeight;
	}
	
	bytesPerPixel = bm.bmBitsPixel / 8;

	if (bytesPerPixel == 3)
	{
		type = GL_RGB;
	}
	else if (bytesPerPixel == 4)
	{
		type = GL_RGBA;
	}
	else if (bytesPerPixel == 1)
	{
		type = GL_LUMINANCE;
	}
		
	imageData = new GLubyte[width*height*bytesPerPixel];
	memcpy(imageData, bm.bmBits, width*height*bytesPerPixel);
	// Convert From BGR To RGB Format And Add An Alpha Value Of 255
	if (bytesPerPixel != 1)
	{
		for (long i = 0; i < width * height; i++)
		{
			GLubyte temp = imageData[0 + i*bytesPerPixel];
			imageData[0 + i*bytesPerPixel] = imageData[2 + i*bytesPerPixel];
			imageData[2 + i*bytesPerPixel] = temp;
		}
	}
	else //等于8位时处理的
	{
		for (int i = 0; i < height; i++)
		{
			memcpy(imageData + i*width, img.GetPixelAddress(0, i), width);
		}
	}
	glGenTextures(1, (GLuint*)&texID);
	glBindTexture(GL_TEXTURE_2D, texID);
	glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
	glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR);//纹理过滤模式
	gluBuild2DMipmaps(GL_TEXTURE_2D, type, width, height, type, GL_UNSIGNED_BYTE, imageData);
}

  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2022-09-13 11:49:37  更:2022-09-13 11:50:19 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/17 3:54:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码