转载请注明作者及出处,谢谢

上文提到了等值线追踪解决方案,在此基础上,我们就可以把等值线画出来了,但是只光秃秃的线条,没有标注还是不行的,别人哪知道那条像蚯蚓一样的线条代表什么呢,本文我们就来讨论下如何在等值线上进行标注。

感谢《等值线标注的一种算法探讨》一文的作者,我正在是使用这篇论文中的重要算法指导了我的工作。

首先标注那些小的封闭式的等值线。

这里我也没有想出来好的方法,就使用方法,找出封闭式等值线中点坐标X最小值,Y最小值,X最大值及最大值;如果XMax - XMin < 指定值以及YMax - YMin < 指定值,则在P((XMin + XMax) / 2,(YMin + YMax) / 2)处把等值线的值画出来,如下代码所示:

if (points[0].X == points[points.Count - 1].X && points[0].Y == points[points.Count - 1].Y)
{
    float xMin = points.Min<VPoint>(p => p.X), xMax = points.Max<VPoint>(p => p.X), yMin = points.Min<VPoint>(p => p.Y), yMax = points.Max<VPoint>(p => p.Y);
    if (xMax - xMin < 25 && yMax - yMin < 25)
    {
        g.DrawString(value.ToString(), font, brush, (xMax + xMin - sf.Width) / 2f, (yMax + yMin - sf.Height) / 2f);
        continue;
    }
}

效果如图中红圈所示:

图1

把过于小的封闭式等值线标注后,大点的封闭式和开放式等值线的标注方法就一样了。

根据《等值线标注的一种算法探讨》一文中的指导思想,需要把曲线转换为折线或是多边形,啥意思咧,看下图所示:

如果我告诉你,现在如果每条线段的长度大于高程值的字符串(value.ToString())所需的长度 - 空白长度,就在那里画一个标注应该就问题不大了吧,对于那些太小的线段,就不要去画了,小到不像话的封闭式等值线,我们在上一步中已经处理过了。

那么如何把一条曲线转换为折线或是多边形的呢?《等值线标注的一种算法探讨》一文中告诉我们(见论文2.2):“设一个dif参数,用来控制多边形近似等值线的误差,将曲线上的点从第0个开始,偶数点相连作为多边形的边,依次查看等值线上的3个点,如果中间的等值点(奇数点)到多边形的边的距离小于参数dif,舍弃中间的等值点,否则保存中间的点。如此不断循环,直到最后生成的多边形的边数不在(再 think8848注)减少为止。这样就得到了等值线近似的多边形。”dif越大,表明曲线越陡,dif越小,表明曲线越平缓。

算法代码示例:

int n = (points[0].X != points[points.Count - 1].X && points[0].Y != points[points.Count - 1].Y) ? points.Count : points.Count - 1,
    minEdge = n + 1, k = n;
float tolerance = 8f;
var indexes = new List<int>(); for (int i = 0; i < points.Count; i++) { indexes.Add(i); }
while (k < minEdge)
{
    minEdge = k;
    var p = 0;
    while (p < minEdge - (n % 2 == 0 ? 3 : 2))
    {
        float straight = this.GetPointToStraight(points[indexes[p + 1]], points[indexes[p]], points[indexes[p + 2]]);
        if (Math.Abs(straight) < tolerance)
        {
            indexes[p + 1] = -1;
            k -= 1;
        }
        p++; p++;
    }
    indexes = indexes.Where<int>(index => index != -1).ToList<int>();
}

另附求点到直线的距离的方法,下列代表示例如何计算p点到直线p1p2最短距离

private float GetPointToStraight(VPoint p, VPoint p1, VPoint p2)
{
    if (p1.X == p2.X)
    {
        return (float)Math.Abs(p1.Y - p2.Y);
    }
    if (p1.Y == p2.Y)
    {
        return (float)Math.Abs(p1.X - p2.X);
    }
    double k = (p2.Y - p1.Y) / (p2.X - p1.X);
    double c = (p2.X * p1.Y - p1.X * p2.Y) / (p2.X - p1.X);
    return (float)((k * p.X - p.Y + c) / (Math.Sqrt(k * k + 1)));
}

最终indexes里面保存了在等值点列表中构成多边形的点的索引。

在本文的最后,我们再来谈谈如果标注字符旋转的问题,有一条线段作为基准,将画布旋转与线段倾斜角度相同度数应该不是一件难事,是的,.NET很容易就能做到,唯一的问题是如何线段的倾斜角度:

var alpha = (float)(Math.Atan((p2.Y - p1.Y) / (p2.X - p1.X)) * 180 / Math.PI);

就是这个角度了,三角函数已经忘了的兄弟可以到网上重新温习下高中数学,呵呵。

g.TranslateTransform(xOffset, yOffset);
g.RotateTransform(alpha);
g.DrawString(value.ToString(), font, brush, new PointF(0, 0));
g.RotateTransform(-alpha);
g.TranslateTransform(-xOffset, -yOffset);

变换坐标系的原点到将要绘制标注的左上角位置,然后旋转画布,(注意,角度为正时为顺时针,角度为负时为逆时针方向,好像和我们数学课中的方向不一致。)绘制完成后,再把坐标系归位,循环,直至将所有的标注都绘制完成。

至此,使用C#绘制等值线的工作基本完成。

posted on   think8848  阅读(6409)  评论(1编辑  收藏  举报
努力加载评论中...
点击右上角即可分享
微信分享提示