c++ - 在圆圈周围放置一个 ofImage

标签 c++ graphics openframeworks

我正在使用 Openframeworks 为 Reactivision 开发,我需要围绕一个圆放置一个 ofImage(可以放置在屏幕中的任何位置),并将它旋转特定角度。我附上一张图片来帮助解决这个问题。

http://i.stack.imgur.com/WCyPu.png

如你所见,x 轴和 y 轴从 0,0 到 1,1,我在屏幕中间有一个圆(圆心位于 0.5,0.5),半径为 0.42 (所以它不会填满屏幕)。

最终,用户将 ofImage(在本例中为红色笑脸)放置在屏幕上的位置 0.2,0.2 处。它的宽度和高度都是 0.08,旋转中心在它的左上角(和往常一样)。

我需要做的是拿那个笑脸,把它放在圆的周围,在最近的位置(如果转起来更容易,它可以跟随半径)它就可以旋转它,所以它会是面向圆心。

我还需要找到距离 ofImage 轴 0.04 0.04 处的笑脸“ Nose ”,移动它并旋转它以保持笑脸 Nose 而不是空隙中的一个点。最好的红色笑脸应该是绿色笑脸。

我的三角学有点生疏,我最好的方法也行不通。我在构造函数中定义了它的最终位置,并在 draw() 步骤中旋转了 Sprite 。

Smiley::Smiley(int _id, float x, float y)
{
    smileyImg.loadImage("smiley.png");

    float tangent = (0.5-x)/(0.5-y);
    //angle is a private float variable; 
    angle = atanf(tangent);
    //those 0.06 just lets me move the smiley a little over the circle itself
    x = 0.5+cos(angle)*(RADIUS+0.06);
    y = 0.5+sin(angle)*(RADIUS+0.06);
    float xNose = 0.5+cos(angle)*(RADIUS+0.06);
    float yNose = 0.5+sin(angle)*(RADIUS+0.06);
    angle = ofRadToDeg(angle);
    //pos and nose are Vector3 classes which hold three floats. Nothing too much important here
    pos.set(x, y, 0.0);
    nose.set(xNose, yNose, 0.0);
}

void Smiley::draw()
{
    ofPushMatrix();
    ofTranslate(pos.x,pos.y);
    ofRotateZ(angle);
    ofTranslate(-pos.x,-pos.y);
    ofSetColor(255,255,255);
    ofEnableAlphaBlending();
    smileyImg.draw(pos.x,pos.y,0.08,0.08);
    ofDisableAlphaBlending();
    ofPopMatrix();
}

有人可以帮我解决这个问题吗?非常感谢。

编辑:采用@datenwolf 解决方案后,我得到的位置是完美的,但是当我将旋转和平移(或只是平移)放入转换矩阵时,它会使笑脸消失。我很确定我做错了矩阵:

Smiley::Smiley(int _id, float x, float y)
{
    smileyImg.loadImage("smiley.png");

    float distance = sqrt( pow((x - 0.46),2) + pow((y - 0.42),2));
    Vector3 director((x - 0.46)/ distance, (y - 0.42)/ distance, 0.0);
    pos.x = 0.46 + RADIUS * director.x;
    pos.y = 0.42 + RADIUS * director.y;

    float distanceNose = sqrt( pow((x - 0.46),2) + pow((y - 0.42),2));
    Vector3 noseDirector((x - 0.46)/ distanceNose, (y - 0.42)/ distanceNose, 0.0);
    nose.x = 0.46 + RADIUS * noseDirector.x;
    nose.y = 0.42 + RADIUS * noseDirector.y;

    /* matrix is a float matrix[16]. I think the indexing is ok, but even going from
    0 4 8 12... to 0 1 2 3..., it still doesn't work*/
    matrix[0] = -director.y; matrix[4] = director.x; matrix[8] = 0.0; matrix[12] = pos.x;
    matrix[1] = director.x; matrix[5] = director.y; matrix[9] = 0.0; matrix[13] = pos.y;
    matrix[2] = 0.0; matrix[6] = 0.0; matrix[10] = 1.0; matrix[14] = 0.0;
    matrix[3] = 0.0; matrix[7] = 0.0; matrix[11] = 0.0; matrix[15] = 1.0;
}

void Smiley::draw()
{
    ofPushMatrix();
    glLoadMatrixf(matrix);
    ofSetColor(255,255,255);
    ofEnableAlphaBlending();
    // Now that I have a transformation matrix that also transposes, I don't need to define the coordinates here, so I put 0.0,0.0
    noseImg.draw(0.0,0.0,0.08,0.08);
    ofDisableAlphaBlending();
    ofPopMatrix();
}

此外,我需要根据每个笑脸旋转 Nose 的位置。你认为哪种方式最好?非常感谢

最佳答案

像往常一样,您不需要三角函数。设 C 为圆心,r 为半径,P 为用户点击的点。那么点击转换到圆上的位置U为

U = C + r * (P - C)/|P - C|

在哪里

| A | = sqrt(A_x² + A_y² + …);

所以在二维的情况下你得到

U.x = C.x + r * (P.x - C.x)/sqrt( (P.x - C.x)² + (P.y - C.y)² )
U.y = C.y + r * (P.y - C.y)/sqrt( (P.x - C.x)² + (P.y - C.y)² )

旋转可以用旋转矩阵 R 来描述,它由从圆 C 到 P 的方向和垂直于它的 T 组成。所以大胖编辑:归一化方向:

D = P - C = (P.x - C.x , P.y - C.y)/sqrt( (P.x - C.x)² + (P.y - C.y)² )
T = Z × D 

另请注意,您可以根据 D 编写 U

U = C + r*D

用它来构造一个旋转矩阵

    | T.x D.x |
R = |         |
    | T.y D.y |

R 是一个旋转矩阵,它遵循 T.y == D.x 并计算 T.x

T.x = -P.y + C.y = -D.y

所以

    | -D.y D.x |
R = |          |
    |  D.x D.y |

大胖 编辑:我在矩阵的最后一列使用了错误的组件。 是 U.{x,y} 不是 D.{x.y}。不好意思!

我们可以把它放到一个完整的变换矩阵M中:

    | -D.y D.x   0 U.x |   | -D.y D.x   0 (C + r*D.x) |
M = |  D.x D.y   0 U.y | = |  D.x D.y   0 (C + r*D.y) |
    |    0   0   1   0 |   |    0   0   1      0      |
    |    0   0   0   1 |   |    0   0   0      1      |

您可以使用 glLoadMatrix 加载此矩阵以放置 Sprite 。

关于c++ - 在圆圈周围放置一个 ofImage,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5290502/

相关文章:

c++ - 引用构造函数

c++ - Visual C++ (2008) 调试快照

c++ - 冒号运算符在 C++ 中对变量声明意味着什么

c# - 如何去除不太集中的像素并留下最密集的部分

c++ - C++ 新手, "EXC_BAD_ACCESS"错误我不明白

c++ - ofVideoPlayer 不起作用

c++ - 对象和值表示的区别

java - java中如何让图形从屏幕上消失?

javascript - 有什么好的 Javascript 图形库吗?

c++ - 什么时候需要无锁数据结构来跨线程读取/写入音频应用程序中的数据?