C++实现数字媒体二维图像变换

必备环境

  • glut.h 头文件
  • glut32.lib 对象文件库
  • glut32.dll 动态连接库

程序说明

C++实现了用glut画正方形,画三角形的功能。并附带放大缩小,旋转,平移的功能。同时实现了填充颜色这个功能。

操作说明

重要说明

define CLIP_METHOD 0

该参数为0时,切割方式为 多边形 切割(默认)

该参数为1时,切割方式为 点 切割

define FILL_ENABLE !CLIP_METHOD&&1

该参数为1并且启用多边形切割时,启用填充对象(默认)

该参数为0时,不填充对象,仅描绘边界

define READ_MULTI_FILE 0

该参数为0时,读取文件为单文件,窗口可移动

该参数为1时,读取文件为多文件,在任意键盘控制确认显示下一个,窗口可移动(默认)

指令说明

  • reset :把变换矩阵重置为单位矩阵
  • scale :把变换矩阵进行相应的放大缩小
  • rotate : 把变换矩阵进行相应的旋转
  • translate : 把变换矩阵进行相应的平移
  • square :根据变换矩阵创建一个矩形对象
  • triangle :根据变换矩阵创建一个三角形对象
  • clearData :清空所有的对象数据
  • clearScreen :清空屏幕
  • end : 结束程序

函数说明

void initial()

输出程序信息,并提示如何输入

void scale(float sx, float sy)

改变变换矩阵进行相应放大缩小

void rotate(float degree)

改变变换矩阵进行相应的角度旋转

void translate(float tx, float ty)

改变变换矩阵进行相应的平移变换

void reset()

重置变换矩阵为单位矩阵

void square()

根据变换矩阵创建一个矩形对象并存储

void triangle()

根据变换矩阵创建一个三角形对象并存储

void view(float wxl, float wxr, float wyb, float wyt, float vxl, float vxr, float vyb, float vyt)

根据参数把对象坐标系的区域进行投影到视图坐标系下,同时把裁剪非可见区域,并对投影区域进行颜色填充

void clearData()

清除全部对象的数据

void clearScreen()

清空整个视窗屏幕

void displayFunc(void)

视窗初始化和恢复的时候调用的函数

void ReadInput(bool& IsExit)

从标准输入输出中读取指令

void Matrix_Multi_Matrix(float a[][CTM_SIZE], float b[][CTM_SIZE], float c[][CTM_SIZE])

矩阵乘法,c = a * b

void update_CurMatrix(float resource[][CTM_SIZE], float destination[][CTM_SIZE])

更新变换矩阵

void update_WVM(int sx, int sy, int tx, int ty)

更新投影函数

void AddtoSquareStore(float matrix[][CTM_SIZE])

增加一个矩形对象到仓库中

void AddtoTriangleStore(float matrix[][CTM_SIZE])

增加一个三角形到仓库中

void GetFromSquareStore(int index, float destination[][CTM_SIZE])

从仓库中获取一个矩形对象

void GetFromTriangleStore(int index, float destination[][CTM_SIZE])

从仓库中获取一个三角形对象

void DrawSquareDirect(int matrix[][CTM_SIZE])

直接绘制出仓库中最新的矩形对象

void DrawSquareByindex(int index)

根据序号绘制出仓库中指定的矩形对象

void DrawTriangleDirect(int matrix[][CTM_SIZE])

直接绘制出仓库中最新的三角形对象

void drawDot(int x, int y)

在视窗坐标系中绘制一个点

void DrawLine(int x0, int x1, int y0, int y1)

各个方向均可使用,绘制一条直线

void drawLine0(int x0, int x1, int y0, int y1)

绘制直线函数,决定调用哪一个直线函数

void drawLine1(int x0, int x1, int y0, int y1, bool xy_interchange)

绘制一号区域和五号区域的直线

void drawLine2(int x0, int x1, int y0, int y1, bool xy_interchange)

绘制二号区域和六号区域的直线

void drawLine3(int x0, int x1, int y0, int y1, bool xy_interchange)

绘制三号区域和七号区域的直线

void drawLine4(int x0, int x1, int y0, int y1, bool xy_interchange)

绘制四号区域和八号区域的直线

void ReadFile(bool& IsExit)

从文件中批量读取指令

void DrawFrame()

绘制投影区域的边框

bool DotToDraw(const int& x, const int& y)

决定这个点是否绘制

void redraw()

重绘整个视窗,恢复显示的时候调用

void FillPolygon(int PointVectorSize, string object)

填充函数,根据object决定填充颜色。方式是通过遍历一个多边形存储的边,以y轴从上往下扫描边的两个端点,然后水平绘制一条直线。颜色的选择是从颜色库中轮询选择。

void DrawHorizontalLine(int x0, int x1, int y)

绘制一条水平直线

template void printMatrix(T matrix, int row, int col, string str)*

打印矩阵函数,debug时使用

vector( pair(int, int) ) ClipPolygon(const vector( pair(int, int) )& PointVector, string boundary)

多边形剪切函数,boundary指定剪切方向。容器中存储一个多边形的各个点,如果需要被裁剪,这容器中的点会变成裁剪边界的点。

bool cmp_pair(const pair(int, int)& a, const pair(int, int)& b)

比较函数,对点进行排序时调用

void myKeyboard(unsigned char key, int x, int y)

键盘回调函数

运行效果











