首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >用c#图到LiveCharts图像

用c#图到LiveCharts图像
EN

Stack Overflow用户
提问于 2017-10-25 12:38:34
回答 3查看 9.2K关注 0票数 2

我有个小问题。我想把我的图表导出到图片中。我知道,用beto (这里)提供的代码是可能的。

但是我有一个问题,我不能通过在屏幕上显示它来得到我能拥有的东西。见上面的图片。在右下角,我将图像保存在png中。在图像的左边,我有一个图表,所有的参数都显示在我想要的地方。

您知道在保存的图像中是否可能有相同的图表(左侧)吗?实际上,我使用一个线程自动捕获(复制/粘贴)图像。

你知道我必须设置哪些参数才能得到正确的图像吗?

双图

提前谢谢。致以问候。

*

抱歉我迟到了。我试过你说的话,但不知怎么说,我做不到我想要的。我将解释我所做的事情:我有一个类"MakeReport“,它可以用以下方式调用另一个类"GraphMaker”:

代码语言:javascript
运行
复制
MyGraph = new GraphMaker();
MyGraph = GiveAGraph(MyGraph);

类"MakeReport“将使用子程序”GiveAGraph()“完成带有一些值的图表:

代码语言:javascript
运行
复制
GraphData.SeriesCollection.Add(new RowSeries
   {
       Title = Criteria, 
       Values = DataValues,
       ScalesYAt = Index,
       DataLabels = true
    });

"GiveAGRaph()“结尾。

现在,我有了一个可以显示的图表,我想用它来制作一个图像,并且为了测试(和调试)我展示它:

代码语言:javascript
运行
复制
// Chart to Image
MyGraph.GiveMeAnImageFromChart();      <-- to make a picture with the chart
// Show the graph
MyGraph.Show();                        <-- for debug and test, i display it.

对于"MyGraph.GiveAnImageFromChart()",我没有与"MyGraph.Show()“相同的结果。图片(另存为png)与显示的图表不同。

包含在"GiveAnImageFromChart“类中的子程序"GraphMaker”是:

代码语言:javascript
运行
复制
public void GiveMeAnImageFromChart()
    {
            var viewbox = new Viewbox();
            myChart.Background = Brushes.White;
            myChart.DataContext = this;

            // myChart  il faut tout mettre en paramètres

            viewbox.Child = myChart;
            viewbox.Measure(myChart.RenderSize);
            viewbox.Arrange(new Rect(new Point(0, 0), myChart.RenderSize));

            myChart.Update(true, true); //force chart redraw
            viewbox.UpdateLayout();

            SaveToPng(myChart, "chart.png");
            //png file was created at the root directory.

    }

"myChart“变量是公共的。使用的"SaveToPng“程序来自您的示例(这里)。

为了保存图片,我尝试使用以下方法:

代码语言:javascript
运行
复制
public System.Drawing.Bitmap ControlToImage(Visual target, double dpiX, double dpiY)
    {
        if (target == null)
        {
            return null;
        }
        // render control content
        Rect bounds = VisualTreeHelper.GetDescendantBounds(target);
        Console.WriteLine("Bounds width = " + bounds.Width + " et bounds height = " + bounds.Height);
        RenderTargetBitmap rtb = new RenderTargetBitmap((int)(bounds.Width * dpiX / 96.0),
                                                        (int)(bounds.Height * dpiY / 96.0),
                                                        dpiX,
                                                        dpiY,
                                                        PixelFormats.Pbgra32);
        DrawingVisual dv = new DrawingVisual();
        using (DrawingContext ctx = dv.RenderOpen())
        {
            VisualBrush vb = new VisualBrush(target);
            ctx.DrawRectangle(vb, null, new Rect(new System.Windows.Point(), bounds.Size));
        }
        rtb.Render(dv);

        //convert image format
        MemoryStream stream = new MemoryStream();
        BitmapEncoder encoder = new BmpBitmapEncoder();
        encoder.Frames.Add(BitmapFrame.Create(rtb));
        encoder.Save(stream);

        return new System.Drawing.Bitmap(stream);
    }

有:

代码语言:javascript
运行
复制
Bitmap ImageChart = MyGraph.ControlToImage(MyGraph, MyGraph.Width, MyGraph.Height);

使用此方法,我有一个错误,因为"bounds.Width“和"bounds.Height”等于-8。最后,我没有任何图表可以转换。

我想我在"ControlImage“中给出了一个错误的”视觉目标“,我错过了一些东西,但我不知道是什么。如果你需要更多的信息,问我。提前谢谢你的帮助。

