欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于OpenGL實現(xiàn)多段Bezier曲線拼接

 更新時間:2020年04月20日 16:47:25   作者:ryfdizuo  
這篇文章主要為大家詳細介紹了基于OpenGL實現(xiàn)多段Bezier曲線拼接,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文實例為大家分享了OpenGL實現(xiàn)多段Bezier曲線拼接的具體代碼,供大家參考,具體內(nèi)容如下

運行程序的交互方式有點類似corelDraw中的自由曲線繪制,或者photoShop中的鋼筆自由路徑繪制。

截圖:

將BezierCurve封裝成了一個類,代碼如下:

#ifndef _BEZIERCURVE_H
#define _BEZIERCURVE_H

#include "vec3.hpp"
#include <vector>
#include <iostream>
#include <gl/glut.h>

using namespace std;
//// 3次bezier曲線: 四個控制節(jié)點。曲線經(jīng)過首末兩個頂點。

class BezierCurve
{
public:
 //cell一共有四個控制頂點
 // -cell經(jīng)過V0和V3頂點,
 // -cell的始端相切于直線:(V0, V1) 和末端相切于(V2,V3)
 class BezierCell
 {
 public:
 BezierCell(int i0, int i1, int i2, int i3)
 {
  setValue(i0, i1, i2, i3);
 }

 void setValue(int i0, int i1, int i2, int i3)
 {
  ctrlVertxIndex[0] = i0;
  ctrlVertxIndex[1] = i1;
  ctrlVertxIndex[2] = i2;
  ctrlVertxIndex[3] = i3;
 }
 const int operator[](int index) const
 {
  if (index > 3 || index < 0)
  return -1;

  return ctrlVertxIndex[index];
 }
 int ctrlVertxIndex[4];
 };

 enum eventType
 {
 LButtonDown = 0,
 MouseMove = 1,
 LButtonUp = 2
 };
 
 enum { Bezier3CtrlPnt = 4 };

 BezierCurve() { clear(); }
 ~BezierCurve(){}

 void begin()
 {
 // 開啟求值器
 glEnable(GL_MAP1_VERTEX_3);
 clear();
 }
 
 void mouseSynchro(eventType type, const Vec3d& v) //響應(yīng)鼠標motion
 {
 //////////////////////////////////////////////////////////////////////////
 //LButtonDown: 壓入點
 if (type == LButtonDown)
 {
  if (isFirstRender)   //for the first cell
  {
  vertexVector.push_back(v); //push V0...
  vertexVector.push_back(v); //push V1...
  }
  else if ( cellRenderState() == cellRenderImple::Push ) //for any cell 
  {
  vertexVector.push_back(v); //push V2...
  vertexVector.push_back(v); //push V3... 

  cellRenderState.setChange(); //set the flag to change V3
  cellNum++;   //increase the cell counter
  }
 }
 //////////////////////////////////////////////////////////////////////////
 //MouseMove: 動態(tài)更新相應(yīng)的頂點數(shù)據(jù)
 else if (type == MouseMove) 
 {
  if (isFirstRender)   //for the first cell
  { 
  vertexVector.back() = v; //change the V1 immediately
  }
  else if ( cellRenderState() == cellRenderImple::Change )//for any cell
  {
  int vecSize = vertexVector.size(); 
  vertexVector[vecSize-2] = v; //change the V2 immediately
  }
 }

 //////////////////////////////////////////////////////////////////////////
 //LButtonUp: 為拼接做準備
 else if (type == LButtonUp) 
 { 
  if (isFirstRender)
  {
  //只有第一個BezierCell可以編輯bezierCell的起始段:(V0,V1)
  isFirstRender = false;
  }
  else if ( cellRenderState() == cellRenderImple::Change)
  {
  //if finish the current cell's render
  //利用v1和中點v0計算出v2:(v1 + v2) / 2 = v0
  //next cell begin: push the next cell's V1... 
  int vecSize = vertexVector.size();
  Vec3d v0 = vertexVector[vecSize-1];
  Vec3d v1 = vertexVector[vecSize-2];
  Vec3d v2 = 2 * v0 - v1;
  vertexVector.push_back(v2);

  //重置cellRenderFlag
  cellRenderState.setPush();
  }
 }
 //////////////////////////////////////////////////////////////////////////
 //更新數(shù)組的長度
 _updateVertexNum();
 }

 void end()
 {
 glDisable(GL_MAP1_VERTEX_3);
 }