完整代码

  1. #include <iostream>
  2. #include <fstream>
  3. #include <stdlib.h>
  4. #include <cmath>
  5. #include <string>
  6. #include <vector>
  7. #include <algorithm>
  8. #include "glut.h"
  9. using namespace std;
  10. #pragma region Data
  11. #define CLIP_METHOD 0
  12. // 0 for compile to clip polygon
  13. // 1 for compile to clip dot
  14. #define FILL_ENABLE !CLIP_METHOD&&1
  15. // only enalbe it when CLIP_METHOD is 0
  16. #define READ_MULTI_FILE 1
  17. #define CTM_SIZE 3
  18. #define MAX_SQUARE 20
  19. #define MAX_TRIANGLE 20
  20. #define MAX_LINE 200
  21. // MAX_LINE >= MAX_SQUARE * 4 + MAX_TRIANGLE * 3 + MAX_VIEW_IN_FIN * 4
  22. #define PI 3.14159265
  23. //const float PI = acos(-1);
  24. #define MAX_COLOR 8
  25. #define MAX_FILL_LINE 1000
  26. #define MAX_FILE 6
  27. struct line_structure{
  28. int x0, x1, y0, y1;
  29. };
  30. struct line_data{
  31. int x, y;
  32. line_data* next;
  33. line_data(int _x = -1, int _y = -1){
  34. x = _x;
  35. y = _y;
  36. next = NULL;
  37. }
  38. };
  39. struct point{
  40. float x, y;
  41. float z;
  42. };
  43. struct square_struct{
  44. point right_up;
  45. point left_up;
  46. point left_down;
  47. point right_down;
  48. }SquareStore[MAX_SQUARE];
  49. struct triangle_struct{
  50. point first;
  51. point second;
  52. point third;
  53. }TriangleStore[MAX_TRIANGLE];
  54. struct fill_line_struct{
  55. int x0, x1, y;
  56. fill_line_struct(int _x0, int _x1, int _y){
  57. x0 = _x0;
  58. x1 = _x1;
  59. y = _y;
  60. }
  61. };
  62. struct frame_struct{
  63. int vxl, vxr, vyb, vyt;
  64. frame_struct(int _vxl, int _vxr, int _vyb, int _vyt){
  65. vxl = _vxl;
  66. vxr = _vxr;
  67. vyb = _vyb;
  68. vyt = _vyt;
  69. }
  70. };
  71. vector<frame_struct> frame_store;
  72. struct color_struct{
  73. float r, g, b;
  74. color_struct(float _r = 0, float _g = 0, float _b = 0){
  75. r = _r;
  76. g = _g;
  77. b = _b;
  78. }
  79. };
  80. const color_struct Color[MAX_COLOR] = { { 0.0, 0.0, 0.0 }, // black for default
  81. { 0.0, 0.0, 1.0 }, // blue for triangle
  82. { 1.0, 0.0, 0.0 }, // red for square
  83. { 1.0, 0.5, 0.0 }, // orange for square
  84. { 1.0, 1.0, 0.0 }, // yellow for square
  85. { 0.0, 1.0, 0.0 }, // green for square
  86. { 0.0, 1.0, 1.0 }, // light blue for square
  87. { 1.0, 0.0, 1.0 } }; // purple for square
  88. int color_index = 2;
  89. string file_list[MAX_FILE] = { "hw2E.in", "hw2B.in", "hw2A.in", "hw2D.in", "hw2F.in", "hw2C.in" };
  90. int cur_file = 0;
  91. int height, width;
  92. int Frame[4];
  93. /*
  94. * Frame[0] = vxl;
  95. * Frame[1] = vxr;
  96. * Frame[2] = vyb;
  97. * Frame[3] = vyt;
  98. */
  99. float current_matrix[CTM_SIZE][CTM_SIZE];
  100. int WVM[CTM_SIZE][CTM_SIZE] = { { 15, 0, 100 }, { 0, 15, 100 }, { 0, 0, 1 } };
  101. line_structure line[MAX_LINE];
  102. line_data line_start[MAX_LINE];
  103. int line_color[MAX_LINE];
  104. line_data* lcp;
  105. line_data* temp;
  106. vector<fill_line_struct> fill_line;
  107. vector<int> fill_line_color;
  108. int num_square = 0;
  109. int num_triangle = 0;
  110. int num_line = 0;
  111. int num_fill_line = 0;
  112. const float default_r = 255.0;
  113. const float default_g = 255.0;
  114. const float default_b = 255.0;
  115. float r = 1.0;
  116. float g = 0.0;
  117. float b = 0.0;
  118. float alpha = 0.0;
  119. bool IsExit = false;
  120. enum ScanFlag{ odd, even };
  121. #pragma endregion
  122. #pragma region FunctionDefinition
  123. void initial();
  124. void scale(float sx, float sy);
  125. void rotate(float degree);
  126. void translate(float tx, float ty);
  127. void reset();
  128. void square();
  129. void triangle();
  130. void view(float wxl, float wxr, float wyb, float wyt, float vxl, float vxr, float vyb, float vyt);
  131. void clearData();
  132. void clearScreen();
  133. void displayFunc(void);
  134. void ReadInput(bool& IsExit);
  135. void Matrix_Multi_Matrix(float a[][CTM_SIZE], float b[][CTM_SIZE], float c[][CTM_SIZE]);
  136. void update_CurMatrix(float resource[][CTM_SIZE], float destination[][CTM_SIZE]);
  137. void update_WVM(int sx, int sy, int tx, int ty);
  138. void AddtoSquareStore(float matrix[][CTM_SIZE]);
  139. void AddtoTriangleStore(float matrix[][CTM_SIZE]);
  140. void GetFromSquareStore(int index, float destination[][CTM_SIZE]);
  141. void GetFromTriangleStore(int index, float destination[][CTM_SIZE]);
  142. void DrawSquareDirect(int matrix[][CTM_SIZE]);
  143. void DrawSquareByindex(int index);
  144. void DrawTriangleDirect(int matrix[][CTM_SIZE]);
  145. void drawDot(int x, int y);
  146. void DrawLine(int x0, int x1, int y0, int y1);
  147. void drawLine0(int x0, int x1, int y0, int y1);
  148. void drawLine1(int x0, int x1, int y0, int y1, bool xy_interchange);
  149. void drawLine2(int x0, int x1, int y0, int y1, bool xy_interchange);
  150. void drawLine3(int x0, int x1, int y0, int y1, bool xy_interchange);
  151. void drawLine4(int x0, int x1, int y0, int y1, bool xy_interchange);
  152. void ReadFile(bool& IsExit);
  153. void DrawFrame();
  154. bool DotToDraw(const int& x, const int& y);
  155. void redraw();
  156. vector<pair<int, int>> ClipPolygon(const vector<pair<int, int>>& PointVector, string boundary);
  157. void FillPolygon(int PointVectorSize, string object);
  158. bool cmp_pair(const pair<int, int>& a, const pair<int, int>& b);
  159. void DrawHorizontalLine(int x0, int x1, int y);
  160. template<class T> void printMatrix(T* matrix, int row, int col, string str);
  161. void myKeyboard(unsigned char key, int x, int y);
  162. #pragma endregion
  163. #pragma region FunctionImplement
  164. template<class T>
  165. void printMatrix(T* matrix, int row, int col, string str){
  166. cout << endl;
  167. cout << str << " : " << endl;
  168. for (int i = 0; i < row; i++){
  169. cout << "\t[ ";
  170. for (int j = 0; j < col; j++){
  171. cout << matrix[i][j] << ", ";
  172. }
  173. cout << " ]" << endl;
  174. }
  175. }
  176. // Set current transform matrix to Identity
  177. void reset(){
  178. for (int i = 0; i < CTM_SIZE; i++){
  179. for (int j = 0; j < CTM_SIZE; j++){
  180. if (i == j)
  181. current_matrix[i][j] = 1.0;
  182. else
  183. current_matrix[i][j] = 0;
  184. }
  185. }
  186. printMatrix(current_matrix, 3, 3, "current_matrix");
  187. }
  188. // Multiply current matrix by scaling matrix
  189. void scale(float sx, float sy){
  190. float scaling_matrix[CTM_SIZE][CTM_SIZE]
  191. = { { sx, 0, 0 },
  192. { 0, sy, 0 },
  193. { 0, 0, 1 } };
  194. float new_matrix[CTM_SIZE][CTM_SIZE];
  195. Matrix_Multi_Matrix(scaling_matrix, current_matrix, new_matrix);
  196. update_CurMatrix(new_matrix, current_matrix);
  197. printMatrix(current_matrix, 3, 3, "current_matrix");
  198. }
  199. // Multiply current matrix by rotation matrix
  200. void rotate(float degree){
  201. float rad = degree * PI / 180.0;
  202. float rotation_matrix[CTM_SIZE][CTM_SIZE]
  203. = { { cos(rad), -sin(rad), 0 },
  204. { sin(rad), cos(rad), 0 },
  205. { 0, 0, 1 } };
  206. //printMatrix(rotation_matrix, 3, 3, "rotation_matrix");
  207. float new_matrix[CTM_SIZE][CTM_SIZE];
  208. Matrix_Multi_Matrix(rotation_matrix, current_matrix, new_matrix);
  209. update_CurMatrix(new_matrix, current_matrix);
  210. printMatrix(current_matrix, 3, 3, "current_matrix");
  211. }
  212. // Multiply current matrix by translation matrix
  213. void translate(float tx, float ty){
  214. float translate_matrix[CTM_SIZE][CTM_SIZE]
  215. = { { 1, 0, tx },
  216. { 0, 1, ty },
  217. { 0, 0, 1 } };
  218. float new_matrix[CTM_SIZE][CTM_SIZE];
  219. Matrix_Multi_Matrix(translate_matrix, current_matrix, new_matrix);
  220. update_CurMatrix(new_matrix, current_matrix);
  221. printMatrix(current_matrix, 3, 3, "current_matrix");
  222. }
  223. // Two 3 * 3 Matrixs Multiplication
  224. void Matrix_Multi_Matrix(float a[][CTM_SIZE], float b[][CTM_SIZE], float c[][CTM_SIZE]){
  225. float multi_sum = 0;
  226. for (int i = 0; i < CTM_SIZE; i++){
  227. for (int j = 0; j < CTM_SIZE; j++){
  228. multi_sum = 0;
  229. for (int k = 0; k < CTM_SIZE; k++){
  230. multi_sum += b[k][j] * a[i][k];
  231. }
  232. c[i][j] = multi_sum;
  233. }
  234. }
  235. }
  236. // update the current_matrix
  237. void update_CurMatrix(float resource[][CTM_SIZE], float destination[][CTM_SIZE]){
  238. for (int i = 0; i < CTM_SIZE; i++){
  239. for (int j = 0; j < CTM_SIZE; j++){
  240. destination[i][j] = resource[i][j];
  241. }
  242. }
  243. }
  244. // Create a square and save it
  245. void square(){
  246. float org_square[4][CTM_SIZE]
  247. = { { 1, 1, 1 },
  248. { -1, 1, 1 },
  249. { -1, -1, 1 },
  250. { 1, -1, 1 } };
  251. float WorldCoord_square[4][CTM_SIZE];
  252. float multi_sum = 0;
  253. for (int i = 0; i < 4; i++){
  254. for (int j = 0; j < CTM_SIZE; j++){
  255. multi_sum = 0;
  256. for (int k = 0; k < CTM_SIZE; k++){
  257. multi_sum += org_square[i][k] * current_matrix[j][k];
  258. }
  259. WorldCoord_square[i][j] = multi_sum;
  260. }
  261. }
  262. AddtoSquareStore(WorldCoord_square);
  263. num_square++;
  264. printMatrix(WorldCoord_square, 4, 3, "WorldCoord_square");
  265. }
  266. // Add a new square to SquareStore
  267. void AddtoSquareStore(float matrix[][CTM_SIZE]){
  268. SquareStore[num_square].right_up.x = matrix[0][0];
  269. SquareStore[num_square].right_up.y = matrix[0][1];
  270. SquareStore[num_square].right_up.z = matrix[0][2];
  271. SquareStore[num_square].left_up.x = matrix[1][0];
  272. SquareStore[num_square].left_up.y = matrix[1][1];
  273. SquareStore[num_square].left_up.z = matrix[1][2];
  274. SquareStore[num_square].left_down.x = matrix[2][0];
  275. SquareStore[num_square].left_down.y = matrix[2][1];
  276. SquareStore[num_square].left_down.z = matrix[2][2];
  277. SquareStore[num_square].right_down.x = matrix[3][0];
  278. SquareStore[num_square].right_down.y = matrix[3][1];
  279. SquareStore[num_square].right_down.z = matrix[3][2];
  280. }
  281. // Show all objects on the screen
  282. void view(float wxl, float wxr, float wyb, float wyt, float vxl, float vxr, float vyb, float vyt){
  283. int scaling_x = (int)((vxr - vxl) / (wxr - wxl));
  284. int scaling_y = (int)((vyt - vyb) / (wyt - wyb));
  285. int shift_x = (int)(vxl - scaling_x * wxl);
  286. int shift_y = (int)(vyb - scaling_y * wyb);
  287. update_WVM(scaling_x, scaling_y, shift_x, shift_y);
  288. printMatrix(WVM, 3, 3, "WVM");
  289. float WorldCoord_matrix[4][CTM_SIZE];
  290. int ScreenCoord_matrix[4][CTM_SIZE];
  291. //Set Frame
  292. Frame[0] = (int)vxl;
  293. Frame[1] = (int)vxr;
  294. Frame[2] = (int)vyb;
  295. Frame[3] = (int)vyt;
  296. frame_store.push_back(frame_struct(Frame[0], Frame[1], Frame[2], Frame[3]));
  297. //Draw Square
  298. r = 1.0; g = 0.0; b = 0.0;
  299. float multi_sum = 0;
  300. int count = 0;
  301. while (count < num_square){
  302. GetFromSquareStore(count, WorldCoord_matrix);
  303. printMatrix(WorldCoord_matrix, 4, 3, "WorldCoord_matrix");
  304. for (int i = 0; i < 4; i++){
  305. for (int j = 0; j < CTM_SIZE; j++){
  306. multi_sum = 0;
  307. for (int k = 0; k < CTM_SIZE; k++){
  308. multi_sum += WorldCoord_matrix[i][k] * WVM[j][k];
  309. }
  310. ScreenCoord_matrix[i][j] = (int)multi_sum;
  311. }
  312. }
  313. printMatrix(ScreenCoord_matrix, 4, 3, "ScreenCoord_matrix");
  314. DrawSquareDirect(ScreenCoord_matrix);
  315. //void DrawSquareByindex(count);
  316. count++;
  317. }
  318. //Draw Triangle
  319. r = 0.0; g = 0.0; b = 1.0;
  320. count = 0;
  321. while (count < num_triangle){
  322. GetFromTriangleStore(count, WorldCoord_matrix);
  323. printMatrix(WorldCoord_matrix, 3, 3, "WorldCoord_matrix");
  324. for (int i = 0; i < 3; i++){
  325. for (int j = 0; j < CTM_SIZE; j++){
  326. multi_sum = 0;
  327. for (int k = 0; k < CTM_SIZE; k++){
  328. multi_sum += WorldCoord_matrix[i][k] * WVM[j][k];
  329. }
  330. ScreenCoord_matrix[i][j] = (int)multi_sum;
  331. }
  332. }
  333. printMatrix(ScreenCoord_matrix, 3, 3, "ScreenCoord_matrix");
  334. DrawTriangleDirect(ScreenCoord_matrix);
  335. count++;
  336. }
  337. // Draw Frame
  338. r = 0.0; g = 0.0; b = 0.0;
  339. DrawFrame();
  340. return;
  341. }
  342. // Draw a frame
  343. void DrawFrame(){
  344. temp = &line_start[num_line];
  345. line_color[num_line] = 0;
  346. DrawLine(Frame[0], Frame[1], Frame[2], Frame[2]);
  347. num_line++;
  348. temp = &line_start[num_line];
  349. line_color[num_line] = 0;
  350. DrawLine(Frame[1], Frame[1], Frame[2], Frame[3]);
  351. num_line++;
  352. temp = &line_start[num_line];
  353. line_color[num_line] = 0;
  354. DrawLine(Frame[1], Frame[0], Frame[3], Frame[3]);
  355. num_line++;
  356. temp = &line_start[num_line];
  357. line_color[num_line] = 0;
  358. DrawLine(Frame[0], Frame[0], Frame[3], Frame[2]);
  359. num_line++;
  360. }
  361. // Draw a square
  362. void DrawSquareDirect(int matrix[][CTM_SIZE]){
  363. #if CLIP_METHOD
  364. temp = &line_start[num_line];
  365. line_color[num_line] = 2;
  366. DrawLine(matrix[1][0], matrix[0][0], matrix[1][1], matrix[0][1]);
  367. num_line++;
  368. temp = &line_start[num_line];
  369. line_color[num_line] = 2;
  370. DrawLine(matrix[0][0], matrix[3][0], matrix[0][1], matrix[3][1]);
  371. num_line++;
  372. temp = &line_start[num_line];
  373. line_color[num_line] = 2;
  374. DrawLine(matrix[3][0], matrix[2][0], matrix[3][1], matrix[2][1]);
  375. num_line++;
  376. temp = &line_start[num_line];
  377. line_color[num_line] = 2;
  378. DrawLine(matrix[2][0], matrix[1][0], matrix[2][1], matrix[1][1]);
  379. num_line++;
  380. #else
  381. vector<pair<int, int>> PointVector;
  382. PointVector.push_back(make_pair(matrix[1][0], matrix[1][1]));
  383. PointVector.push_back(make_pair(matrix[0][0], matrix[0][1]));
  384. PointVector.push_back(make_pair(matrix[3][0], matrix[3][1]));
  385. PointVector.push_back(make_pair(matrix[2][0], matrix[2][1]));
  386. PointVector.push_back(make_pair(matrix[1][0], matrix[1][1]));
  387. PointVector = ClipPolygon(PointVector, "left");
  388. PointVector = ClipPolygon(PointVector, "bottom");
  389. PointVector = ClipPolygon(PointVector, "right");
  390. PointVector = ClipPolygon(PointVector, "top");
  391. for(int i = 1; i < PointVector.size(); i++){
  392. temp = &line_start[num_line];
  393. line_color[num_line] = 2;
  394. DrawLine(PointVector[i - 1].first, PointVector[i].first, PointVector[i - 1].second, PointVector[i].second);
  395. num_line++;
  396. }
  397. #if FILL_ENABLE
  398. FillPolygon(PointVector.size(), "square");
  399. #endif
  400. #endif
  401. }
  402. // Clip the Polygon by one side
  403. vector<pair<int, int>> ClipPolygon(const vector<pair<int, int>>& PointVector, string boundary){
  404. pair<int, int> PointS, PointP;
  405. vector<pair<int, int>> NewPointVector;
  406. // 1-st pass
  407. if(boundary == "left"){
  408. for(int i = 1; i < PointVector.size(); i++){
  409. PointS = PointVector[i - 1];
  410. PointP = PointVector[i];
  411. // Rule 1
  412. if(PointS.first >= Frame[0] && PointP.first >= Frame[0]){
  413. NewPointVector.push_back(PointP);
  414. }
  415. // Rule 2
  416. else if(PointS.first >= Frame[0] && PointP.first < Frame[0]){
  417. int left_y = (int)((PointS.second - PointP.second)*1.0 / (PointS.first - PointP.first)*(Frame[0] - PointP.first) + PointP.second);
  418. NewPointVector.push_back(make_pair(Frame[0], left_y));
  419. }
  420. // Rule 4
  421. else if(PointS.first < Frame[0] && PointP.first >= Frame[0]){
  422. int left_y = (int)((PointS.second - PointP.second)*1.0 / (PointS.first - PointP.first)*(Frame[0] - PointP.first) + PointP.second);
  423. NewPointVector.push_back(make_pair(Frame[0], left_y));
  424. NewPointVector.push_back(PointP);
  425. }
  426. }
  427. if(!NewPointVector.empty())
  428. NewPointVector.push_back(NewPointVector[0]);
  429. }
  430. // 2-nd pass
  431. else if(boundary == "bottom"){
  432. for(int i = 1; i < PointVector.size(); i++){
  433. PointS = PointVector[i - 1];
  434. PointP = PointVector[i];
  435. // Rule 1
  436. if(PointS.second >= Frame[2] && PointP.second >= Frame[2]){
  437. NewPointVector.push_back(PointP);
  438. }
  439. // Rule 2
  440. else if(PointS.second >= Frame[2] && PointP.second < Frame[2]){
  441. int bottom_x = (int)((PointS.first - PointP.first)*1.0 / (PointS.second - PointP.second)*(Frame[2] - PointP.second) + PointP.first);
  442. NewPointVector.push_back(make_pair(bottom_x, Frame[2]));
  443. }
  444. // Rule 4
  445. else if(PointS.second < Frame[2] && PointP.second >= Frame[2]){
  446. int bottom_x = (int)((PointS.first - PointP.first)*1.0 / (PointS.second - PointP.second)*(Frame[2] - PointP.second) + PointP.first);
  447. NewPointVector.push_back(make_pair(bottom_x, Frame[2]));
  448. NewPointVector.push_back(PointP);
  449. }
  450. }
  451. if(!NewPointVector.empty())
  452. NewPointVector.push_back(NewPointVector[0]);
  453. }
  454. // 3-rd pass
  455. else if(boundary == "right"){
  456. for(int i = 1; i < PointVector.size(); i++){
  457. PointS = PointVector[i - 1];
  458. PointP = PointVector[i];
  459. // Rule 1
  460. if(PointS.first <= Frame[1] && PointP.first <= Frame[1]){
  461. NewPointVector.push_back(PointP);
  462. }
  463. // Rule 2
  464. else if(PointS.first <= Frame[1] && PointP.first > Frame[1]){
  465. int right_y = (int)((PointS.second - PointP.second)*1.0 / (PointS.first - PointP.first)*(Frame[1] - PointP.first) + PointP.second);
  466. NewPointVector.push_back(make_pair(Frame[1], right_y));
  467. }
  468. // Rule 4
  469. else if(PointS.first > Frame[1] && PointP.first <= Frame[1]){
  470. int right_y = (int)((PointS.second - PointP.second)*1.0 / (PointS.first - PointP.first)*(Frame[1] - PointP.first) + PointP.second);
  471. NewPointVector.push_back(make_pair(Frame[1], right_y));
  472. NewPointVector.push_back(PointP);
  473. }
  474. }
  475. if(!NewPointVector.empty())
  476. NewPointVector.push_back(NewPointVector[0]);
  477. }
  478. // 4-th pass
  479. else if(boundary == "top"){
  480. for(int i = 1; i < PointVector.size(); i++){
  481. PointS = PointVector[i - 1];
  482. PointP = PointVector[i];
  483. // Rule 1
  484. if(PointS.second <= Frame[3] && PointP.second <= Frame[3]){
  485. NewPointVector.push_back(PointP);
  486. }
  487. // Rule 2
  488. else if(PointS.second <= Frame[3] && PointP.second > Frame[3]){
  489. int top_x = (int)((PointS.first - PointP.first)*1.0 / (PointS.second - PointP.second)*(Frame[3] - PointP.second) + PointP.first);
  490. NewPointVector.push_back(make_pair(top_x, Frame[3]));
  491. }
  492. // Rule 4
  493. else if(PointS.second > Frame[3] && PointP.second <= Frame[3]){
  494. int top_x = (int)((PointS.first - PointP.first)*1.0 / (PointS.second - PointP.second)*(Frame[3] - PointP.second) + PointP.first);
  495. NewPointVector.push_back(make_pair(top_x, Frame[3]));
  496. NewPointVector.push_back(PointP);
  497. }
  498. }
  499. if(!NewPointVector.empty())
  500. NewPointVector.push_back(NewPointVector[0]);
  501. }
  502. return NewPointVector;
  503. }
  504. // Draw a square
  505. void DrawSquareByindex(int index){
  506. temp = &line_start[num_line];
  507. DrawLine(SquareStore[index].left_up.x, SquareStore[index].right_up.x,
  508. SquareStore[index].left_up.y, SquareStore[index].right_up.y);
  509. num_line++;
  510. temp = &line_start[num_line];
  511. DrawLine(SquareStore[index].right_up.x, SquareStore[index].right_down.x,
  512. SquareStore[index].right_up.y, SquareStore[index].right_down.y);
  513. num_line++;
  514. temp = &line_start[num_line];
  515. DrawLine(SquareStore[index].right_down.x, SquareStore[index].left_down.x,
  516. SquareStore[index].right_down.y, SquareStore[index].left_down.y);
  517. num_line++;
  518. temp = &line_start[num_line];
  519. DrawLine(SquareStore[index].left_down.x, SquareStore[index].left_up.x,
  520. SquareStore[index].left_down.y, SquareStore[index].left_up.y);
  521. num_line++;
  522. }
  523. // Draw a triangle
  524. void DrawTriangleDirect(int matrix[][CTM_SIZE]){
  525. #if CLIP_METHOD
  526. temp = &line_start[num_line];
  527. line_color[num_line] = 1;
  528. DrawLine(matrix[0][0], matrix[1][0], matrix[0][1], matrix[1][1]);
  529. num_line++;
  530. temp = &line_start[num_line];
  531. line_color[num_line] = 1;
  532. DrawLine(matrix[1][0], matrix[2][0], matrix[1][1], matrix[2][1]);
  533. num_line++;
  534. temp = &line_start[num_line];
  535. line_color[num_line] = 1;
  536. DrawLine(matrix[2][0], matrix[0][0], matrix[2][1], matrix[0][1]);
  537. num_line++;
  538. #else
  539. vector<pair<int, int>> PointVector;
  540. PointVector.push_back(make_pair(matrix[0][0], matrix[0][1]));
  541. PointVector.push_back(make_pair(matrix[1][0], matrix[1][1]));
  542. PointVector.push_back(make_pair(matrix[2][0], matrix[2][1]));
  543. PointVector.push_back(make_pair(matrix[0][0], matrix[0][1]));
  544. PointVector = ClipPolygon(PointVector, "left");
  545. PointVector = ClipPolygon(PointVector, "bottom");
  546. PointVector = ClipPolygon(PointVector, "right");
  547. PointVector = ClipPolygon(PointVector, "top");
  548. for(int i = 1; i < PointVector.size(); i++){
  549. temp = &line_start[num_line];
  550. line_color[num_line] = 1;
  551. DrawLine(PointVector[i - 1].first, PointVector[i].first, PointVector[i - 1].second, PointVector[i].second);
  552. num_line++;
  553. }
  554. #if FILL_ENABLE
  555. FillPolygon(PointVector.size(), "triangle");
  556. #endif
  557. #endif
  558. }
  559. // Judge wheather to draw dot
  560. bool DotToDraw(const int& x, const int& y){
  561. if (x >= Frame[0] && x <= Frame[1] && y >= Frame[2] && y <= Frame[3]){
  562. return true;
  563. }
  564. return false;
  565. }
  566. // draw a dot at location with integer coordinates (x,y)
  567. void drawDot(int x, int y){
  568. glBegin(GL_POINTS);
  569. // set the color of dot
  570. glColor3f(r, g, b);
  571. // invert height because the opengl origin is at top-left instead of bottom-left
  572. //glVertex2i(x, height - y);
  573. glVertex2i(x, y);
  574. glEnd();
  575. }
  576. // Draw a line in any direction
  577. void DrawLine(int x0, int x1, int y0, int y1) {
  578. int max_dis = abs(x1 - x0);
  579. max_dis = max_dis > abs(y0 - y1) ? max_dis : abs(y0 - y1);
  580. lcp = temp;
  581. if (max_dis != 0) {
  582. for (int i = 0; i < max_dis; ++i) {
  583. int new_p_x = x0 + i * (x1 - x0) / max_dis;
  584. int new_p_y = y0 + i * (y1 - y0) / max_dis;
  585. #if CLIP_METHOD
  586. if (DotToDraw(new_p_x, new_p_y)){
  587. drawDot(new_p_x, new_p_y);
  588. lcp->next = new line_data(new_p_x, new_p_y);
  589. lcp = lcp->next;
  590. }
  591. #else
  592. drawDot(new_p_x, new_p_y);
  593. lcp->next = new line_data(new_p_x, new_p_y);
  594. lcp = lcp->next;
  595. #endif
  596. }
  597. }
  598. glFlush();
  599. return;
  600. }
  601. // Judge to call which drawLine function to draw a line
  602. void drawLine0(int x0, int x1, int y0, int y1){
  603. int dx = x1 - x0;
  604. int dy = y1 - y0;
  605. if(dx >= 0 && dy > 0 && abs(dx) < abs(dy)){
  606. drawLine1(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, false);
  607. }
  608. else if(dx > 0 && dy >= 0 && abs(dx) >= abs(dy)){
  609. drawLine2(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, false);
  610. }
  611. else if(dx > 0 && dy <= 0 && abs(dx) > abs(dy)){
  612. drawLine3(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, false);
  613. }
  614. else if(dx >= 0 && dy < 0 && abs(dx) <= abs(dy)){
  615. drawLine4(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, false);
  616. }
  617. else if(dx <= 0 && dy < 0 && abs(dx) < abs(dy)){
  618. drawLine1(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, true);
  619. }
  620. else if(dx < 0 && dy <= 0 && abs(dx) >= abs(dy)){
  621. drawLine2(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, true);
  622. }
  623. else if(dx < 0 && dy >= 0 && abs(dx) > abs(dy)){
  624. drawLine3(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, true);
  625. }
  626. else{
  627. drawLine4(line[num_line].x0, line[num_line].x1, line[num_line].y0, line[num_line].y1, true);
  628. }
  629. num_line++;
  630. }
  631. // Draw line for dx>0 and dy>0
  632. void drawLine1(int x0, int x1, int y0, int y1, bool xy_interchange){
  633. if(xy_interchange){
  634. int change = x0; x0 = x1; x1 = change;
  635. change = y0; y0 = y1; y1 = change;
  636. }
  637. int x = x1;
  638. int y = y1;
  639. int a = y1 - y0;
  640. int b = -(x1 - x0);
  641. int d = -a - 2 * b;
  642. int IncE = -2 * b;
  643. int IncNE = -2 * a - 2 * b;
  644. line_start[num_line] = line_data(x, y);
  645. lcp = &line_start[num_line];
  646. while(y >= y0){
  647. lcp->next = new line_data(x, y);
  648. lcp = lcp->next;
  649. drawDot(x, y);
  650. if(d <= 0){
  651. y--;
  652. d += IncE;
  653. }
  654. else{
  655. x--;
  656. y--;
  657. d += IncNE;
  658. }
  659. }
  660. glFlush();
  661. return;
  662. }
  663. // Draw line for dx>0 and dy<0
  664. void drawLine2(int x0, int x1, int y0, int y1, bool xy_interchange){
  665. if(xy_interchange){
  666. int change = x0; x0 = x1; x1 = change;
  667. change = y0; y0 = y1; y1 = change;
  668. }
  669. int x = x0;
  670. int y = y0;
  671. int a = y1 - y0;
  672. int b = -(x1 - x0);
  673. int d = 2 * a + b;
  674. int IncE = 2 * a;
  675. int IncNE = 2 * a + 2 * b;
  676. line_start[num_line] = line_data(x, y);
  677. lcp = &line_start[num_line];
  678. while(x <= x1){
  679. lcp->next = new line_data(x, y);
  680. lcp = lcp->next;
  681. drawDot(x, y);
  682. if(d <= 0){
  683. x++;
  684. d += IncE;
  685. }
  686. else{
  687. x++;
  688. y++;
  689. d += IncNE;
  690. }
  691. }
  692. glFlush();
  693. return;
  694. }
  695. // Draw line for dx<0 and dy>0
  696. void drawLine3(int x0, int x1, int y0, int y1, bool xy_interchange){
  697. if(xy_interchange){
  698. int change = x0; x0 = x1; x1 = change;
  699. change = y0; y0 = y1; y1 = change;
  700. }
  701. int x = x0;
  702. int y = y0;
  703. int a = y0 - y1;
  704. int b = -(x1 - x0);
  705. int d = 2 * a + b;
  706. int IncE = 2 * a;
  707. int IncNE = 2 * a + 2 * b;
  708. line_start[num_line] = line_data(x, y);
  709. lcp = &line_start[num_line];
  710. while(x <= x1){
  711. lcp->next = new line_data(x, y);
  712. lcp = lcp->next;
  713. drawDot(x, y);
  714. if(d <= 0){
  715. x++;
  716. d += IncE;
  717. }
  718. else{
  719. x++;
  720. y--;
  721. d += IncNE;
  722. }
  723. }
  724. glFlush();
  725. return;
  726. }
  727. // Draw line for dx<0 and dy<0
  728. void drawLine4(int x0, int x1, int y0, int y1, bool xy_interchange){
  729. if(xy_interchange){
  730. int change = x0; x0 = x1; x1 = change;
  731. change = y0; y0 = y1; y1 = change;
  732. }
  733. int x = x0;
  734. int y = y0;
  735. int a = y1 - y0;
  736. int b = -(x1 - x0);
  737. int d = a - 2 * b;
  738. int IncE = -2 * b;
  739. int IncNE = 2 * a - 2 * b;
  740. line_start[num_line] = line_data(x, y);
  741. lcp = &line_start[num_line];
  742. while(y >= y1){
  743. lcp->next = new line_data(x, y);
  744. lcp = lcp->next;
  745. drawDot(x, y);
  746. if(d <= 0){
  747. y--;
  748. d += IncE;
  749. }
  750. else{
  751. x++;
  752. y--;
  753. d += IncNE;
  754. }
  755. }
  756. glFlush();
  757. return;
  758. }
  759. // Draw a horizontal line
  760. void DrawHorizontalLine(int x0, int x1, int y){
  761. for(int x = x0; x <= x1; x++){
  762. drawDot(x, y);
  763. }
  764. glFlush();
  765. return;
  766. }
  767. // Get matrix from SquareStore
  768. void GetFromSquareStore(int index, float destination[][CTM_SIZE]){
  769. destination[0][0] = SquareStore[index].right_up.x;
  770. destination[0][1] = SquareStore[index].right_up.y;
  771. destination[0][2] = SquareStore[index].right_up.z;
  772. destination[1][0] = SquareStore[index].left_up.x;
  773. destination[1][1] = SquareStore[index].left_up.y;
  774. destination[1][2] = SquareStore[index].left_up.z;
  775. destination[2][0] = SquareStore[index].left_down.x;
  776. destination[2][1] = SquareStore[index].left_down.y;
  777. destination[2][2] = SquareStore[index].left_down.z;
  778. destination[3][0] = SquareStore[index].right_down.x;
  779. destination[3][1] = SquareStore[index].right_down.y;
  780. destination[3][2] = SquareStore[index].right_down.z;
  781. }
  782. // update the WVM matrix
  783. void update_WVM(int sx, int sy, int tx, int ty){
  784. WVM[0][0] = sx;
  785. WVM[1][1] = sy;
  786. WVM[0][2] = tx;
  787. WVM[1][2] = ty;
  788. WVM[2][2] = 1;
  789. WVM[0][1] = WVM[1][0] = WVM[2][0] = WVM[2][1] = 0;
  790. }
  791. // Draw a triangle and Save it
  792. void triangle(){
  793. float org_triangle[3][CTM_SIZE]
  794. = { { 0, 1, 1 },
  795. { -1, -1, 1 },
  796. { 1, -1, 1 } };
  797. float WorldCoord_square[3][CTM_SIZE];
  798. float multi_sum = 0;
  799. for (int i = 0; i < 3; i++){
  800. for (int j = 0; j < CTM_SIZE; j++){
  801. multi_sum = 0;
  802. for (int k = 0; k < CTM_SIZE; k++){
  803. multi_sum += org_triangle[i][k] * current_matrix[j][k];
  804. }
  805. WorldCoord_square[i][j] = multi_sum;
  806. }
  807. }
  808. AddtoTriangleStore(WorldCoord_square);
  809. num_triangle++;
  810. printMatrix(WorldCoord_square, 3, 3, "WorldCoord_square");
  811. }
  812. // Add a new triangle to TriangleStore
  813. void AddtoTriangleStore(float matrix[][CTM_SIZE]){
  814. TriangleStore[num_triangle].first.x = matrix[0][0];
  815. TriangleStore[num_triangle].first.y = matrix[0][1];
  816. TriangleStore[num_triangle].first.z = matrix[0][2];
  817. TriangleStore[num_triangle].second.x = matrix[1][0];
  818. TriangleStore[num_triangle].second.y = matrix[1][1];
  819. TriangleStore[num_triangle].second.z = matrix[1][2];
  820. TriangleStore[num_triangle].third.x = matrix[2][0];
  821. TriangleStore[num_triangle].third.y = matrix[2][1];
  822. TriangleStore[num_triangle].third.z = matrix[2][2];
  823. }
  824. // Get a triangle data from TriangleStore
  825. void GetFromTriangleStore(int index, float destination[][CTM_SIZE]){
  826. destination[0][0] = TriangleStore[index].first.x;
  827. destination[0][1] = TriangleStore[index].first.y;
  828. destination[0][2] = TriangleStore[index].first.z;
  829. destination[1][0] = TriangleStore[index].second.x;
  830. destination[1][1] = TriangleStore[index].second.y;
  831. destination[1][2] = TriangleStore[index].second.z;
  832. destination[2][0] = TriangleStore[index].third.x;
  833. destination[2][1] = TriangleStore[index].third.y;
  834. destination[2][2] = TriangleStore[index].third.z;
  835. }
  836. // Clear all the objects in store
  837. void clearData(){
  838. for(int i = 0; i < num_line; i++){
  839. if(line_start[i].next != NULL){
  840. lcp = line_start[i].next;
  841. while(lcp != NULL){
  842. temp = lcp;
  843. lcp = lcp->next;
  844. delete(temp);
  845. }
  846. }
  847. }
  848. num_line = 0;
  849. num_square = 0;
  850. num_triangle = 0;
  851. num_fill_line = 0;
  852. fill_line.clear();
  853. fill_line_color.clear();
  854. frame_store.clear();
  855. return;
  856. }
  857. // Clear the screen
  858. void clearScreen(){
  859. glClearColor(default_r, default_g, default_b, alpha);
  860. glClear(GL_COLOR_BUFFER_BIT);
  861. glFlush();
  862. }
  863. // Redraw all objects stored
  864. void redraw(){
  865. r = 0; g = 1.0; b = 1.0;
  866. int i;
  867. line_data* rd_line;
  868. for (i = 0; i < num_line; i++){
  869. rd_line = &line_start[i];
  870. r = Color[line_color[i]].r;
  871. g = Color[line_color[i]].g;
  872. b = Color[line_color[i]].b;
  873. while (rd_line != NULL){
  874. drawDot(rd_line->x, rd_line->y);
  875. rd_line = rd_line->next;
  876. }
  877. }
  878. for(i = 0; i < fill_line.size(); i++){
  879. r = Color[fill_line_color[i]].r;
  880. g = Color[fill_line_color[i]].g;
  881. b = Color[fill_line_color[i]].b;
  882. DrawHorizontalLine(fill_line[i].x0, fill_line[i].x1, fill_line[i].y);
  883. }
  884. r = 0.0; g = 0.0; b = 0.0;
  885. for(i = 0; i < frame_store.size(); i++){
  886. DrawLine(frame_store[i].vxl, frame_store[i].vxr, frame_store[i].vyb, frame_store[i].vyb);
  887. DrawLine(frame_store[i].vxr, frame_store[i].vxr, frame_store[i].vyb, frame_store[i].vyt);
  888. DrawLine(frame_store[i].vxr, frame_store[i].vxl, frame_store[i].vyt, frame_store[i].vyt);
  889. DrawLine(frame_store[i].vxl, frame_store[i].vxl, frame_store[i].vyt, frame_store[i].vyb);
  890. }
  891. glFlush();
  892. }
  893. // Read the command from std
  894. void ReadInput(bool& IsExit){
  895. float sx, sy, degree, tx, ty, wxl, wxr, wyb, wyt, vxl, vxr, vyt, vyb;
  896. string command, comment;
  897. cin >> command;
  898. if (command == "scale"){
  899. cin >> sx;
  900. cin >> sy;
  901. scale(sx, sy);
  902. cout << command << endl;
  903. }
  904. else if (command == "rotate"){
  905. cin >> degree;
  906. rotate(degree);
  907. cout << command << endl;
  908. }
  909. else if (command == "translate"){
  910. cin >> tx;
  911. cin >> ty;
  912. translate(tx, ty);
  913. cout << command << endl;
  914. }
  915. else if (command == "reset"){
  916. reset();
  917. cout << command << endl;
  918. }
  919. else if (command == "square"){
  920. square();
  921. cout << command << endl;
  922. }
  923. else if (command == "triangle"){
  924. triangle();
  925. cout << command << endl;
  926. }
  927. else if (command == "view"){
  928. cin >> wxl >> wxr >> wyb >> wyt >> vxl >> vxr >> vyb >> vyt;
  929. view(wxl, wxr, wyb, wyt, vxl, vxr, vyb, vyt);
  930. cout << command << endl;
  931. }
  932. else if (command == "clearData"){
  933. clearData();
  934. cout << command << endl;
  935. }
  936. else if (command == "clearScreen"){
  937. cout << "Screen is cleared" << endl;
  938. clearScreen();
  939. cout << command << endl;
  940. }
  941. else if (command == "end"){
  942. IsExit = true;
  943. cout << command << endl;
  944. exit(0);
  945. }
  946. else if (command == "#"){
  947. getline(cin, comment);
  948. }
  949. }
  950. // Read the command from file
  951. void ReadFile(bool& IsExit){
  952. #if READ_MULTI_FILE
  953. ifstream fin(file_list[cur_file]);
  954. #else
  955. ifstream fin("myhw2.in");
  956. #endif
  957. if (fin.is_open()){
  958. cout << "open the file successfully" << endl;
  959. }
  960. float sx, sy, degree, tx, ty, wxl, wxr, wyb, wyt, vxl, vxr, vyt, vyb;
  961. string command, comment;
  962. while (!fin.eof()){
  963. fin >> command;
  964. if (command == "scale"){
  965. fin >> sx;
  966. fin >> sy;
  967. scale(sx, sy);
  968. cout << command << endl;
  969. }
  970. else if (command == "rotate"){
  971. fin >> degree;
  972. rotate(degree);
  973. cout << command << endl;
  974. }
  975. else if (command == "translate"){
  976. fin >> tx;
  977. fin >> ty;
  978. translate(tx, ty);
  979. cout << command << endl;
  980. }
  981. else if (command == "reset"){
  982. reset();
  983. cout << command << endl;
  984. }
  985. else if (command == "square"){
  986. square();
  987. cout << command << endl;
  988. }
  989. else if (command == "triangle"){
  990. triangle();
  991. cout << command << endl;
  992. }
  993. else if (command == "view"){
  994. fin >> wxl >> wxr >> wyb >> wyt >> vxl >> vxr >> vyb >> vyt;
  995. view(wxl, wxr, wyb, wyt, vxl, vxr, vyb, vyt);
  996. cout << command << endl;
  997. }
  998. else if (command == "clearData"){
  999. clearData();
  1000. cout << command << endl;
  1001. }
  1002. else if (command == "clearScreen"){
  1003. cout << "Screen is cleared" << endl;
  1004. clearScreen();
  1005. cout << command << endl;
  1006. }
  1007. else if (command == "end"){
  1008. IsExit = true;
  1009. cout << command << endl;
  1010. fin.close();
  1011. return;
  1012. }
  1013. else if (command == "#"){
  1014. getline(fin, comment);
  1015. }
  1016. }
  1017. }
  1018. // Display function
  1019. void displayFunc(void){
  1020. // clear the entire window to the background color
  1021. glClear(GL_COLOR_BUFFER_BIT);
  1022. while (!IsExit){
  1023. glClearColor(default_r, default_g, default_b, alpha);
  1024. #if READ_MULTI_FILE
  1025. if(cur_file < MAX_FILE){
  1026. clearScreen();
  1027. clearData();
  1028. //ReadInput(IsExit);
  1029. ReadFile(IsExit);
  1030. glFlush();
  1031. cout << "Enter any key to run next: ";
  1032. cur_file++;
  1033. }
  1034. else if(cur_file == MAX_FILE){
  1035. exit(0);
  1036. }
  1037. #else
  1038. ReadFile(IsExit);
  1039. #endif
  1040. return;
  1041. }
  1042. if (IsExit){
  1043. redraw();
  1044. }
  1045. //cout << "enter any key to exit: "; getchar();
  1046. //exit(0);
  1047. return;
  1048. }
  1049. // Print some formation and Set the parameter
  1050. void initial(){
  1051. cout << "Welcome GLUT painter" << endl;
  1052. cout << "Here are command key: (case insensitive)" << endl;
  1053. cout << "\treset - reset matrix to identity" << endl;
  1054. cout << "\tscale - scale the matrix" << endl;
  1055. cout << "\trotate - totate the matrix" << endl;
  1056. cout << "\ttranslate - translate the matrix" << endl;
  1057. cout << "\tsquare - create a square" << endl;
  1058. cout << "\ttriangle - create a triangle" << endl;
  1059. cout << "\tview - draw all the objects" << endl;
  1060. cout << "\tclearData - clear up data" << endl;
  1061. cout << "\tclearScreen - clear the screen" << endl;
  1062. cout << "\tend - Quit" << endl;
  1063. }
  1064. // Return the order of pair->x
  1065. bool cmp_pair(const pair<int, int>& a, const pair<int, int>& b){
  1066. return (a.first < b.first);
  1067. }
  1068. // Fill the Polygon
  1069. void FillPolygon(int PointVectorSize, string object){
  1070. vector<pair<int, int>> Intersections;
  1071. ScanFlag flag = even;
  1072. int cur_num_line = num_line;
  1073. if(object == "triangle"){
  1074. r = Color[1].r;
  1075. g = Color[1].g;
  1076. b = Color[1].b;
  1077. }
  1078. else if(object == "square"){
  1079. r = Color[color_index].r;
  1080. g = Color[color_index].g;
  1081. b = Color[color_index].b;
  1082. }
  1083. for(int y = Frame[3]; y >= Frame[2]; y--){
  1084. flag = even;
  1085. Intersections.clear();
  1086. for(int x = Frame[0]; x <= Frame[1]; x++){
  1087. for(int k = PointVectorSize - 1; k > 0; k--){
  1088. lcp = &line_start[cur_num_line - k];
  1089. while(lcp){
  1090. if(lcp->x == x && lcp->y == y){
  1091. Intersections.push_back(make_pair(x, y));
  1092. break;
  1093. }
  1094. lcp = lcp->next;
  1095. }
  1096. }
  1097. }
  1098. //sort(Intersections.begin(), Intersections.end(), cmp_pair);
  1099. int SIZE = Intersections.size();
  1100. if(SIZE >= 2){
  1101. if(object == "triangle")
  1102. fill_line_color.push_back(1);
  1103. else
  1104. fill_line_color.push_back(color_index);
  1105. fill_line.push_back(fill_line_struct(Intersections[0].first, Intersections[SIZE - 1].first, y));
  1106. DrawHorizontalLine(Intersections[0].first, Intersections[SIZE - 1].first, y);
  1107. }
  1108. }
  1109. color_index = (color_index + 1 == MAX_COLOR) ? 2 : color_index + 1;
  1110. return;
  1111. }
  1112. // call for the keyboard
  1113. void myKeyboard(unsigned char key, int x, int y){
  1114. IsExit = false;
  1115. displayFunc();
  1116. return;
  1117. }
  1118. #pragma endregion
  1119. // Main
  1120. void main(int argc, char** argv) {
  1121. int winSizeX, winSizeY;
  1122. string name;
  1123. initial();
  1124. if (argc == 3) {
  1125. winSizeX = atoi(argv[1]);
  1126. winSizeY = atoi(argv[2]);
  1127. // cout<<"Done";
  1128. }
  1129. else { // default window size
  1130. winSizeX = 800;
  1131. winSizeY = 600;
  1132. }
  1133. width = winSizeX;
  1134. height = winSizeY;
  1135. // initialize OpenGL utility toolkit (glut)
  1136. glutInit(&argc, argv);
  1137. // single disply and RGB color mapping
  1138. glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB); // set display mode
  1139. glutInitWindowSize(winSizeX, winSizeY); // set window size
  1140. glutInitWindowPosition(500, 100); // set window position on screen
  1141. glutCreateWindow("Lab2 Window"); // set window title
  1142. #if READ_MULTI_FILE
  1143. // set up the mouse and keyboard callback functions
  1144. glutKeyboardFunc(myKeyboard); // register the keyboard action function
  1145. //glutMouseFunc(Mymouse);
  1146. #endif
  1147. // displayFunc is called whenever there is a need to redisplay the window,
  1148. // e.g., when the window is exposed from under another window or when the window is de-iconified
  1149. glutDisplayFunc(displayFunc); // register the redraw function
  1150. // set background color
  1151. glClearColor(default_r, default_g, default_b, alpha); // set the background to white
  1152. glClear(GL_COLOR_BUFFER_BIT); // clear the buffer
  1153. // misc setup
  1154. glMatrixMode(GL_PROJECTION); // setup coordinate system
  1155. glLoadIdentity();
  1156. gluOrtho2D(0, winSizeX, 0, winSizeY);
  1157. glShadeModel(GL_FLAT);
  1158. glFlush();
  1159. glutMainLoop();
  1160. }

