C#管道控件,中间管道填充渐变色与预想的不符?

画一个如下图的管道控件,要求在设计器里可以拉伸改变管道的倾斜度,但是目前中间的管道渐变颜色一直控制不好,下面跟管弯对不上,求大佬助我!

我自己研究了下,稍微有点思路:

我画了个三角形找到管道直管段的垂线黄色线段(事后证明不是垂线=。=),拉伸的时候渐变位置变化了。

现在的问题是如何能找到上图两条平行线的垂线?然后算出垂线跟平行线的焦点坐标,以这2个坐标填充渐变应该可以保证拉伸的情况下保持渐变不变形。

附上控件代码,求大佬帮我改下代码,感激不尽!!!

public partial class PipeLine : UserControl
{
    public PipeLine()
    {
        InitializeComponent();
        SetStyle(ControlStyles.UserPaint | ControlStyles.SupportsTransparentBackColor, value: true);
        SetStyle(ControlStyles.OptimizedDoubleBuffer, value: true);
        SetStyle(ControlStyles.AllPaintingInWmPaint, value: true);
    }
    protected override void OnPaint(PaintEventArgs e)
    {
        e.Graphics.SmoothingMode = SmoothingMode.HighQuality;
        base.OnPaint(e);
        Graphics graphics = e.Graphics;
 
        int pipeLineWidth = 50;//管道宽度
        ColorBlend colorBlend = new ColorBlend();
        colorBlend.Positions = new float[3] { 0f, 0.5f, 1f };
        colorBlend.Colors = new Color[3] { Color.DimGray, Color.LightGray, Color.DimGray };
        GraphicsPath path = new GraphicsPath();
 
        double angleUp = Math.Atan((ClientRectangle.Height - pipeLineWidth) / (double)ClientRectangle.Width);
        angleUp = angleUp / Math.PI * 180f;
 
        path.AddLine(new Point(ClientRectangle.Right, ClientRectangle.Top + pipeLineWidth), new Point((int)(ClientRectangle.Right + 1 - Math.Sin(Math.PI * (angleUp / 180.00F)) * pipeLineWidth), (int)(ClientRectangle.Top + pipeLineWidth - Math.Cos(Math.PI * (angleUp / 180.00F)) * pipeLineWidth)));
        path.AddLine(new Point(ClientRectangle.Left, ClientRectangle.Bottom - pipeLineWidth), new Point((int)(ClientRectangle.Left + 1 + Math.Sin(Math.PI * (angleUp / 180.00F)) * pipeLineWidth), (int)(ClientRectangle.Bottom - pipeLineWidth + Math.Cos(Math.PI * (angleUp / 180.00F)) * pipeLineWidth + 2)));
        path.CloseAllFigures();
 
        LinearGradientBrush line_Brush = new LinearGradientBrush(new Point((int)(ClientRectangle.Right + 1 - Math.Sin(Math.PI * (angleUp / 180.00F)) * pipeLineWidth), (int)(ClientRectangle.Top + pipeLineWidth - Math.Cos(Math.PI * (angleUp / 180.00F)) * pipeLineWidth)), new Point(ClientRectangle.Right, ClientRectangle.Top + pipeLineWidth), Color.DimGray, Color.LightGray);
 
        line_Brush.InterpolationColors = colorBlend;
        graphics.FillPath(line_Brush, path);
 
        //上圆弧
        PaintEllipse(graphics, colorBlend, new Rectangle(ClientRectangle.Right - pipeLineWidth, ClientRectangle.Top, pipeLineWidth * 2, pipeLineWidth * 2), 270, -1 * (float)angleUp);
        //下圆弧
        PaintEllipse(graphics, colorBlend, new Rectangle(ClientRectangle.Left - pipeLineWidth, ClientRectangle.Bottom - pipeLineWidth * 2, pipeLineWidth * 2, pipeLineWidth * 2), 90, -1 * (float)angleUp);
    }
    void PaintEllipse(Graphics g, ColorBlend colorBlend, Rectangle rect, float startAngle, float sweepAngle)
    {
        GraphicsPath graphicsPath = new GraphicsPath();
        graphicsPath.AddEllipse(rect);
        PathGradientBrush pathGradientBrush = new PathGradientBrush(graphicsPath);
        pathGradientBrush.CenterPoint = new Point(rect.X + rect.Width / 2, rect.Y + rect.Height / 2);
        pathGradientBrush.InterpolationColors = colorBlend;
        g.FillPie(pathGradientBrush, rect, startAngle, sweepAngle);
        g.DrawArc(new Pen(Color.DimGray, 1f), rect, startAngle, sweepAngle);
        pathGradientBrush.Dispose();
        graphicsPath.Dispose();
    }
}
阅读 2.5k
1 个回答
✓ 已被采纳新手上路,请多包涵


搞定了,上图橘黄色是LinearGradientBrush的渐变方向。
利用直线斜率找出一条斜边的垂线,然后找到垂线跟两条平行斜边的交点得出渐变坐标点。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进