MFC下用矩阵实现图形学之立方体平移、比例、旋转、投影变换算法

本文主要是介绍MFC下用矩阵实现图形学之立方体平移、比例、旋转、投影变换算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

//*********************************
//绘制坐标系以及初始立方体
//*********************************
void CTransGraphicsView::OnDraw(CDC* pDC)
{
 CTransGraphicsDoc* pDoc = GetDocument();
 ASSERT_VALID(pDoc);
 HBRUSH hbrush;
 CPen pen;
 HPEN hPen;
 // TODO: add draw code for native data here
 pen.CreatePen(PS_SOLID,3,RGB(255,0,0));
 hPen=(HPEN)pDC->SelectObject(pen);
 pDC->SetViewportOrg(cxClient/2,cyClient/2);
 pDC->MoveTo(-cxClient/2,0);
 pDC->LineTo(cxClient/2,0);
 pDC->MoveTo(0,-cyClient/2);
 pDC->LineTo(0,cyClient/2);
 pen.DeleteObject();
    pDC->SelectObject(hPen);
 this->ShadowTrans(-45);
 for(int i=0;i<4;i++)
 {
   pDC->MoveTo(cpoints[i]);
   pDC->LineTo(cpoints[i+4]);
 }
 hbrush = (HBRUSH)pDC->SelectObject(GetStockObject(NULL_BRUSH)) ;
 pDC->Polygon(cpoints,4);
 pDC->Polygon(cpoints+4,4);
 
}
//********************
//矩阵的乘法
//*********************
void CTransGraphicsView::MutiTransMarix(double a[1][4],double b[4][4],double MutiResult[1][4])
{
 for(int i=0;i<4;i++) MutiResult[0][i] = 0;
 for(i=0; i<1;i++)
  for(int j=0;j<4;j++)
   for(int k=0;k<4;k++)
    MutiResult[i][j] += a[i][k]*b[k][j];
}
//******************************
//将坐标转化为矩阵形式(向量)
//******************************
void CTransGraphicsView::TransPointToMarix(ThPoint &tp,double a[1][4])
{
 a[0][0] = tp.x;
 a[0][1] = tp.y;
 a[0][2] = tp.z;
 a[0][3] = 1;
}
//******************************
//将矩阵(向量)转化为三维坐标形式
//******************************
void CTransGraphicsView::TransMarixToPoint(double a[1][4],ThPoint &tp)
{
 tp.x = a[0][0];
 tp.y = a[0][1];
 tp.z = a[0][2];
}
//******************************
//立方体的平移算法
//******************************
void CTransGraphicsView::MoveTrans(double x,double y,double z)
{
    double a[8][4],result[1][4];
 double moveMarix[4][4] =
 {
  {1,0,0,0},{0,1,0,0},
  {0,0,1,0},{x,y,z,1}
 };
 
     for(int i=0;i<8;i++)
 {
          TransPointToMarix(points[i],a+i);
   MutiTransMarix(a+i,moveMarix,result);
   TransMarixToPoint(result,points[i]);
 }
 ShadowTrans(-45);
}
//****************************************************
//立方体的斜交投影算法(将三维坐标转化为二维在屏幕显示)
//****************************************************
void CTransGraphicsView::ShadowTrans(int degree)
{
 double a[8][4],result[1][4];
 ThPoint pts[8];
 for(int i=0;i<8;i++)
 {
  pts[i].x = points[i].x;
  pts[i].y = points[i].y;
  pts[i].z = points[i].z;
 }
 double shadowMarix[4][4] =
 {
  {1,0,0,0},{0,1,0,0},
  {cos(degree),sin(degree),0,0},{0,0,0,1}
 };
 
    for(i=0;i<8;i++)
 {
          TransPointToMarix(pts[i],a+i);
   MutiTransMarix(a+i,shadowMarix,result);
   TransMarixToPoint(result,pts[i]);
 }
 for(i=0;i<8;i++)
 {
  cpoints[i].x = (int)pts[i].x;
  cpoints[i].y = (int)pts[i].y;
 }
}
//***************************************
//立方体旋转算法,旋转中心可以是x,y或z轴
//degree旋转的度数, rc旋转中心
//****************************************
void CTransGraphicsView::RotateTrans(double degree,CString rc)

    double a[8][4],result[1][4];
    int i = 0;
   
 if(rc == "z")
 {
  double xRotateMarix[4][4] =
   {
    {cos(degree*3.14/180),sin(degree*3.14/180),0,0},
    {-sin(degree*3.14/180),cos(degree*3.14/180),0,0},
    {0,0,1,0},{0,0,0,1},
   };
         for(i=0;i<8;i++)
   {
           TransPointToMarix(points[i],a+i);
        MutiTransMarix(a+i,xRotateMarix,result);
     TransMarixToPoint(result,points[i]);
   }
 }
 else if(rc == "x")
  {
  double yRotateMarix[4][4]=
   {
    {1,0,0,0},
    {0,cos(degree*3.14/180),sin(degree*3.14/180),0},
    {0,-sin(degree*3.14/180),cos(degree*3.14/180),0},
    {0,0,0,1}
   };
               for(i=0;i<8;i++)
   {
                   TransPointToMarix(points[i],a+i);
            MutiTransMarix(a+i,yRotateMarix,result);
     TransMarixToPoint(result,points[i]);
   }
     
  }
  else if(rc == "y")
  {
  double zRotateMarix[4][4]=
   {
    {cos(degree*3.14/180),0,-sin(degree*3.14/180),0},
    {0,1,0,0},
    {sin(degree*3.14/180),0,cos(degree*3.14/180),0},
    {0,0,0,1}
  };
         for(i=0;i<8;i++)
   {
           TransPointToMarix(points[i],a+i);
        MutiTransMarix(a+i,zRotateMarix,result);
     TransMarixToPoint(result,points[i]);
   }
     
  }
    ShadowTrans(-45);
}
//*********************************
//立方体的比例变换算法
//*********************************
void CTransGraphicsView::BigerOrSmallerTrans(double timeX,double timeY,double timeZ)
{
    double a[8][4],result[1][4];
 double bsMarix[4][4] =
 {
  {timeX,0,0,0},{0,timeY,0,0},
  {0,0,timeZ,0},{0,0,0,1}
 };
       for(int i=0;i<8;i++)
 {
          TransPointToMarix(points[i],a+i);
   MutiTransMarix(a+i,bsMarix,result);
   TransMarixToPoint(result,points[i]);
 }
 ShadowTrans(-45);
}
#endif //_DEBUG

