/******************************************************************************* * You may amend and distribute as you like, but don't remove this header! * * EPPlus provides server-side generation of Excel 2007/2010 spreadsheets. * See http://www.codeplex.com/EPPlus for details. * * Copyright (C) 2011 Jan Källman * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU Lesser General Public License for more details. * * The GNU Lesser General Public License can be viewed at http://www.opensource.org/licenses/lgpl-license.php * If you unfamiliar with this license or have questions about it, here is an http://www.gnu.org/licenses/gpl-faq.html * * All code and executables are provided "as is" with no warranty either express or implied. * The author accepts no liability for any damage or loss of business that this product may cause. * * Code change notes: * * Author Change Date * ****************************************************************************** * Jan Källman Added 21-MAR-2011 * Jan Källman License changed GPL-->LGPL 2011-12-16 *******************************************************************************/ using System; using System.Collections.Generic; using System.Text; using System.Xml; using System.IO.Packaging; using System.Linq; namespace OfficeOpenXml.Table.PivotTable { public enum eSourceType { /// /// Indicates that the cache contains data that consolidates ranges. /// Consolidation, /// /// Indicates that the cache contains data from an external data source. /// External, /// /// Indicates that the cache contains a scenario summary report /// Scenario, /// /// Indicates that the cache contains worksheet data /// Worksheet } /// /// Cache definition. This class defines the source data. Note that one cache definition can be shared between many pivot tables. /// public class ExcelPivotCacheDefinition : XmlHelper { internal ExcelPivotCacheDefinition(XmlNamespaceManager ns, ExcelPivotTable pivotTable) : base(ns, null) { foreach (var r in pivotTable.Part.GetRelationshipsByType(ExcelPackage.schemaRelationships + "/pivotCacheDefinition")) { Relationship = r; } CacheDefinitionUri = PackUriHelper.ResolvePartUri(Relationship.SourceUri, Relationship.TargetUri); var pck = pivotTable.WorkSheet._package.Package; Part = pck.GetPart(CacheDefinitionUri); CacheDefinitionXml = new XmlDocument(); LoadXmlSafe(CacheDefinitionXml, Part.GetStream()); TopNode = CacheDefinitionXml.DocumentElement; PivotTable = pivotTable; if (CacheSource == eSourceType.Worksheet) { var worksheetName = GetXmlNodeString(_sourceWorksheetPath); if (pivotTable.WorkSheet.Workbook.Worksheets.Any(t => t.Name == worksheetName)) { _sourceRange = pivotTable.WorkSheet.Workbook.Worksheets[worksheetName].Cells[GetXmlNodeString(_sourceAddressPath)]; } } } internal ExcelPivotCacheDefinition(XmlNamespaceManager ns, ExcelPivotTable pivotTable, ExcelRangeBase sourceAddress, int tblId) : base(ns, null) { PivotTable = pivotTable; var pck = pivotTable.WorkSheet._package.Package; //CacheDefinition CacheDefinitionXml = new XmlDocument(); LoadXmlSafe(CacheDefinitionXml, GetStartXml(sourceAddress), Encoding.UTF8); CacheDefinitionUri = GetNewUri(pck, "/xl/pivotCache/pivotCacheDefinition{0}.xml", tblId); Part = pck.CreatePart(CacheDefinitionUri, ExcelPackage.schemaPivotCacheDefinition); TopNode = CacheDefinitionXml.DocumentElement; //CacheRecord. Create an empty one. CacheRecordUri = GetNewUri(pck, "/xl/pivotCache/pivotCacheRecords{0}.xml", tblId); var cacheRecord = new XmlDocument(); cacheRecord.LoadXml(""); var recPart = pck.CreatePart(CacheRecordUri, ExcelPackage.schemaPivotCacheRecords); cacheRecord.Save(recPart.GetStream()); RecordRelationship = Part.CreateRelationship(PackUriHelper.ResolvePartUri(CacheDefinitionUri, CacheRecordUri), TargetMode.Internal, ExcelPackage.schemaRelationships + "/pivotCacheRecords"); RecordRelationshipID = RecordRelationship.Id; CacheDefinitionXml.Save(Part.GetStream()); } /// /// Reference to the internal package part /// internal PackagePart Part { get; set; } /// /// Provides access to the XML data representing the cache definition in the package. /// public XmlDocument CacheDefinitionXml { get; private set; } /// /// The package internal URI to the pivottable cache definition Xml Document. /// public Uri CacheDefinitionUri { get; internal set; } internal Uri CacheRecordUri { get; set; } internal PackageRelationship Relationship { get; set; } internal PackageRelationship RecordRelationship { get; set; } internal string RecordRelationshipID { get { return GetXmlNodeString("@r:id"); } set { SetXmlNodeString("@r:id", value); } } /// /// Referece to the PivoTable object /// public ExcelPivotTable PivotTable { get; private set; } const string _sourceWorksheetPath="d:cacheSource/d:worksheetSource/@sheet"; const string _sourceAddressPath = "d:cacheSource/d:worksheetSource/@ref"; internal ExcelRangeBase _sourceRange = null; /// /// The source data range when the pivottable has a worksheet datasource. /// The number of columns in the range must be intact if this property is changed. /// The range must be in the same workbook as the pivottable. /// public ExcelRangeBase SourceRange { get { if (_sourceRange == null) { if (CacheSource == eSourceType.Worksheet) { var ws = PivotTable.WorkSheet.Workbook.Worksheets[GetXmlNodeString(_sourceWorksheetPath)]; if (ws != null) { _sourceRange = ws.Cells[GetXmlNodeString(_sourceAddressPath)]; } } else { throw (new ArgumentException("The cachesource is not a worksheet")); } } return _sourceRange; } set { if (PivotTable.WorkSheet.Workbook != value.Worksheet.Workbook) { throw (new ArgumentException("Range must be in the same package as the pivottable")); } var sr=SourceRange; if (value.End.Column - value.Start.Column != sr.End.Column - sr.Start.Column) { throw (new ArgumentException("Can not change the number of columns(fields) in the SourceRange")); } SetXmlNodeString(_sourceWorksheetPath, value.Worksheet.Name); SetXmlNodeString(_sourceAddressPath, value.FirstAddress); _sourceRange = value; } } /// /// Type of source data /// public eSourceType CacheSource { get { var s=GetXmlNodeString("d:cacheSource/@type"); if (s == "") { return eSourceType.Worksheet; } else { return (eSourceType)Enum.Parse(typeof(eSourceType), s, true); } } } private string GetStartXml(ExcelRangeBase sourceAddress) { string xml=""; xml += ""; xml += string.Format(" ", sourceAddress.Address, sourceAddress.WorkSheet); xml += ""; xml += string.Format("", sourceAddress._toCol - sourceAddress._fromCol + 1); var sourceWorksheet = PivotTable.WorkSheet.Workbook.Worksheets[sourceAddress.WorkSheet]; for (int col = sourceAddress._fromCol; col <= sourceAddress._toCol; col++) { if (sourceWorksheet == null || sourceWorksheet.Cell(sourceAddress._fromRow, col).Value == null || sourceWorksheet.Cell(sourceAddress._fromRow, col).Value.ToString().Trim() == "") { xml += string.Format("", col - sourceAddress._fromCol + 1); } else { xml += string.Format("", sourceWorksheet.Cell(sourceAddress._fromRow, col).Value); } //xml += " "; xml += " "; xml += ""; } xml += ""; xml += ""; return xml; } } }