hw2A.in文件

# Computer Graphics Hw2 Test Input A
# copy file content to "hw2.in" and run "CG_Hw2_Sample.exe" # create a square and its transformation
reset
scale 2.0 2.0
rotate 45.0
translate 10.0 10.0
square view 0.0 20.0 0.0 20.0 100 400 100 400 end

hw2B.in文件

# Computer Graphics Hw2 Test Input B
# copy file content to "hw2.in" and run "CG_Hw2_Sample.exe" # create 4 triangles and place them such that
# the assembly looks like a 4-winged fan reset
translate 0.0 -1.0
scale 0.5 1.0
rotate 0.0
triangle rotate 90.0
triangle rotate 90.0
triangle rotate 90.0
triangle view -3.0 3.0 -3.0 3.0 0.0 400.0 0.0 400.0 end

hw2C.in文件

# Computer Graphics Hw2 Test Input C
# copy file content to "hw2.in" and run "CG_Hw2_Sample.exe" reset
scale 0.9 0.9
rotate 15.0
translate 0.0 0.0
square view -3.0 3.0 -3.0 3.0 20 220 20 220 reset
scale 0.9 0.9
rotate 15.0
translate 0.0 0.0
square view -3.0 3.0 -3.0 3.0 240 440 20 220 reset
scale 0.8 0.8
rotate 30.0
translate 0.0 0.0
square view -3.0 3.0 -3.0 3.0 460 680 20 220 reset
scale 0.7 0.7
rotate 45.0
translate 0.0 0.0
square view -3.0 3.0 -3.0 3.0 20 220 240 440 reset
scale 0.6 0.6
rotate 60.0
translate 0.0 0.0
square view -3.0 3.0 -3.0 3.0 240 440 240 440 reset
scale 0.5 0.5
rotate 75.0
translate 0.0 0.0
square view -3.0 3.0 -3.0 3.0 460 660 240 440 end