/
// CTransGraphicsView message handlers
//********************************
//获取屏幕的大小
//********************************
void CTransGraphicsView::OnSize(UINT nType, int cx, int cy)
{
 CView::OnSize(nType, cx, cy);
 
 // TODO: Add your message handler code here
 cxClient = cx;
 cyClient = cy;
}
//*****************************************
//根据对话框输入的参数进行比例变换
//*****************************************
void CTransGraphicsView::OnBgsm()
{
 // TODO: Add your command handler code here
 BGSMDialog B_dlg;
 if(IDOK == B_dlg.DoModal())
 {
         BiggerOrSmallerTrans(B_dlg.m_sx,B_dlg.m_sy,B_dlg.m_sz);
 }
 Invalidate();
 UpdateWindow();
}
//********************************
//根据对话框输入的参数进行平移变换
//********************************
void CTransGraphicsView::OnMove()
{
 // TODO: Add your command handler code here
 CMoveDialog M_dlg;
 if(IDOK == M_dlg.DoModal())
 {
        MoveTrans(M_dlg.m_dx,M_dlg.m_dy,M_dlg.m_dz);
 }
 Invalidate();
 UpdateWindow();
}
//********************************
//根据对话框输入的参数进行旋转变换
//********************************
void CTransGraphicsView::OnRotate()
{
 // TODO: Add your command handler code here
 RoateDialog BT_dlg;
 if(IDOK == BT_dlg.DoModal())
 {
      RotateTrans(BT_dlg.m_degree,BT_dlg.m_rc);
 }
 Invalidate();
 UpdateWindow();
}
//**************************************************
//         根据键盘调用相应的算法
//其中down向下平移一个单位,up向上平移一个单位,
//left向左平移一个单位,right向右平移一个单位
//***************************************************
void CTransGraphicsView::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
{
 // TODO: Add your message handler code here and/or call default
 switch(nChar)
 {
  case VK_DOWN:
        MoveTrans(0,1,0);
  break;
  case VK_UP:
   MoveTrans(0,-1,0);
   break;
  case VK_LEFT:
         MoveTrans(-1,0,0);
   break;
  case VK_RIGHT:
   MoveTrans(1,0,0);
   break;
 }
 Invalidate();
 UpdateWindow();
 CView::OnKeyDown(nChar, nRepCnt, nFlags);
}
//******************************************************
//           根据键盘字符键调用相应的算法
//其中x以x轴为旋转中心旋转一度,y以y轴为旋转中心旋转一度,
//z以z轴为旋转中心旋转一度,b变大,s缩小
//*******************************************************
void CTransGraphicsView::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags)
{
 // TODO: Add your message handler code here and/or call default
 switch(nChar)
 {
 case 'x':
        RotateTrans(1,"x");
  break;
 case 'y':
        RotateTrans(1,"y");
  break;
 case 'z':
  RotateTrans(1,"z");
  break;
 case 'b':
        BiggerOrSmallerTrans(1.1,1.1,1.1);
  break;
 case 's':
  BiggerOrSmallerTrans(0.9,0.9,0.9);
  break;
 }
 Invalidate();
 UpdateWindow();
 CView::OnChar(nChar, nRepCnt, nFlags);
}

 

