|  | /******************************************************************************* | 
|  | * 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.Data.SqlClient; | 
|  | using System.Text; | 
|  | using System.Xml; | 
|  | using System.Linq; | 
|  | using OfficeOpenXml.Utils; | 
|  | namespace OfficeOpenXml.Table.PivotTable | 
|  | { | 
|  | public enum eSourceType | 
|  | { | 
|  | /// <summary> | 
|  | /// Indicates that the cache contains data that consolidates ranges. | 
|  | /// </summary> | 
|  | Consolidation, | 
|  | /// <summary> | 
|  | /// Indicates that the cache contains data from an external data source. | 
|  | /// </summary> | 
|  | External, | 
|  | /// <summary> | 
|  | /// Indicates that the cache contains a scenario summary report | 
|  | /// </summary> | 
|  | Scenario, | 
|  | /// <summary> | 
|  | /// Indicates that the cache contains worksheet data | 
|  | /// </summary> | 
|  | Worksheet | 
|  | } | 
|  | /// <summary> | 
|  | /// Cache definition. This class defines the source data. Note that one cache definition can be shared between many pivot tables. | 
|  | /// </summary> | 
|  | 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 = UriHelper.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("<pivotCacheRecords xmlns=\"http://schemas.openxmlformats.org/spreadsheetml/2006/main\" xmlns:r=\"http://schemas.openxmlformats.org/officeDocument/2006/relationships\" count=\"0\" />"); | 
|  | var recPart = pck.CreatePart(CacheRecordUri, ExcelPackage.schemaPivotCacheRecords); | 
|  | cacheRecord.Save(recPart.GetStream()); | 
|  |  | 
|  | RecordRelationship = Part.CreateRelationship(UriHelper.ResolvePartUri(CacheDefinitionUri, CacheRecordUri), Packaging.TargetMode.Internal, ExcelPackage.schemaRelationships + "/pivotCacheRecords"); | 
|  | RecordRelationshipID = RecordRelationship.Id; | 
|  |  | 
|  | CacheDefinitionXml.Save(Part.GetStream()); | 
|  | } | 
|  | /// <summary> | 
|  | /// Reference to the internal package part | 
|  | /// </summary> | 
|  | internal Packaging.ZipPackagePart Part | 
|  | { | 
|  | get; | 
|  | set; | 
|  | } | 
|  | /// <summary> | 
|  | /// Provides access to the XML data representing the cache definition in the package. | 
|  | /// </summary> | 
|  | public XmlDocument CacheDefinitionXml { get; private set; } | 
|  | /// <summary> | 
|  | /// The package internal URI to the pivottable cache definition Xml Document. | 
|  | /// </summary> | 
|  | public Uri CacheDefinitionUri | 
|  | { | 
|  | get; | 
|  | internal set; | 
|  | } | 
|  | internal Uri CacheRecordUri | 
|  | { | 
|  | get; | 
|  | set; | 
|  | } | 
|  | internal Packaging.ZipPackageRelationship Relationship | 
|  | { | 
|  | get; | 
|  | set; | 
|  | } | 
|  | internal Packaging.ZipPackageRelationship RecordRelationship | 
|  | { | 
|  | get; | 
|  | set; | 
|  | } | 
|  | internal string RecordRelationshipID | 
|  | { | 
|  | get | 
|  | { | 
|  | return GetXmlNodeString("@r:id"); | 
|  | } | 
|  | set | 
|  | { | 
|  | SetXmlNodeString("@r:id", value); | 
|  | } | 
|  | } | 
|  | /// <summary> | 
|  | /// Referece to the PivoTable object | 
|  | /// </summary> | 
|  | public ExcelPivotTable PivotTable | 
|  | { | 
|  | get; | 
|  | private set; | 
|  | } | 
|  |  | 
|  | const string _sourceWorksheetPath="d:cacheSource/d:worksheetSource/@sheet"; | 
|  | const string _sourceNamePath = "d:cacheSource/d:worksheetSource/@name"; | 
|  | const string _sourceAddressPath = "d:cacheSource/d:worksheetSource/@ref"; | 
|  | internal ExcelRangeBase _sourceRange = null; | 
|  | /// <summary> | 
|  | /// 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. | 
|  | /// </summary> | 
|  | public ExcelRangeBase SourceRange | 
|  | { | 
|  | get | 
|  | { | 
|  | if (_sourceRange == null) | 
|  | { | 
|  | if (CacheSource == eSourceType.Worksheet) | 
|  | { | 
|  | var ws = PivotTable.WorkSheet.Workbook.Worksheets[GetXmlNodeString(_sourceWorksheetPath)]; | 
|  | if (ws == null) //Not worksheet, check name or table name | 
|  | { | 
|  | var name = GetXmlNodeString(_sourceNamePath); | 
|  | foreach (var n in PivotTable.WorkSheet.Workbook.Names) | 
|  | { | 
|  | if(name.Equals(n.Name,StringComparison.InvariantCultureIgnoreCase)) | 
|  | { | 
|  | _sourceRange = n; | 
|  | return _sourceRange; | 
|  | } | 
|  | } | 
|  | foreach (var w in PivotTable.WorkSheet.Workbook.Worksheets) | 
|  | { | 
|  | if (w.Tables._tableNames.ContainsKey(name)) | 
|  | { | 
|  | _sourceRange = w.Cells[w.Tables[name].Address.Address]; | 
|  | break; | 
|  | } | 
|  | foreach (var n in w.Names) | 
|  | { | 
|  | if (name.Equals(n.Name, StringComparison.InvariantCultureIgnoreCase)) | 
|  | { | 
|  | _sourceRange = n; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | else | 
|  | { | 
|  | _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; | 
|  | } | 
|  | } | 
|  | /// <summary> | 
|  | /// Type of source data | 
|  | /// </summary> | 
|  | 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="<pivotCacheDefinition xmlns=\"http://schemas.openxmlformats.org/spreadsheetml/2006/main\" xmlns:r=\"http://schemas.openxmlformats.org/officeDocument/2006/relationships\" r:id=\"\" refreshOnLoad=\"1\" refreshedBy=\"SomeUser\" refreshedDate=\"40504.582403125001\" createdVersion=\"1\" refreshedVersion=\"3\" recordCount=\"5\" upgradeOnRefresh=\"1\">"; | 
|  |  | 
|  | xml += "<cacheSource type=\"worksheet\">"; | 
|  | xml += string.Format("<worksheetSource ref=\"{0}\" sheet=\"{1}\" /> ", sourceAddress.Address, sourceAddress.WorkSheet); | 
|  | xml += "</cacheSource>"; | 
|  | xml += string.Format("<cacheFields count=\"{0}\">", 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._values.GetValue(sourceAddress._fromRow, col) == null || sourceWorksheet._values.GetValue(sourceAddress._fromRow, col).ToString().Trim() == "") | 
|  | { | 
|  | xml += string.Format("<cacheField name=\"Column{0}\" numFmtId=\"0\">", col - sourceAddress._fromCol + 1); | 
|  | } | 
|  | else | 
|  | { | 
|  | xml += string.Format("<cacheField name=\"{0}\" numFmtId=\"0\">", sourceWorksheet._values.GetValue(sourceAddress._fromRow, col)); | 
|  | } | 
|  | //xml += "<sharedItems containsNonDate=\"0\" containsString=\"0\" containsBlank=\"1\" /> "; | 
|  | xml += "<sharedItems containsBlank=\"1\" /> "; | 
|  | xml += "</cacheField>"; | 
|  | } | 
|  | xml += "</cacheFields>"; | 
|  | xml += "</pivotCacheDefinition>"; | 
|  |  | 
|  | return xml; | 
|  | } | 
|  | } | 
|  | } |