hw2D.in

# Computer Graphics Hw2 Test Input D
# copy file content to "hw2.in" and run "CG_Hw2_Sample.exe" reset
scale 0.5 0.5
rotate 0.0
translate -.3 -1.0
square reset
scale 1.5 1.5
rotate 45.0
translate 0.6 -0.2
square reset
scale 0.5 0.5
rotate 0.0
translate -0.2 0.2
square view -2.0 2.0 -2.0 2.0 100 400 100 400 view 0.0 2.0 0.0 2.0 420 450 100 500 view 0.0 2.0 0.0 2.0 600 650 200 300 view -2.0 0.0 0.0 2.0 550 600 200 300 view -2.0 0.0 -2.0 0.0 550 600 100 200 view 0.0 2.0 -2.0 0.0 600 650 100 200 end

hw2E.in

# Computer Graphics Hw2 Test Input E
# copy file content to "hw2.in" and run "CG_Hw2_Sample.exe" # HOUSE
reset
scale 4 2
rotate 0
translate 4.0 2.0
square # GROUND
reset
scale 20 0.5
rotate 0
translate -10 -0.5
square # DOOR
reset
scale 0.24 1.0
rotate 0
translate 4.0 1.25
square # WINDOW
reset
scale 0.25 0.5
rotate 0
translate 2.0 2.5
square # WINDOW
reset
scale 0.25 0.5
rotate 0
translate 6.0 2.5
square # ROOF
reset
scale 5.0 1.0
rotate 0
translate 4.0 5.0
triangle # SUPPORT
reset
scale 0.25 3.0
rotate 0
translate -4.0 3.0
triangle # BLADE
reset
scale 0.25 3.5
rotate -45
translate -4 5.75
square # BLADE
reset
scale 0.25 3.5
rotate 45
translate -4 5.75
square # VIEW SCENE
view -10.0 10.0 -2.0 10.0 20 220 20 140 # VIEW SCENE
view -5.0 5.0 -2.0 10.0 270 500 20 140 # VIEW SCENE
view -10.0 10.0 1.0 4.0 20 320 160 580 # VIEW SCENE
view -10.0 10.0 1.0 4.0 340 740 220 340 end

