- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想画一些形状,然后用选定的颜色填充这些形状。此时,我刚刚完成了一些基本形状的绘制,但我仍然停留在用 flood-fill 算法给它们上色。
在细节上,我不能在点击它后给整个形状上色。它所能做的只是在屏幕边缘画一条线,将所选颜色泄漏到形状边缘之外(忽略形状的边界颜色)。
这是我所有的工作,谁能帮帮我???
#include <GL/glut.h>
#include <math.h>
#include <iostream>
using namespace std;
static int window;
static int value = 0;
GLsizei width, height;
int flag = 0;
bool up = false, down = false;
struct Point {
GLint x;
GLint y;
};
struct RGBColor
{
GLfloat r;
GLfloat g;
GLfloat b;
};
struct Position
{
Position() : x(0), y(0) {}
float x;
float y;
};
Position start;
Position finish;
void init()
{
glClearColor(0, 0, 0, 0);
glPointSize(2.0);
glLineWidth(2.0);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
gluOrtho2D(0.0, 500.0, 500.0, 0.0);
}
void menu(int num) {
if (num == 0) {
glutDestroyWindow(window);
exit(0);
}
else {
value = num;
}
glutPostRedisplay();
}
void createMenu(void)
{
int sub_Triangle = glutCreateMenu(menu);
glutAddMenuEntry("Isoceles right", 2);
glutAddMenuEntry("Equilateral", 3);
int menu_Triangle = glutCreateMenu(menu);
int sub_Quadrilateral = glutCreateMenu(menu);
glutAddMenuEntry("Rectangle", 4);
glutAddMenuEntry("Square", 5);
int menu__Quadrilateral = glutCreateMenu(menu);
int sub_Oval = glutCreateMenu(menu);
glutAddMenuEntry("Circle", 6);
glutAddMenuEntry("Ellipse", 7);
int menu_Oval = glutCreateMenu(menu);
int sub_RegularPolygon = glutCreateMenu(menu);
glutAddMenuEntry("Pentagon", 8);
glutAddMenuEntry("Hexagon", 9);
int menu_RegularPolygon = glutCreateMenu(menu);
int sub_OtherShape = glutCreateMenu(menu);
glutAddMenuEntry("Arrow", 10);
glutAddMenuEntry("Star", 11);
int menu_OtherShape = glutCreateMenu(menu);
int sub_Operation = glutCreateMenu(menu);
glutAddMenuEntry("Add", 12);
glutAddMenuEntry("Subtract", 13);
glutAddMenuEntry("Multiply", 14);
glutAddMenuEntry("Divide", 15);
int menu_Operation = glutCreateMenu(menu);
int sub_ColorFill = glutCreateMenu(menu);
glutAddMenuEntry("Red", 16);
glutAddMenuEntry("Green", 17);
glutAddMenuEntry("Blue", 18);
int menu_ColorFill = glutCreateMenu(menu);
glutAddMenuEntry("Line", 1);
glutAddSubMenu("Triangle", sub_Triangle);
glutAddSubMenu("Quadrilateral", sub_Quadrilateral);
glutAddSubMenu("Oval", sub_Oval);
glutAddSubMenu("Regular Polygon", sub_RegularPolygon);
glutAddSubMenu("Other Shape", sub_OtherShape);
glutAddSubMenu("Operation", sub_Operation);
glutAddSubMenu("Color Fill", sub_ColorFill);
glutAddMenuEntry("Shape Choice", 19);
glutAttachMenu(GLUT_RIGHT_BUTTON);
}
void draw_Line()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
glVertex2f(start.x, start.y);
glVertex2f(finish.x, finish.y);
glEnd();
glutSwapBuffers();
}
void draw_IsocelesRight()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
int a = finish.y - start.y;
glVertex2f(start.x, start.y);
glVertex2f(start.x, start.y + a);
glVertex2f(start.x + a, start.y + a);
}
void draw_Equilateral()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
int midx = (finish.x + start.x) / 2;
int midbotx = (start.y - finish.y)*(1 / sqrt(3));
int midboty = finish.y;
glVertex2f(midx, start.y);
glVertex2f(midx - midbotx, finish.y);
glVertex2f(midx + midbotx, finish.y);
}
void draw_Rectangle()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
glVertex2f(start.x, start.y);
glVertex2f(finish.x, start.y);
glVertex2f(finish.x, finish.y);
glVertex2f(start.x, finish.y);
}
void draw_Square()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
int a = finish.y - start.y;
glVertex2f(start.x, start.y);
glVertex2f(start.x, start.y + a);
glVertex2f(start.x + a, start.y + a);
glVertex2f(start.x + a, start.y);
}
void draw_Circle()
{
float PI = 3.14;
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
int size = sqrt(pow(finish.x - start.x, 2) + pow(finish.y - start.y, 2));
for (int i = 0; i <= 360; i++)
{
float theta = (2 * 3.14 * i) / 360;
glVertex2f((size / 2) * cos(theta) + finish.x, (size / 2) * sin(theta) + finish.y);
}
}
void draw_Ellipse()
{
float PI = 3.14;
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
int size1 = sqrt(pow(finish.x - start.x, 2) + pow(finish.x - start.x, 2));
int size2 = sqrt(pow(finish.y - start.y, 2) + pow(finish.y - start.y, 2));
for (int i = 0; i <= 360; i++)
{
float theta = (2 * 3.14 * i) / 360;
glVertex2f((size1 / 2) * cos(theta) + finish.x, (size2 / 2) * sin(theta) + finish.y);
}
}
void draw_Pentagon()
{
const double PI = 3.14159265358979323846;
double r = sqrt(pow(finish.x - start.x, 2));
int sides = 5;
int center_x = start.x + (finish.x - start.x) / 2;
int center_y = start.y + (finish.y - start.y) / 2;
glBegin(GL_LINE_LOOP);
for (int i = 0; i < sides; i++) {
double angle = i * 2 * PI / sides;
glVertex2d(center_x + r*cos(angle), center_y + r*sin(angle));
}
}
void draw_Hexagon()
{
const double PI = 3.14159265358979323846;
double r = sqrt(pow(finish.x - start.x, 2));
int sides = 6;
int center_x = start.x + (finish.x - start.x) / 2;
int center_y = start.y + (finish.y - start.y) / 2;
glBegin(GL_LINE_LOOP);
for (int i = 0; i < sides; i++) {
double angle = i * 2 * PI / sides;
glVertex2d(center_x + r*cos(angle), center_y + r*sin(angle));
}
}
void draw_Arrow()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINES);
const double PI = 3.14159265358979323846;
double length = finish.x - start.x;
double e = length / 3;
double x1 = finish.x - e * (1/tan(45 * PI /180));
double y1 = start.y - e * tan(45 * PI / 180);
double x2 = finish.x - e * (1 / tan(45 * PI / 180));
double y2 = start.y + e * tan(45 * PI / 180);
glVertex2f(start.x, start.y);
glVertex2f(finish.x, start.y);
glVertex2f(finish.x, start.y);
glVertex2f(x1, y1);
glVertex2f(finish.x, start.y);
glVertex2f(x2, y2);
}
void draw_Star()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINE_LOOP);
glVertex2f(100, 225);
glVertex2f(210, 225);
glVertex2f(250, 100);
glVertex2f(290, 225);
glVertex2f(400, 225);
glVertex2f(315, 290);
glVertex2f(350, 400);
glVertex2f(250, 330);
glVertex2f(150, 400);
glVertex2f(185, 290);
}
void draw_Add()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINES);
double length = finish.x - start.x;
double half_length = length / 2;
glVertex2f(start.x + half_length, start.y);
glVertex2f(start.x + half_length, start.y + length);
glVertex2f(start.x, start.y + half_length);
glVertex2f(start.x + length, start.y + half_length);
}
void draw_Subtract()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINES);
double length = finish.x - start.x;
glVertex2f(start.x, start.y);
glVertex2f(start.x + length, start.y);
}
void draw_Multiply()
{
glColor3f(1.0, 1.0, 1.0);
glBegin(GL_LINES);
double length = finish.x - start.x;
glVertex2f(start.x, start.y);
glVertex2f(start.x + length, start.y + length);
glVertex2f(start.x + length , start.y);
glVertex2f(start.x, start.y + length);
}
RGBColor getPixelColor(GLint x, GLint y)
{
RGBColor color;
glReadPixels(x, y, 1, 1, GL_RGB, GL_FLOAT, &color);
return color;
}
void setPixelColor(GLint x, GLint y, RGBColor color)
{
glColor3f(color.r, color.g, color.b);
glBegin(GL_POINTS);
glVertex2i(x, y);
glEnd();
glFlush();
}
void floodFill(GLint x, GLint y, RGBColor oldColor, RGBColor newColor)
{
RGBColor color;
color = getPixelColor(x, y);
if (color.r == oldColor.r && color.g == oldColor.g && color.b == oldColor.b)
{
setPixelColor(x, y, newColor);
floodFill(x + 1, y, oldColor, newColor);
floodFill(x, y + 1, oldColor, newColor);
floodFill(x - 1, y, oldColor, newColor);
floodFill(x, y - 1, oldColor, newColor);
}
return;
}
bool IsSameColor(RGBColor x, RGBColor y) {
if (x.r == y.r && x.b == y.b && x.g == y.g) {
cout << "is same color true" << endl;
return true;
}
else {
cout << "is same color false" << endl;
return false;
}
}
void BoundaryFill(int x, int y, RGBColor F_Color, RGBColor B_Color) {
cout << "bound fill" << endl;
RGBColor currentColor;
currentColor = getPixelColor(x, y);
if (!IsSameColor(currentColor, B_Color) && !IsSameColor(currentColor, F_Color)) {
cout << "bound" << endl;
setPixelColor(x, y, F_Color);
BoundaryFill(x - 1, y, F_Color, B_Color);
BoundaryFill(x, y + 1, F_Color, B_Color);
BoundaryFill(x + 1, y, F_Color, B_Color);
BoundaryFill(x, y - 1, F_Color, B_Color);
}
}
void display()
{
if (flag == 1)
{
if (value == 1) {
draw_Line();
}
else if (value == 2) {
draw_IsocelesRight();
}
else if (value == 3) {
draw_Equilateral();
}
else if (value == 4) {
draw_Rectangle();
}
else if (value == 5) {
draw_Square();
}
else if (value == 6) {
draw_Circle();
}
else if (value == 7) {
draw_Ellipse();
}
else if (value == 8) {
draw_Pentagon();
}
else if (value == 9) {
draw_Hexagon();
}
else if (value == 10) {
draw_Arrow();
}
else if (value == 11) {
draw_Star();
}
else if (value == 12) {
draw_Add();
}
else if (value == 13) {
draw_Subtract();
}
else if (value == 14) {
draw_Multiply();
}
else if (value == 16)
{
RGBColor newColor = { 1.0f, 0.0f, 0.0f }; // red
RGBColor oldColor = { 0.0f, 0.0f, 0.0f }; // black
floodFill(start.x, start.y, oldColor, newColor);
}
glEnd();
glutSwapBuffers();
glFlush();
}
flag = 0;
}
void reshape(int w, int h)
{
glViewport(0, 0, (GLsizei)w, (GLsizei)h);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
gluOrtho2D(0, w, h, 0);
width = w;
height = h;
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
}
void mouse(int button, int state, int x, int y)
{
if (button == GLUT_LEFT_BUTTON && state == GLUT_DOWN)
{
down = true;
start.x = x; //x1
start.y = y; //y1
}
if (button == GLUT_LEFT_BUTTON && state == GLUT_UP)
{
if (down == true) {
up = true;
flag = 1;
finish.x = x;
finish.y = y;
}
down = false;
up = false;
}
glutPostRedisplay();
}
int main(int argc, char** argv)
{
glutInit(&argc, argv);
glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB);
glutInitWindowSize(500, 500);
glutInitWindowPosition(450, 100);
window = glutCreateWindow("Coloring Shapes");
init();
createMenu();
glutDisplayFunc(display);
glutReshapeFunc(reshape);
glutMouseFunc(mouse);
glutMainLoop();
return 0;
}
最佳答案
主要问题是,您传递给 glReadPixels
的坐标是错误的。
绘制点时,通过 glVertex
然后坐标由模型 View 和投影矩阵转换。
所以应用了正交投影
gluOrtho2D(0.0, 500.0, 500.0, 0.0);
但是 glReadPixels
直接读取帧缓冲区的片段。坐标是帧缓冲区中片段的 x 和 y 坐标,其中左下角始终为 (0, 0),右上角为 (widht, height) .阅读片段时必须翻转 y 坐标:
RGBColor getPixelColor(GLint x, GLint y)
{
RGBColor color;
glReadPixels(x, height-y, 1, 1, GL_RGB, GL_FLOAT, &color);
return color;
}
为了在调用glVertex2i
时不出现舍入错误, 我推荐使用 glRasterPos2i
和 glDrawPixels
反而。坐标为glRasterPos2i
由当前模型 View 和投影矩阵转换:
void setPixelColor(GLint x, GLint y, RGBColor color)
{
glRasterPos2i( x, y );
glDrawPixels( 1, 1, GL_RGB, GL_FLOAT, &color );
glFlush();
}
由于颜色值是浮点值,因此在从 GPU 读回颜色时,您必须考虑不准确。比较颜色时请考虑这一点:
例如
color1.r == color2.r
fabs(color1.r - color2.r) < 0.001f
编写一个比较颜色的函数并在floodFill
中使用它:
bool isEqual( const RGBColor &c1, const RGBColor &c2 )
{
return fabs(c1.r - c2.r) < 0.001f && fabs(c1.g - c2.g) < 0.001f && fabs(c1.b - c2.b) < 0.001f;
}
void floodFill(GLint x, GLint y, RGBColor oldColor, RGBColor newColor)
{
RGBColor color;
color = getPixelColor(x, y);
if ( isEqual(color, oldColor) )
{
setPixelColor(x, y, newColor);
floodFill(x + 1, y, oldColor, newColor);
floodFill(x, y + 1, oldColor, newColor);
floodFill(x - 1, y, oldColor, newColor);
floodFill(x, y - 1, oldColor, newColor);
}
return;
}
无论如何,您的代码只适用于小范围,因为该算法是递归的。堆栈使用量迅速增加,并立即导致堆栈溢出。
但它适用于尖齿区域。
关于c++ - 达到边界颜色时,洪水填充无法填充整个形状?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55514519/
我已经下载了 RStudio,在打开我的代码所在的文件时,我似乎已经达到了容量限制: The file is 2.3MB the maximum file size is 2MB The file i
我有一个按钮,每次单击时,都会将 1 添加到变量中。当此变量超过 5 时,将触发警报。然而,此后触发器仍不断激活。我尝试使用 == 而不是 > 进行检查,但它做同样的事情。有什么想法吗? http:/
我正在将Slick 3.0与HikariCP 2.3.8一起使用(也可以玩2.4) 我做了很多数据库IO,并且不断达到队列限制。 有没有一种方法可以获取当前的队列大小,以及如何增加队列大小? 还是建议
在 Salesforce 中,您可以设置各种工作流程或构建用于发送电子邮件的 API 应用程序。对于大多数标准 Salesforce 组织,每天有 1000 封电子邮件的限制。 (例如,参见 here
我有一个类是这样的: public sealed class Contract { public bool isExpired { get; set; } public DateTim
我有一个带有特殊符号按钮的输入作为附加组件。 HTML
我正在尝试压缩 pdf 文件(有时是图像)。我需要一个 java 压缩器来帮助我压缩文件。我需要尺寸小于原始文档尺寸的一半。我尝试了java api中给出的deflator。但它并不是很成功。请帮我解
我正在使用这条线来创建淡入效果。 $('#div').css({opacity: 0, visibility:"visible"}).animate({opacity: 1}, 500); 可见类达到
我使用 URLCache 来缓存请求响应,最大容量如下: let diskCapacity = 100 * 1024 * 1024 let memoryCapacity = 100
我有一个计数器函数,我从这个 Answer 得到它: function countDown(i) { var int = setInterval(function () {
下面是一段代码,用于检查给定数字是否为 Lychrel 数字。这基本上意味着该程序取一个数及其倒数之和,然后取那个数及其倒数之和,等等,直到找到回文。如果它在一定的迭代次数内没有找到这样的数字(我在这
我即将对这个可怕的旧 Java Web 应用程序做一些工作,这是我的一个 friend 不久前继承的。 在我设置 tomcat、导入项目和所有这些到我的 eclipse 工作区后,我收到此错误,指出
我有一个 NSDictionary 对象,其中包含深层结构,例如包含包含字典的进一步数组的数组... 我想在层次结构中向下获取一个对象。是否有任何直接索引方法可以使用键名或其他方式获取它们? 多次调用
正如标题所说,我的 .border div 的边框跨度比它里面的要宽。它只会在达到 710px 时发生,因此您需要在 this fiddle 中展开结果窗口。 . 我希望边框保持在其内容周围而不超过它
我在 MySQL 中有一个表,通过 Microsoft Access 2013 中的链接表(通过 ODBC) Access 。 此表包含超过 124,000 条记录,我需要一个表单中的 ComboBo
一旦上一个输入达到其最大长度值,我如何才能聚焦下一个输入? a: b: c: 如果用户粘贴的文本大于最大长度,理想情况下它应该溢出到下一个输入。 jsFiddle: http://jsfiddl
我的任务是在客户的 QA 服务器上提供服务器性能报告。理想情况下,客户希望对约 900 个并发用户进行负载测试,因为这是他们在高峰时段通常使用的数量。然而,我一直在做的负载测试正在使他们的 QA 服务
我在 django 应用程序中对我的 celery worker 运行任务,其中每个任务执行大约需要 1-2 秒。通常这些执行都很好,但有时,特别是如果 Django 应用程序已经部署了一段时间,我开
我有一个 one_for_one 主管来处理类似且完全独立的 child 。 当一个 child 出现问题时,反复崩溃并触发: =SUPERVISOR REPORT==== 30-Mar-2011::
根据该网站,他们在免费计划中限制了 100 个并发连接,但是当第 101 个连接尝试连接时,它被拒绝,那么什么时候允许新连接? 例如:用户是否必须等待一定时间或一旦一个连接关闭,另一个连接就有机会连接
我是一名优秀的程序员,十分优秀!