 void renderCurve()
 {
 //////////////////////////////////////////////////////////////////////////
 //rendering vertex...
 for (int i=0; i<vertexVector.size(); i++)
 {
  Vec3d v = vertexVector[i];
  glBegin(GL_POINTS); 
  glVertex3dv(v.getValue());
  glEnd();
 }

 //////////////////////////////////////////////////////////////////////////
 //rendering moving tangent(切線)
 //(vertexNum-1, vertexNum-2)
 if ( vertexNum>=2 )
 {
  glEnable(GL_LINE_STIPPLE);
  {
  glLineStipple(1, 0x0101);
  glBegin(GL_LINES);
  {
   Vec3d v1 = vertexVector[vertexNum-1];
   Vec3d v2 = vertexVector[vertexNum-2];
   glVertex3dv(v1.getValue());
   glVertex3dv(v2.getValue());
  } glEnd();
  }glDisable(GL_LINE_STIPPLE);
 }

 //////////////////////////////////////////////////////////////////////////
 //if ( !_check() )
 // return;

 //rendering bezier cells...
 system("CLS");
 for (int i=0; i<cellNum; i++)
 {
  int pos = i * 3;
  if ( (pos+3) < vertexNum )
  renderBezierCell( BezierCell(pos, pos+1, pos+2, pos+3) );
 }
 //////////////////////////////////////////////////////////////////////////
 }

 // 3次bezier曲線經(jīng)過vetex0和vextex3
 void renderBezierCell(const BezierCell& cell)
 {
 double *pBuffer = new double[Bezier3CtrlPnt * 3];

 cout << "----------------------------------------------------" << endl;

 cout << "Vertex number : " << vertexNum << endl;
 cout << "Cell number : " << cellNum << endl;
 cout << "The render cell: " << cell[0] << " " << cell[1] << " " << cell[2] << " " << cell[3] << endl;

 for (int i = 0, bg = 0; i<4; i++)
 {
  Vec3d v = vertexVector[ cell[i] ];
  pBuffer[bg++] = v.x();
  pBuffer[bg++] = v.y();
  pBuffer[bg++] = v.z();
  
  cout << v.x() << " " << v.y() << " " << v.z() << endl;
 }cout << "----------------------------------------------------" << endl;

 glMap1d(GL_MAP1_VERTEX_3, 0.0, 1.0, 3, Bezier3CtrlPnt, pBuffer);
 glBegin(GL_LINE_STRIP);
 {
  for (int i = 0; i <= 30; i++) 
  glEvalCoord1f((GLfloat) i/30.0f);
 } glEnd();

 delete pBuffer; pBuffer = 0;
 }

 void clear()
 {
 cellNum = 0;
 vertexNum = 0;

 isFirstRender = true;

 vertexVector.clear();
 }
protected:
 bool _check() { vertexNum =vertexVector.size(); 
   return vertexNum == (cellNum - 1) * 3 + 4; }
 void _updateVertexNum() { vertexNum=vertexVector.size();}

 int cellNum;   //單元個數(shù)
 int vertexNum;   //頂點個數(shù)

 bool isFirstRender;   //首次標志
 std::vector<Vec3d> vertexVector; //頂點數(shù)組

 class cellRenderImple
 {
 public:
 enum RenderStep
 {
  Push = 0,
  Change = 1
 };
 cellRenderImple(){ setPush(); }
 bool operator()(void) { return flag; }
 void setPush() { flag = Push; }
 void setChange() { flag = Change; }

 RenderStep flag;  //cell的渲染狀態(tài)
 } cellRenderState;
};

測試程序如下:

#include <iostream>
#include <vector>
#include <GL/glut.h>

#include "BezierCurve.h"

using namespace std;

enum WindowSize{
 WinWidth = 1024,
 WinHeight = 768
};

int  g_Viewport[4];
double g_ModelMatrix[16];
double g_ProjMatrix[16];
BezierCurve myBezier;

//////////////////////////////////////////////////////////////////////////
void init();
void display();
void reshape(int w, int h);
void keyboard(unsigned char key, int x, int y);
void mouse(int button, int state, int x, int y);
void motion(int x, int y);

int main(int argc, char** argv)
{
 glutInit(&argc, argv);
 glutInitDisplayMode (GLUT_DOUBLE | GLUT_RGB);
 glutInitWindowSize (WinWidth, WinHeight);
 glutInitWindowPosition (100, 100);
 glutCreateWindow (argv[0]);

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

 init ();
 glutDisplayFunc(display);
 glutReshapeFunc(reshape);
 glutKeyboardFunc (keyboard);
 glutMouseFunc(mouse);
 glutMotionFunc(motion);

 glutMainLoop();

 return 0;
}

void init(void)
{
 glClearColor(0.0, 0.0, 0.0, 0.0);
 glShadeModel(GL_SMOOTH);

 myBezier.begin();
}

void display(void)
{
 glClear(GL_COLOR_BUFFER_BIT);

 glColor3f(1.0, 1.0, 0.0);
 glPointSize(5.0);

 glPushMatrix();
 {
 myBezier.renderCurve();
 }glPopMatrix();

 glutSwapBuffers();
}

void reshape(int w, int h)
{
 glViewport(0, 0, (GLsizei) w, (GLsizei) h);
 glMatrixMode(GL_PROJECTION);
 glLoadIdentity();
 if (w <= h)
 glOrtho(-5.0, 5.0, -5.0*(GLfloat)h/(GLfloat)w, 
 5.0*(GLfloat)h/(GLfloat)w, -5.0, 5.0);
 else
 glOrtho(-5.0*(GLfloat)w/(GLfloat)h, 
 5.0*(GLfloat)w/(GLfloat)h, -5.0, 5.0, -5.0, 5.0);
 glMatrixMode(GL_MODELVIEW);
 glLoadIdentity();
}