hw2F.in

# Computer Graphics Hw2 Test Input F
# copy file content to "hw2.in" and run "CG_Hw2_Sample.exe" reset
scale 5.0 4.0
rotate 0
translate 0.0 0.0
square reset
scale 8.5 0.5
rotate 45
translate 0.5 5.5
square reset
scale 8.5 0.5
rotate -45
translate 0.5 5.5
square reset
scale 8.5 0.5
rotate -135
translate 0.5 5.5
square reset
scale 8.5 0.5
rotate 135
translate 0.5 5.5
square # VIEW SCENE
view -10.0 10.0 0.0 10.0 500 700 20 140 end

C++实现数字媒体二维图像变换的更多相关文章

  1. php 二维数组验证一个值是否存在

    php 判断数字在二维数组里 $arr = array( array('a', 'b'), array('c', 'd') ); in_array('a', $arr); // 此时返回的永远都是 f ...

  2. C++实现数字媒体三维图像变换

    C++实现数字媒体三维图像变换 必备环境 glut.h 头文件 glut32.lib 对象文件库 glut32.dll 动态连接库 程序说明 C++实现了用glut画物体对象的功能.并附带放大缩小,旋 ...

  3. PHP 统计一维数组value同样的元素的个数num,并将其转化为下标为数字,值是value和num的二维数组

    近期做一个项目.从数据库查询某个字段得到一个数组key是数字值是channel的一维数组$res,现须要将这个数组变成键是数字值是channel和num(num为同样channel的数量,默觉得0). ...

  4. javascript使用H5新版媒体接口navigator.mediaDevices.getUserMedia,做扫描二维码,并识别内容

    本文代码测试要求,最新的chrome浏览器(手机APP),并且要允许chrome拍照录像权限,必须要HTTPS协议,http不支持. 原理:调用摄像头,将摄像头返回的媒体流渲染到视频标签中,再通过ca ...

  5. 利用 Python django 框架 输入汉字,数字,字符,等。。转成二维码!

    利用 Python django 框架 输入汉字,数字,字符,等..转成二维码! 模块必备:Python环境 + pillow  + qrcode 模块 核心代码import qrcode qr = ...

  6. 剑指offer1: 组类型——二维数组中的查找(给定一个数字,查找是否在该数组中)

    1. 思路: 缩小范围 2. 方法: (1)要查找的数字等于数组中的数字,结束查找过程: (2)要查找的数字小于数组中的数字,去除该数字右边的数字,在剩下的数字里查找: (3)要查找的数字大于数组中的 ...

  7. 数字信号处理实验(零)—— 一维声音信号处理和二维图像处理

    一.在matlab下声音信号的I/O 1.读wav文件函数 •y = wavread('filename') •[y,Fs,bits] = wavread('filename') •[...] = w ...

  8. c语言数字图像处理(六):二维离散傅里叶变换

    基础知识 复数表示 C = R + jI 极坐标:C = |C|(cosθ + jsinθ) 欧拉公式:C = |C|ejθ 有关更多的时域与复频域的知识可以学习复变函数与积分变换,本篇文章只给出DF ...

  9. PHP一维数组和二维数字排序整理

    <?php /** 一维数组排序 sort() - 以升序对数组排序 rsort() - 以降序对数组排序 asort() - 根据值,以升序对关联数组进行排序 ksort() - 根据键,以升 ...

随机推荐

  1. jQuery选择器之可见性过滤选择器Demo

    测试代码 05-可见性过滤选择器.html <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" &q ...

  2. #316 div.2

    主要记录下被坑的B.果然大晚上脑子就是不知道在干嘛,明明都测到 “1 1” 这个样例错了都没发现直接给放过去了,白白让人hack爽了... 题意就是给你一个数m,让你从1~n个数中选一个作a,使1~n ...

  3. Hbse的读写过程

    1.Hbase的读取过程. 以读取test_region表,row key为this is row value 400000为例. 1: 到zookeeper中去读取/hbase/root-regio ...

  4. 使用开源库MagicalRecord操作CoreData

      1. 将 MagicalRecord 文件夹拖入到工程文件中,引入 CoreData.frame 框架 2. 在 .pch 文件中引入头文件 CoreData+MagicalRecord.h 注: ...

  5. MacOS显示和不显示隐藏文件

    defaults write com.apple.finder AppleShowAllFiles Yes && killall Finder //显示隐藏文件 defaults wr ...

  6. JavaScript--DOM事件(笔记)

    第1章 事件流1-1.事件冒泡:事件最开始由最具体的元素(文档中嵌套层次最深的那个节点)接收; 然后逐级向上传播至最不具体的那个节点(文档);1-2.事件捕获:不太具体的节点应该更早接收到事件,而最具 ...

  7. $_SERVER 中重要的元素

      元素/代码 描述 $_SERVER['PHP_SELF'] 返回当前执行脚本的文件名. $_SERVER['GATEWAY_INTERFACE'] 返回服务器使用的 CGI 规范的版本. $_SE ...

  8. 谷歌浏览器Chrome不再支持showModalDialog的解决办法

    问题重现 弹出窗口编码: JavaScript 0 1 2 3 4 5 6 7   var obj = new Object();   var retval = window.showModalDia ...

  9. UI6_UIAlertContrller

    // // ViewController.m // UI6_UIAlertContrller // // Created by zhangxueming on 15/7/7. // Copyright ...

  10. (转)Yale CAS + .net Client 实现 SSO(1)

    由于信息系统集成需要,最近研究了一下CAS.从网上找了不少资料,很多是针对Java平台的,为数不多的针对.net Client的文章往往片面的介绍某个方面,照着去做确会遇到大量的问题,特别是“重定向循 ...