这篇关于MFC下用矩阵实现图形学之立方体平移、比例、旋转、投影变换算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/636179

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

计算绕原点旋转某角度后的点的坐标

问题: A点(x, y)按顺时针旋转 theta 角度后点的坐标为A1点(x1,y1)  ,求x1 y1坐标用(x,y)和 theta 来表示 方法一: 设 OA 向量和x轴的角度为 alpha , 那么顺时针转过 theta后 ,OA1 向量和x轴的角度为 (alpha - theta) 。 使用圆的参数方程来表示点坐标。A的坐标可以表示为: \[\left\{ {\begin{ar

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

基于Springboot + vue 的抗疫物质管理系统的设计与实现

目录 📚 前言 📑摘要 📑系统流程 📚 系统架构设计 📚 数据库设计 📚 系统功能的具体实现    💬 系统登录注册 系统登录 登录界面   用户添加  💬 抗疫列表展示模块     区域信息管理 添加物资详情 抗疫物资列表展示 抗疫物资申请 抗疫物资审核 ✒️ 源码实现 💖 源码获取 😁 联系方式 📚 前言 📑博客主页:

人工智能机器学习算法总结神经网络算法(前向及反向传播)

1.定义,意义和优缺点 定义: 神经网络算法是一种模仿人类大脑神经元之间连接方式的机器学习算法。通过多层神经元的组合和激活函数的非线性转换,神经网络能够学习数据的特征和模式,实现对复杂数据的建模和预测。(我们可以借助人类的神经元模型来更好的帮助我们理解该算法的本质,不过这里需要说明的是,虽然名字是神经网络,并且结构等等也是借鉴了神经网络,但其原型以及算法本质上还和生物层面的神经网络运行原理存在

探索蓝牙协议的奥秘:用ESP32实现高质量蓝牙音频传输

蓝牙(Bluetooth)是一种短距离无线通信技术,广泛应用于各种电子设备之间的数据传输。自1994年由爱立信公司首次提出以来,蓝牙技术已经经历了多个版本的更新和改进。本文将详细介绍蓝牙协议,并通过一个具体的项目——使用ESP32实现蓝牙音频传输,来展示蓝牙协议的实际应用及其优点。 蓝牙协议概述 蓝牙协议栈 蓝牙协议栈是蓝牙技术的核心,定义了蓝牙设备之间如何进行通信。蓝牙协议