图表的 SetSourceData 为数据透视表返回 HRESULT E_FAIL (Excel C#)

Posted

技术标签:

【中文标题】图表的 SetSourceData 为数据透视表返回 HRESULT E_FAIL (Excel C#)【英文标题】:SetSourceData for Chart returns HRESULT E_FAIL for PivotTable (Excel C#) 【发布时间】:2017-02-06 01:34:35 【问题描述】:

我想在工作表中创建多个数据透视表,并从每个数据透视表中创建一个图表。 第一个数据透视表和第一个图表是正常创建的。创建第二个数据透视表后,我想添加第二个图表,其中包含来自第二个数据透视表的源,但这比 HRESULT E_FAIL 重新调整,并且第二个图表具有第一个数据透视表的源,我无法更改。

        string pitpivotNam1 = "table1";
        Dictionary<String, Excel.PivotField> pFields = new Dictionary<String, Excel.PivotField>();
        Excel.PivotTable pivotTable1 = null;
        Excel.Range pivotDest = excelApp.Worksheets["PivotTable"].Cells[1, 1];

        excelApp.Worksheets[sheetData].Activate();
        int countColmn = excelApp.Cells[1, 1].End(Excel.XlDirection.xlToRight).Column;
        var headerNam = excelApp.Range[excelApp.Cells[1, 1], excelApp.Cells[1, countColmn]].Value;

        // Data Source for PivotTable
        Excel.Range pivotData = excelApp.Worksheets[sheetData].usedRange;
        Excel.PivotCache pivotcache = excelApp.ActiveWorkbook.PivotCaches().Create(Excel.XlPivotTableSourceType.xlDatabase, pivotData.Address/*, Excel.XlPivotTableVersionList.xlPivotTableVersion15*/);
        excelApp.Worksheets[sheetDest].Activate();
        pivotTable1 = excelApp.Worksheets[sheetDest].PivotTables.Add(pivotcache, pivotDest, pitpivotNam1, useDefault/*, Excel.XlPivotTableVersionList.xlPivotTableVersion15*/);

        // Create Dynamically PivotFields
        for (int i = 1; i <= headerNam.Length - 1; i++)
        

            pFields.Add("test" + i.ToString(), (Excel.PivotField)pivotTable1.PivotFields(i));

        

        pFields["test20"].Orientation = Excel.XlPivotFieldOrientation.xlDataField;
        pFields["test20"].Function = Excel.XlConsolidationFunction.xlAverage;
        pFields["test43"].Orientation = Excel.XlPivotFieldOrientation.xlDataField;
        pFields["test43"].Function = Excel.XlConsolidationFunction.xlAverage;
        pFields["test66"].Orientation = Excel.XlPivotFieldOrientation.xlDataField;
        pFields["test66"].Function = Excel.XlConsolidationFunction.xlAverage;
        pFields["test20"].Name = "DVP & R Progress";

        // First Chart
        excelApp.ActiveSheet.Shapes.AddChart.Select();
        excelApp.ActiveChart.SetSourceData(pivotTable1.TableRange1);

        // Second PivotTable
        string pitpivotNam2 = "table2";
        Excel.PivotTable pivotTable2 = null;
        Dictionary<String, Excel.PivotField> pFields2 = new Dictionary<String, Excel.PivotField>();
        Excel.Range pivotDest2 = excelApp.Worksheets["PivotTable"].Cells[20, 1];

        excelApp.Worksheets[sheetData].Activate();
        int countColmn2 = excelApp.Cells[1, 1].End(Excel.XlDirection.xlToRight).Column;
        var headerNam2 = excelApp.Range[excelApp.Cells[1, 1], excelApp.Cells[1, countColmn2]].Value;

        // Data Source for PivotTable
        Excel.Range pivotData2 = excelApp.Worksheets[sheetData].usedRange;
        Excel.PivotCache pivotcache2 = excelApp.ActiveWorkbook.PivotCaches().Create(Excel.XlPivotTableSourceType.xlDatabase, pivotData2.Address/*, Excel.XlPivotTableVersionList.xlPivotTableVersion15*/);
        excelApp.Worksheets[sheetDest].Activate();
        pivotTable2 = excelApp.Worksheets[sheetDest].PivotTables.Add(pivotcache2, pivotDest2, pitpivotNam2, useDefault/*, Excel.XlPivotTableVersionList.xlPivotTableVersion15*/);

        // Create Dynamically PivotFields
        for (int i = 1; i <= headerNam2.Length - 1; i++)
        

            pFields2.Add("test" + i.ToString(), (Excel.PivotField)pivotTable2.PivotFields(i));

        

        pFields2["test17"].Orientation = Excel.XlPivotFieldOrientation.xlColumnField;
        Excel.PivotField dataField = pivotTable2.DataPivotField;
        pFields2["test17"].Orientation = Excel.XlPivotFieldOrientation.xlDataField;
        pFields2["test17"].Function = Excel.XlConsolidationFunction.xlCount;
        pFields2["test40"].Orientation = Excel.XlPivotFieldOrientation.xlDataField;
        pFields2["test40"].Function = Excel.XlConsolidationFunction.xlCount;
        pFields2["test63"].Orientation = Excel.XlPivotFieldOrientation.xlDataField;
        pFields2["test63"].Function = Excel.XlConsolidationFunction.xlCount;
        dataField.Orientation = Excel.XlPivotFieldOrientation.xlRowField;

        // Secound Chart
        excelApp.ActiveSheet.Shapes.AddChart.Select();
        excelApp.ActiveChart.SetSourceData(pivotTable2.TableRange1);

提前感谢您的帮助。

问候,

丹尼尔

【问题讨论】:

【参考方案1】:

我在 VSTO C# 项目中遇到了类似的问题。

就我而言,我必须在创建图表之前选择数据透视范围。

我发现,如果在创建图表时选择了第一个枢轴的单元格,则图表将绑定到该枢轴并且您无法更改源数据(事实上,您甚至不能这样做在 Excel 中,图表的源范围为灰色)。

我通过创建图表之前选择数据透视单元来解决。

Range rangePivot2 = pivotTable2.TableRange1;
rangePivot2.Select();
Chart chart = sheet.Controls.AddChart(cellsWhereTheChartWillBeDrawn, "secondChart");
chart.SetSourceData(rangePivot2); /* this won't give you E_FAIL... but it's not really needed */

我的代码示例是 VSTO 伪代码,但我相信类似的方法也适用于纯 Excel COM 自动化。

【讨论】:

以上是关于图表的 SetSourceData 为数据透视表返回 HRESULT E_FAIL (Excel C#)的主要内容,如果未能解决你的问题,请参考以下文章

将处理后的数据用数据透视表 生成有价值的图表

excel数据透视表如何生成柱状图表

动态图表制作——数据透视表+函数

如何防止数据透视图成为工作表副本上的常规图表?

2022年最新Python大数据之Excel基础

VBA SetSourceData函数起作用,但给出错误?