PS:对不起我的英语。别犹豫要纠正我。

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2017-12-14 09:38:16

多亏了bto-rdz,我找到了解决办法。我发现我没有正确地使用Livecharts。所以如果有人需要的话我会发布我的代码。

这是我的GraphMaker.xaml文件:

代码语言:javascript
运行
复制
<UserControl x:Class="MyProject.GraphMaker"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
        xmlns:local="clr-namespace:MyProject"
        mc:Ignorable="d" 
        d:DesignHeight="730" d:DesignWidth="1660">
<Grid>
</Grid>
</UserControl>

还有我的GraphMaker.xaml.cs文件

代码语言:javascript
运行
复制
    using System;
using System.IO;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using LiveCharts;
using LiveCharts.Wpf;
using System.Collections.Generic;
using System.ComponentModel;


namespace MyProject
{    
    public partial class GraphMaker : UserControl
    {
        // PUBLIC
        public CartesianChart MyTestChart;
        public SeriesCollection MySeriesCollection { get; set; }
        public string[] Labels { get; set; }
        public string AxisTitle { get; set; }
        public Func<double, string> YFormatter { get; set; }
        public Axis Axis1, Axis2, Axis3, Axis4, Axis5, Axis6, Axis7, Axis8, Axis9, Axis10, AxisXChart;


        public GraphMaker()
        {
            InitializeComponent();
            MySeriesCollection = new SeriesCollection();

            MyTestChart = new CartesianChart
            {
                DisableAnimations = true,
                Width = 1600,
                Height = 700,
                Series = MySeriesCollection
            };

            MyTestChart.LegendLocation = LegendLocation.Right;

            // *** Axis 1 ***
            Axis1 = new Axis();
            Axis1.Foreground = Brushes.DodgerBlue;
            Axis1.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis1.LabelFormatter = YFormatter;
            MyTestChart.AxisY.Add(Axis1);

            // *** Axis 2 ***
            Axis2 = new Axis();
            Axis2.Foreground = Brushes.IndianRed;
            Axis2.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis2.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis2);

            // *** Axis 3 ***
            Axis3 = new Axis();
            Axis3.Foreground = Brushes.Gold;
            Axis3.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis3.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis3);

            // *** Axis 4 ***
            Axis4 = new Axis();
            Axis4.Foreground = Brushes.Gray;
            Axis4.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis4.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis4);

            // *** Axis 5 ***
            Axis5 = new Axis();
            Axis5.Foreground = Brushes.DeepSkyBlue;
            Axis5.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis5.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis5);

            // *** Axis 6 ***
            Axis6 = new Axis();
            Axis6.Foreground = Brushes.HotPink;
            Axis6.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis6.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis6);

            // *** Axis 7 ***
            Axis7 = new Axis();
            Axis7.Foreground = Brushes.Orange;
            Axis7.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis7.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis7);

            // *** Axis 8 ***
            Axis8 = new Axis();
            Axis8.Foreground = Brushes.RoyalBlue;
            Axis8.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis8.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis8);

            // *** Axis 9 ***
            Axis9 = new Axis();
            Axis9.Foreground = Brushes.Black;
            Axis9.Position = AxisPosition.RightTop;
            Axis9.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis9);

            // *** Axis 10 ***
            Axis10 = new Axis();
            Axis10.Foreground = Brushes.DarkTurquoise;
            Axis10.Position = AxisPosition.RightTop;
            YFormatter = value => value.ToString("N2");
            Axis10.LabelFormatter = YFormatter;
            //MyTestChart.AxisY.Add(Axis10);

            AxisXChart = new Axis();
            AxisXChart.Title = AxisTitle;
            AxisXChart.Labels = Labels;
        }


        public void TakeTheChart()
        {
            var viewbox = new Viewbox();
            viewbox.Child = MyTestChart;
            viewbox.Measure(MyTestChart.RenderSize);
            viewbox.Arrange(new Rect(new Point(0, 0), MyTestChart.RenderSize));
            MyTestChart.Update(true, true); //force chart redraw
            viewbox.UpdateLayout();

            SaveToPng(MyTestChart, "Chart.png");
            //png file was created at the root directory.
        }

        public void SaveToPng(FrameworkElement visual, string fileName)
        {
            var encoder = new PngBitmapEncoder();
            EncodeVisual(visual, fileName, encoder);
        }

        private static void EncodeVisual(FrameworkElement visual, string fileName, BitmapEncoder encoder)
        {
            var bitmap = new RenderTargetBitmap((int)visual.ActualWidth, (int)visual.ActualHeight, 96, 96, PixelFormats.Pbgra32);
            bitmap.Render(visual);
            var frame = BitmapFrame.Create(bitmap);
            encoder.Frames.Add(frame);
            using (var stream = File.Create(fileName)) encoder.Save(stream);
        }
    }
}

