summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--pose-widget/glwidget.cpp232
-rw-r--r--pose-widget/glwidget.h2
2 files changed, 131 insertions, 103 deletions
diff --git a/pose-widget/glwidget.cpp b/pose-widget/glwidget.cpp
index b2ec68a7..5de26eef 100644
--- a/pose-widget/glwidget.cpp
+++ b/pose-widget/glwidget.cpp
@@ -51,44 +51,13 @@ void GLWidget::rotateBy(float xAngle, float yAngle, float zAngle, float x, float
update();
}
-
class Triangle {
using num = GLWidget::num;
using vec2 = GLWidget::vec2;
using vec3 = GLWidget::vec3;
public:
- Triangle(const vec2& p1,
- const vec2& p2,
- const vec2& p3)
- {
- origin = p1;
- v0 = vec2(p3.x() - p1.x(), p3.y() - p1.y());
- v1 = vec2(p2.x() - p1.x(), p2.y() - p1.y());
- dot00 = v0.dot(v0);
- dot01 = v0.dot(v1);
- dot11 = v1.dot(v1);
- const num denom = dot00 * dot11 - dot01 * dot01;
- if (std::fabs(denom) < 1e3f)
- {
- // for perpendicular plane, ensure u and v don't come out right
- // this is done here to avoid branching below, in a hot loop
- invDenom = 0;
- dot00 = dot01 = dot11 = 0;
- v0 = v1 = vec2(0, 0);
- }
- else
- invDenom = 1 / denom;
- }
- bool barycentric_coords(const vec2& px, vec2& uv) const
- {
- const vec2 v2 = px - origin;
- const num dot12 = v1.dot(v2);
- const num dot02 = v0.dot(v2);
- const num u = (dot11 * dot02 - dot01 * dot12) * invDenom;
- const num v = (dot00 * dot12 - dot01 * dot02) * invDenom;
- uv = vec2(u, v);
- return (u >= 0) && (v >= 0) && (u + v <= 1);
- }
+ Triangle(const vec2& p1, const vec2& p2, const vec2& p3);
+ bool barycentric_coords(const vec2& px, vec2& uv, int& i) const;
private:
num dot00, dot01, dot11, invDenom;
@@ -109,7 +78,56 @@ inline GLWidget::vec3 GLWidget::normal(const vec3& p1, const vec3& p2, const vec
return tmp * i;
}
-void GLWidget::project_quad_texture() {
+Triangle::Triangle(const Triangle::vec2& p1, const Triangle::vec2& p2, const Triangle::vec2& p3)
+{
+ using std::fabs;
+
+ origin = p1;
+
+ v0 = vec2(p3 - p1);
+ v1 = vec2(p2 - p1);
+
+ dot00 = v0.dot(v0);
+ dot01 = v0.dot(v1);
+ dot11 = v1.dot(v1);
+
+ const num denom = dot00 * dot11 - dot01 * dot01;
+
+ if (fabs(denom) < num(1e3))
+ {
+ // for perpendicular plane, ensure u and v don't come out right
+ // this is done here to avoid branching below, in a hot loop
+ invDenom = 0;
+ dot00 = dot01 = dot11 = 0;
+ v0 = v1 = vec2(0, 0);
+ }
+ else
+ invDenom = 1 / denom;
+}
+
+bool Triangle::barycentric_coords(const Triangle::vec2& px, Triangle::vec2& uv, int& i) const
+{
+ i = 0;
+ const vec2 v2 = px - origin;
+ const num dot12 = v1.dot(v2);
+ const num dot02 = v0.dot(v2);
+ num u = (dot11 * dot02 - dot01 * dot12) * invDenom;
+ num v = (dot00 * dot12 - dot01 * dot02) * invDenom;
+ if (!(u >= 0 && v >= 0))
+ return false;
+ if (u + v > 1)
+ {
+ i = 1;
+
+ u = 1 - u;
+ v = 1 - v;
+ }
+ uv = vec2(u, v);
+ return u >= 0 && v >= 0 && u + v <= 1;
+}
+
+void GLWidget::project_quad_texture()
+{
const int sx = width(), sy = height();
const int ow = front.width(), oh = front.height();
const vec3 corners[] = {
@@ -141,12 +159,27 @@ void GLWidget::project_quad_texture() {
QColor bgColor = palette().color(QPalette::Current, QPalette::Window);
texture.fill(bgColor);
- const vec2 projected[2][3] = { { pt[0], pt[1], pt[2] }, { pt[3], pt[1], pt[2] } };
- const vec2 origs[2][3] = {
- { vec2(0, 0), vec2(ow-1, 0), vec2(0, oh-1) },
- { vec2(ow-1, oh-1), vec2(ow-1, 0), vec2(0, oh-1) }
+ const vec2 projected[2][3] =
+ {
+ { pt[0], pt[1], pt[2] },
+ { pt[3], pt[1], pt[2] }
};
- const Triangle triangles[2] = {
+
+ const vec2 origs[2][3] =
+ {
+ {
+ vec2(0, 0),
+ vec2(ow-1, 0),
+ vec2(0, oh-1) },
+ {
+ vec2(ow-1, oh-1),
+ vec2(0, oh-1),
+ vec2(ow-1, 0)
+ }
+ };
+
+ const Triangle triangles[2] =
+ {
Triangle(projected[0][0], projected[0][1], projected[0][2]),
Triangle(projected[1][0], projected[1][1], projected[1][2])
};
@@ -165,70 +198,64 @@ void GLWidget::project_quad_texture() {
return;
for (int y = 0; y < sy; y++)
- for (int x = 0; x < sx; x++) {
+ for (int x = 0; x < sx; x++)
+ {
vec2 pos(x, y);
- for (int i = 0; i < 2; i++) {
- vec2 uv;
- // XXX knowing center of the lookup pos,
- // we have symmetry so only one lookup is needed -sh 20150831
- if (triangles[i].barycentric_coords(pos, uv))
- {
- const float fx = origs[i][0].x()
- + uv.x() * (origs[i][2].x() - origs[i][0].x())
- + uv.y() * (origs[i][1].x() - origs[i][0].x());
- const float fy = origs[i][0].y()
- + uv.x() * (origs[i][2].y() - origs[i][0].y())
- + uv.y() * (origs[i][1].y() - origs[i][0].y());
-
- const int px_ = fx + .5f;
- const int py_ = fy + .5f;
- const int px = fx;
- const int py = fy;
- const float ax_ = fx - px;
- const float ay_ = fy - py;
- const float ax = 1.f - ax_;
- const float ay = 1.f - ay_;
-
- // 0, 0 -- ax, ay
- const int orig_pos = py * orig_pitch + px * orig_depth;
- const unsigned char r = orig[orig_pos + 2];
- const unsigned char g = orig[orig_pos + 1];
- const unsigned char b = orig[orig_pos + 0];
-
- // 1, 1 -- ax_, ay_
- const int orig_pos_ = py_ * orig_pitch + px_ * orig_depth;
- const unsigned char r_ = orig[orig_pos_ + 2];
- const unsigned char g_ = orig[orig_pos_ + 1];
- const unsigned char b_ = orig[orig_pos_ + 0];
-
- // 1, 0 -- ax_, ay
- const int orig_pos__ = py * orig_pitch + px_ * orig_depth;
- const unsigned char r__ = orig[orig_pos__ + 2];
- const unsigned char g__ = orig[orig_pos__ + 1];
- const unsigned char b__ = orig[orig_pos__ + 0];
-
- // 0, 1 -- ax, ay_
- const int orig_pos___ = py_ * orig_pitch + px * orig_depth;
- const unsigned char r___ = orig[orig_pos___ + 2];
- const unsigned char g___ = orig[orig_pos___ + 1];
- const unsigned char b___ = orig[orig_pos___ + 0];
-
- const unsigned char a1 = orig[orig_pos + 3];
- const unsigned char a2 = orig[orig_pos_ + 3];
- const unsigned char a3 = orig[orig_pos__ + 3];
- const unsigned char a4 = orig[orig_pos___ + 3];
-
- const int pos = y * dest_pitch + x * dest_depth;
-
- //qDebug() << "pos" << fx << fy << "uv" << ax << ay;
-
- dest[pos + 0] = (r * ax + r__ * ax_) * ay + (r___ * ax + r_ * ax_) * ay_;
- dest[pos + 1] = (g * ax + g__ * ax_) * ay + (g___ * ax + g_ * ax_) * ay_;
- dest[pos + 2] = (b * ax + b__ * ax_) * ay + (b___ * ax + b_ * ax_) * ay_;
- dest[pos + 3] = (a1 * ax + a3 * ax_) * ay + (a4 * ax + a2 * ax_) * ay_;
-
- break;
- }
+ vec2 uv;
+ int i;
+ if (triangles[0].barycentric_coords(pos, uv, i))
+ {
+ const float fx = origs[i][0].x()
+ + uv.x() * (origs[i][2].x() - origs[i][0].x())
+ + uv.y() * (origs[i][1].x() - origs[i][0].x());
+ const float fy = origs[i][0].y()
+ + uv.x() * (origs[i][2].y() - origs[i][0].y())
+ + uv.y() * (origs[i][1].y() - origs[i][0].y());
+
+ const int px_ = fx + .5f;
+ const int py_ = fy + .5f;
+ const int px = fx;
+ const int py = fy;
+ const float ax_ = fx - px;
+ const float ay_ = fy - py;
+ const float ax = 1.f - ax_;
+ const float ay = 1.f - ay_;
+
+ // 0, 0 -- ax, ay
+ const int orig_pos = py * orig_pitch + px * orig_depth;
+ const unsigned char r = orig[orig_pos + 2];
+ const unsigned char g = orig[orig_pos + 1];
+ const unsigned char b = orig[orig_pos + 0];
+
+ // 1, 1 -- ax_, ay_
+ const int orig_pos_ = py_ * orig_pitch + px_ * orig_depth;
+ const unsigned char r_ = orig[orig_pos_ + 2];
+ const unsigned char g_ = orig[orig_pos_ + 1];
+ const unsigned char b_ = orig[orig_pos_ + 0];
+
+ // 1, 0 -- ax_, ay
+ const int orig_pos__ = py * orig_pitch + px_ * orig_depth;
+ const unsigned char r__ = orig[orig_pos__ + 2];
+ const unsigned char g__ = orig[orig_pos__ + 1];
+ const unsigned char b__ = orig[orig_pos__ + 0];
+
+ // 0, 1 -- ax, ay_
+ const int orig_pos___ = py_ * orig_pitch + px * orig_depth;
+ const unsigned char r___ = orig[orig_pos___ + 2];
+ const unsigned char g___ = orig[orig_pos___ + 1];
+ const unsigned char b___ = orig[orig_pos___ + 0];
+
+ const unsigned char a1 = orig[orig_pos + 3];
+ const unsigned char a2 = orig[orig_pos_ + 3];
+ const unsigned char a3 = orig[orig_pos__ + 3];
+ const unsigned char a4 = orig[orig_pos___ + 3];
+
+ const int pos = y * dest_pitch + x * dest_depth;
+
+ dest[pos + 0] = (r * ax + r__ * ax_) * ay + (r___ * ax + r_ * ax_) * ay_;
+ dest[pos + 1] = (g * ax + g__ * ax_) * ay + (g___ * ax + g_ * ax_) * ay_;
+ dest[pos + 2] = (b * ax + b__ * ax_) * ay + (b___ * ax + b_ * ax_) * ay_;
+ dest[pos + 3] = (a1 * ax + a3 * ax_) * ay + (a4 * ax + a2 * ax_) * ay_;
}
}
image = texture;
@@ -252,3 +279,4 @@ GLWidget::vec3 GLWidget::project2(const vec3 &point)
{
return rotation * point;
}
+
diff --git a/pose-widget/glwidget.h b/pose-widget/glwidget.h
index 25823985..0d48ede9 100644
--- a/pose-widget/glwidget.h
+++ b/pose-widget/glwidget.h
@@ -31,7 +31,7 @@ public:
~GLWidget();
void rotateBy(float xAngle, float yAngle, float zAngle, float x, float y, float z);
protected:
- void paintEvent ( QPaintEvent * event ) override;
+ void paintEvent (QPaintEvent *event) override;
private:
vec2 project(const vec3& point);
vec3 project2(const vec3& point);