How to draw a tapered arc (curve with decreasing thickness) in OpenGL?

1.5k Views Asked by At

I have the following code to draw an arbitrary arc:

void CenteredArc::drawPolygonArc(float radius, float thickness, float startAngle, float arcAngle) {
    float num_segments = 360.0;

    float radiusOuter = radius + thickness / 2;
    float radiusInner = radius - thickness / 2;
    float theta = arcAngle / num_segments; 
    float tangetial_factor = tanf(theta);//calculate the tangential factor 

    float radial_factor = cosf(theta);//calculate the radial factor 

    float xxOuter = radiusOuter * cosf(startAngle);
    float yyOuter = radiusOuter * sinf(startAngle);
    float xxInner = radiusInner * cosf(startAngle);
    float yyInner = radiusInner * sinf(startAngle);  

    float prevXXOuter = -1;
    float prevYYOuter = -1;
    float prevXXInner = -1;
    float prevYYInner = -1;

    glPolygonMode(GL_FRONT, GL_FILL);
    for(int ii = 0; ii < num_segments; ii++) 
    { 
        if (prevXXOuter != -1) {
            glBegin(GL_POLYGON);
                glVertex2f(prevXXOuter, prevYYOuter);
                glVertex2f(xxOuter,     yyOuter);
                glVertex2f(xxInner,     yyInner);
                glVertex2f(prevXXInner, prevYYInner);
            glEnd();
        }

        //calculate the tangential vector 
        //remember, the radial vector is (x, y) 
        //to get the tangential vector we flip those coordinates and negate one of them 

        float txOuter = -yyOuter; 
        float tyOuter =  xxOuter; 
        float txInner = -yyInner; 
        float tyInner =  xxInner; 

        //add the tangential vector 

        prevXXOuter = xxOuter;
        prevYYOuter = yyOuter;
        prevXXInner = xxInner;
        prevYYInner = yyInner;

        xxOuter += txOuter * tangetial_factor; 
        yyOuter += tyOuter * tangetial_factor; 
        xxInner += txInner * tangetial_factor; 
        yyInner += tyInner * tangetial_factor; 

        //correct using the radial factor 
        xxOuter *= radial_factor; 
        yyOuter *= radial_factor; 
        xxInner *= radial_factor; 
        yyInner *= radial_factor; 
    }
}

However, I would like for the arc to start off with the specified thickness on one end and gradually decrease to a thickness of zero on the other end. Any suggestions?

Edit: I am not using GL_LINE_STRIP because I am trying to avoid having overlapping lines and gaps like so:

enter image description here

1

There are 1 best solutions below

7
On

I would use a line strip with decreasing glLineWidth.

This is my implementation, it doesn't gradially reduce the lineWidth but it could be modified to do so. Sorry for the extra stuff, it's from my game engine.

for(int i=0;i<arcs().size();i++)
{
        Entities::Arc temp = arcs().at(i);
        glLineWidth(temp.LW.value); // change LWidth

        glColor3f( temp.CL.R, temp.CL.G, temp.CL.B );

        // theta is now calculated from the arc angle instead, the
        // - 1 part comes from the fact that the arc is open
        float theta = temp.A.value*DEG2RAD / float(WW_SPLINE_ACCURACY - 1);

        float tan = tanf(theta);
        float cos = cosf(theta);

        // we are now at the starting angle
        double x = temp.R.value * cosf(temp.A.value*DEG2RAD);
        double y = temp.R.value * sinf(temp.A.value*DEG2RAD);

        // since the arc is not a closed curve, this is a strip now
        glBegin(GL_LINE_STRIP);
        for(int ii = 0; ii < WW_SPLINE_ACCURACY; ii++)
        {
            glVertex2d(x + temp.C.X, y + temp.C.Y);
            double tx = -y;
            double ty = x;
            x += tx * tan;
            y += ty * tan;
            x *= cos;
            y *= cos; //y = ( y + (ty*tan) )*cos;
        }
    glEnd();
    glLineWidth(WW_DEFAULT_LWIDTH); // reset LWidth
}

I also used these values

#define WW_SPLINE_ACCURACY  72 // 72 for extra smooth arcs/circles, 32 minimum
#define WW_BEZIER_ACCURACY  20

/* Math stuff */
#define DEG2RAD 3.14159/180
#define PI  3.1415926535897932384626433832795;

...

glDisable(GL_TEXTURE_2D);
glDisable(GL_DEPTH_TEST);
glDisable(GL_COLOR_MATERIAL);
glEnable (GL_LINE_SMOOTH);
glEnable (GL_BLEND);
//glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
glHint (GL_LINE_SMOOTH_HINT, GL_DONT_CARE);
glEnable(GL_POLYGON_SMOOTH);
glClearColor(0.188f, 0.169f, 0.329f, 1.0f); //#302b54

I'm not allowed to release the full source since I wrote it for a company but sharing a part or two wont hurt anybody :D