最近在写GBA的程序。GBA运行的是C的裸机代码,而中途使用的很多小工具则用C#写的,例如:图片转换到.h头文件,制作三角函数表,还有像这次介绍的将圆柱面贴图映射到半球贴图的小工具。这样的小功能,用C#写就是一小会的事,效率非常高。

这时候就能体现出来——各语言有各自的用途,C用来做硬件开发,C++做软件开发,C#做快速功能。

这次要解决的问题是:

制作3D时,一个球体的贴图,我们通常映射成一个柱面,就像我们常见的世界地图一样。如图(这张图片是从NASA上下载的卫星图)

但是如果我们想用半球映射来贴图,例如分成北半球和南半球分别映射,那不一定能找到合适的贴图。

但我们可以很简单的将柱面贴图转换成半球贴图。如图(半球之外的区域我也映射了):

代码如下:

using System;
using System.Drawing;
using System.Drawing.Imaging;  

public void ToSphereMap(int spWidth, int spHeight)
{
    //加载bmp柱面贴图
    Bitmap bmp = newBitmap("CylinderMap.png");
    //创建球面贴图
    Bitmap sphereMap = new Bitmap(spWidth, spHeight * 2);

    int clHeight = bmp.Height;
    int clWidth = bmp.Width;

    double a, b;    //半球面贴图平面上xy坐标(a,b)
    double z;       //对应球体上点(x,y,z)的z轴坐标
    double s;       //轴面角度
    int clX, clY1, clY2;

  for (int spY = 0; spY < spHeight; spY++)
  {     for (int spX = 0; spX < spWidth; spX++)     {       //r = sqrt(a^2+b^2) = x^2+y^2 = 1-z^2       a = spX * 2 / (float)spWidth - 1;       b = spY * 2 / (float)spHeight - 1;       z = 1 - Math.Sqrt(a * a + b * b);       //if (z < 0) z = -Math.Sqrt(-z);       //else z = Math.Sqrt(z);       s = Math.Atan2(y, x) + Math.PI;       clX = s * clWidth / (2 * Math.PI);       clY1 = (int)((1 - z) * clHeight * 0.5);       clY2 = (int)((1 + z) * clHeight * 0.5);       if (clX < 0) clX = 0;       else if (clX > clWidth - 1) clX = clWidth - 1;       if (clY1 < 0) clY1 = 0;       else if (clY1 > clHeight - 1) clY1 = clHeight - 1;       if (clY2 < 0) clY2 = 0;       else if (clY2 > clHeight - 1) clY2 = clHeight - 1;       sphereMap.SetPixel(spX, spY, bmp.GetPixel(clX, clY1));       sphereMap.SetPixel(spX, spY + spHeight, bmp.GetPixel(clX, clY2));       }     }
  //保存图片   sphereMap.Save("SphereMap.png", ImageFormat.Png); }

  

补充一些注释:

1、转换后的半球贴图分上下两部分,分别是北南半球。

2、转换的主要思路是从半球贴图上的位置寻找对应的柱面贴图坐标。

  (a, b)是半球平面上的坐标,对应于球体上的点坐标(x, y, z);

3、具体的映射转换根据柱面映射和半球映射的方法有所区别,需要按情况分析。

  这里使用的映射方法稍后我会上一张图解释。

01-22 04:45