void keyboard(unsigned char key, int x, int y)
{
 switch (key) {
  case 27:
  exit(0);
  break;
 }
}

void mouse(int button, int state, int x, int y)
{
 double vertex[3];

 //獲取矩陣信息
 glGetIntegerv(GL_VIEWPORT, g_Viewport);
 glGetDoublev(GL_MODELVIEW_MATRIX, g_ModelMatrix);
 glGetDoublev(GL_PROJECTION_MATRIX, g_ProjMatrix);

 y = g_Viewport[3] - y;
 gluUnProject( x, y, 0,
 g_ModelMatrix, g_ProjMatrix, g_Viewport,
 &vertex[0], &vertex[1], &vertex[2] );

 if (button==GLUT_LEFT && state==GLUT_DOWN)
 {
 myBezier.mouseSynchro( BezierCurve::LButtonDown, vertex );
 glutSetCursor( GLUT_CURSOR_RIGHT_ARROW );
 }
 else if (button == GLUT_LEFT && state == GLUT_UP)
 {
 myBezier.mouseSynchro( BezierCurve::LButtonUp, vertex );
 }

 glutPostRedisplay();
}

//////////////////////////////////////////////////////////////////////////
// 計算控制節(jié)點
void motion(int x, int y)
{
 double vertex[3];

 glutSetCursor( GLUT_CURSOR_CROSSHAIR );
 y = g_Viewport[3] - y;

 gluUnProject( x, y, 0,
 g_ModelMatrix, g_ProjMatrix, g_Viewport,
 &vertex[0], &vertex[1], &vertex[2] );

 myBezier.mouseSynchro( BezierCurve::MouseMove, vertex );
 glutPostRedisplay();
}

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • C語言實現(xiàn)ATM機存取款系統(tǒng)

    C語言實現(xiàn)ATM機存取款系統(tǒng)

    這篇文章主要為大家詳細介紹了C語言實現(xiàn)ATM機存取款系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • C++如何計算結(jié)構(gòu)體與對象的大小

    C++如何計算結(jié)構(gòu)體與對象的大小

    這篇文章主要給大家介紹了關(guān)于C++如何計算結(jié)構(gòu)體與對象大小的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-05-05
  • C++中rapidjson將map轉(zhuǎn)為json的方法

    C++中rapidjson將map轉(zhuǎn)為json的方法

    今天小編就為大家分享一篇關(guān)于C++中rapidjson將map轉(zhuǎn)為json的方法,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-04-04
  • VC++ 獲取系統(tǒng)時間的方法匯總

    VC++ 獲取系統(tǒng)時間的方法匯總

    本文給大家匯總介紹了5種VC++中獲取系統(tǒng)時間的方法,十分的簡單實用,有需要的小伙伴可以參考下。
    2015-07-07
  • C++右值引用問題解決

    C++右值引用問題解決

    本文主要介紹了C++右值引用問題解決,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-06-06
  • Qt實現(xiàn)簡易計時器的示例代碼

    Qt實現(xiàn)簡易計時器的示例代碼

    計時器實現(xiàn)四個功能:開始計時、停止計時、暫停計時以及打點。當點擊暫停時,開始按鈕和停止按鈕無法點擊。當點擊停止時,開始按鈕和暫停按鈕無法點擊,此時停止按鈕變?yōu)榍辶?。本文將用Qt實現(xiàn)這樣的一個計時器,需要的可以參考一下
    2022-06-06
  • C++ 頭文件系列(set)詳解

    C++ 頭文件系列(set)詳解

    一般而言,每個C++/C程序通常由頭文件和定義文件組成。頭文件作為一種包含功能函數(shù)、數(shù)據(jù)接口聲明的載體文件,主要用于保存程序的聲明,而定義文件用于保存程序的實現(xiàn) 。
    2017-02-02
  • C++哈希表之線性探測法實現(xiàn)詳解

    C++哈希表之線性探測法實現(xiàn)詳解

    線性探測法的優(yōu)點:只要散列表未滿,總能找到一個不沖突的散列地址;缺點:每個產(chǎn)生沖突的記錄被散列到離沖突最近的空地址上,從而又增加了更多的沖突機會
    2022-05-05
  • C/C++?Linux?Socket網(wǎng)絡(luò)編程流程分析

    C/C++?Linux?Socket網(wǎng)絡(luò)編程流程分析

    這篇文章主要介紹了C/C++?Linux?Socket網(wǎng)絡(luò)編程,Linux環(huán)境中的C/C++?socket?與Window環(huán)境中的C/C++?socket類似,本文所記錄的是TCP協(xié)議的socket編程,圖文實例相結(jié)合給大家介紹的非常詳細,需要的朋友可以參考下
    2023-02-02
  • C語言設(shè)計簡易電話簿

    C語言設(shè)計簡易電話簿

    這篇文章主要為大家詳細介紹了C語言設(shè)計簡易電話簿,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-12-12

最新評論