希望能帮助你。

祝您今天愉快。再见。

票数 3
EN

Stack Overflow用户

发布于 2017-10-26 00:55:58

您指出的示例在内存中创建了一个新的图表实例,这就是为什么您要获得不同的图像,您可以在这两个图表中再现相同的属性,即在UI中显示的属性,以及在内存中创建的属性,或者您只需在UI中打印当前实例,如本文所述:

https://github.com/beto-rodriguez/Live-Charts/issues/243

希望它能帮上忙

票数 1
EN

Stack Overflow用户

发布于 2018-12-20 07:23:56

我已经写了扩展方法,以保存为图像从笛卡儿图表窗口窗体。它可以编辑你自己的方式得到最好的。

代码语言:javascript
运行
复制
        public static bool ChartToImage(this LiveCharts.WinForms.CartesianChart cartesianChart, LineSeries data, Axis AxisX, Axis AxisY,double Width, double Height, string fileName, string targetPath, out string locationOfImage, out Exception returnEx)
    {

        bool status = false;
        returnEx = null;
        locationPath = null;
        try
        {
            var myChart = new LiveCharts.Wpf.CartesianChart
            {
                DisableAnimations = true,
                Width = Width,
                Height = Height,
                Series = new SeriesCollection(cartesianChart.Series.Configuration)
                {
                   new LineSeries
                   {
                       Title = data.Title,
                       LineSmoothness = data.LineSmoothness,
                       StrokeThickness = data.StrokeThickness,
                       PointGeometrySize = data.PointGeometrySize,
                       Stroke = data.Stroke,
                       Values=data.Values
                   }
                }

            };
            myChart.AxisX.Add(new Axis { IsMerged = AxisX.IsMerged, FontSize = AxisX.FontSize, FontWeight = AxisX.FontWeight, Foreground = AxisX.Foreground, Separator = new LiveCharts.Wpf.Separator { Step = AxisX.Separator.Step, StrokeThickness = AxisX.Separator.StrokeThickness, StrokeDashArray = AxisX.Separator.StrokeDashArray, Stroke = AxisX.Separator.Stroke }, Title = AxisX.Title, MinValue = AxisX.MinValue, MaxValue = AxisX.MaxValue });
            myChart.AxisY.Add(new Axis { IsMerged = AxisY.IsMerged, FontSize = AxisY.FontSize, FontWeight = AxisY.FontWeight, Foreground = AxisY.Foreground, Separator = new LiveCharts.Wpf.Separator { Step = AxisY.Separator.Step, StrokeThickness = AxisY.Separator.StrokeThickness, StrokeDashArray = AxisY.Separator.StrokeDashArray, Stroke = AxisX.Separator.Stroke }, Title = AxisY.Title, MinValue = AxisY.MinValue, MaxValue = AxisY.MaxValue });


            var viewbox = new Viewbox();
            viewbox.Child = myChart;
            viewbox.Measure(myChart.RenderSize);
            viewbox.Arrange(new Rect(new System.Windows.Point(0, 0), myChart.RenderSize));
            myChart.Update(true, true); //force chart redraw
            viewbox.UpdateLayout();

            var encoder = new PngBitmapEncoder();
            var bitmap = new RenderTargetBitmap((int)myChart.ActualWidth, (int)myChart.ActualHeight, 96, 96, PixelFormats.Pbgra32);
            bitmap.Render(myChart);
            var frame = BitmapFrame.Create(bitmap);
            encoder.Frames.Add(frame);
            string path = Path.Combine(targetPath, fileName);
            using (var stream = File.Create(path))
            {
                encoder.Save(stream);
                locationPath = path;
            }
            myChart = null;
            viewbox = null;
            encoder = null;
            bitmap = null;
            frame = null;
            status = true;
        }
        catch (Exception ex)
        {
            returnEx = ex;
            status = false;
        }
        return status;
    }

在你的主要节目中:

代码语言:javascript
运行
复制
                    if(cartesianChart1.ChartToImage(data,axisX,axisY,600,200,"test.png", locationImage, out string locationOfFile, out Exception returnEx))
                {
                    System.Windows.Forms.MessageBox.Show(locationOfFile);
                }
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/46932672

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档